@rindo/core 2.16.0-beta.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +27 -0
- package/bin/rindo +57 -0
- package/cli/index.cjs +1784 -0
- package/cli/index.d.ts +15 -0
- package/cli/index.js +1758 -0
- package/cli/package.json +14 -0
- package/compiler/lib.d.ts +24 -0
- package/compiler/lib.dom.d.ts +17791 -0
- package/compiler/lib.dom.iterable.d.ts +323 -0
- package/compiler/lib.es2015.collection.d.ts +89 -0
- package/compiler/lib.es2015.core.d.ts +559 -0
- package/compiler/lib.es2015.d.ts +30 -0
- package/compiler/lib.es2015.generator.d.ts +79 -0
- package/compiler/lib.es2015.iterable.d.ts +497 -0
- package/compiler/lib.es2015.promise.d.ts +78 -0
- package/compiler/lib.es2015.proxy.d.ts +41 -0
- package/compiler/lib.es2015.reflect.d.ts +123 -0
- package/compiler/lib.es2015.symbol.d.ts +48 -0
- package/compiler/lib.es2015.symbol.wellknown.d.ts +324 -0
- package/compiler/lib.es2016.array.include.d.ts +118 -0
- package/compiler/lib.es2016.d.ts +22 -0
- package/compiler/lib.es2016.full.d.ts +25 -0
- package/compiler/lib.es2017.d.ts +26 -0
- package/compiler/lib.es2017.full.d.ts +25 -0
- package/compiler/lib.es2017.intl.d.ts +32 -0
- package/compiler/lib.es2017.object.d.ts +51 -0
- package/compiler/lib.es2017.sharedmemory.d.ts +137 -0
- package/compiler/lib.es2017.string.d.ts +47 -0
- package/compiler/lib.es2017.typedarrays.d.ts +55 -0
- package/compiler/lib.es2018.asyncgenerator.d.ts +79 -0
- package/compiler/lib.es2018.asynciterable.d.ts +45 -0
- package/compiler/lib.es2018.d.ts +26 -0
- package/compiler/lib.es2018.full.d.ts +25 -0
- package/compiler/lib.es2018.intl.d.ts +73 -0
- package/compiler/lib.es2018.promise.d.ts +32 -0
- package/compiler/lib.es2018.regexp.d.ts +39 -0
- package/compiler/lib.es2019.array.d.ts +85 -0
- package/compiler/lib.es2019.d.ts +25 -0
- package/compiler/lib.es2019.full.d.ts +25 -0
- package/compiler/lib.es2019.object.d.ts +35 -0
- package/compiler/lib.es2019.string.d.ts +39 -0
- package/compiler/lib.es2019.symbol.d.ts +26 -0
- package/compiler/lib.es2020.bigint.d.ts +728 -0
- package/compiler/lib.es2020.d.ts +27 -0
- package/compiler/lib.es2020.full.d.ts +25 -0
- package/compiler/lib.es2020.intl.d.ts +368 -0
- package/compiler/lib.es2020.promise.d.ts +49 -0
- package/compiler/lib.es2020.sharedmemory.d.ts +99 -0
- package/compiler/lib.es2020.string.d.ts +30 -0
- package/compiler/lib.es2020.symbol.wellknown.d.ts +39 -0
- package/compiler/lib.es2021.d.ts +25 -0
- package/compiler/lib.es2021.full.d.ts +25 -0
- package/compiler/lib.es2021.intl.d.ts +44 -0
- package/compiler/lib.es2021.promise.d.ts +50 -0
- package/compiler/lib.es2021.string.d.ts +35 -0
- package/compiler/lib.es2021.weakref.d.ts +75 -0
- package/compiler/lib.es5.d.ts +4495 -0
- package/compiler/lib.es6.d.ts +25 -0
- package/compiler/lib.esnext.d.ts +22 -0
- package/compiler/lib.esnext.full.d.ts +25 -0
- package/compiler/lib.esnext.intl.d.ts +23 -0
- package/compiler/lib.esnext.promise.d.ts +43 -0
- package/compiler/lib.esnext.string.d.ts +35 -0
- package/compiler/lib.esnext.weakref.d.ts +75 -0
- package/compiler/lib.scripthost.d.ts +327 -0
- package/compiler/lib.webworker.d.ts +5733 -0
- package/compiler/lib.webworker.importscripts.d.ts +26 -0
- package/compiler/lib.webworker.iterable.d.ts +160 -0
- package/compiler/package.json +8 -0
- package/compiler/rindo.d.ts +95 -0
- package/compiler/rindo.js +67222 -0
- package/compiler/rindo.min.js +4 -0
- package/dependencies.json +109 -0
- package/dev-server/client/app-error.d.ts +18 -0
- package/dev-server/client/events.d.ts +6 -0
- package/dev-server/client/hmr-components.d.ts +1 -0
- package/dev-server/client/hmr-external-styles.d.ts +1 -0
- package/dev-server/client/hmr-images.d.ts +1 -0
- package/dev-server/client/hmr-inline-styles.d.ts +1 -0
- package/dev-server/client/hmr-util.d.ts +9 -0
- package/dev-server/client/hmr-window.d.ts +10 -0
- package/dev-server/client/index.d.ts +6 -0
- package/dev-server/client/index.js +808 -0
- package/dev-server/client/logger.d.ts +5 -0
- package/dev-server/client/package.json +8 -0
- package/dev-server/client/progress.d.ts +3 -0
- package/dev-server/client/status.d.ts +4 -0
- package/dev-server/connector.html +6 -0
- package/dev-server/index.d.ts +3 -0
- package/dev-server/index.js +264 -0
- package/dev-server/open-in-editor-api.js +1 -0
- package/dev-server/package.json +8 -0
- package/dev-server/server-process.js +1763 -0
- package/dev-server/server-worker-thread.js +39 -0
- package/dev-server/static/favicon.ico +0 -0
- package/dev-server/templates/directory-index.html +132 -0
- package/dev-server/templates/initial-load.html +160 -0
- package/dev-server/visualstudio.vbs +82 -0
- package/dev-server/ws.js +1 -0
- package/dev-server/xdg-open +1066 -0
- package/internal/app-data/index.cjs +92 -0
- package/internal/app-data/index.d.ts +4 -0
- package/internal/app-data/index.js +88 -0
- package/internal/app-data/package.json +15 -0
- package/internal/client/css-shim.js +4 -0
- package/internal/client/dom.js +73 -0
- package/internal/client/index.js +3059 -0
- package/internal/client/package.json +8 -0
- package/internal/client/patch-browser.js +124 -0
- package/internal/client/patch-esm.js +23 -0
- package/internal/client/polyfills/core-js.js +11 -0
- package/internal/client/polyfills/css-shim.js +1 -0
- package/internal/client/polyfills/dom.js +79 -0
- package/internal/client/polyfills/es5-html-element.js +1 -0
- package/internal/client/polyfills/index.js +34 -0
- package/internal/client/polyfills/system.js +6 -0
- package/internal/client/shadow-css.js +387 -0
- package/internal/hydrate/index.js +1132 -0
- package/internal/hydrate/package.json +7 -0
- package/internal/hydrate/runner.d.ts +217 -0
- package/internal/hydrate/runner.js +777 -0
- package/internal/hydrate/shadow-css.js +143 -0
- package/internal/index.d.ts +4 -0
- package/internal/index.js +2 -0
- package/internal/package.json +9 -0
- package/internal/rindo-core/index.cjs +1 -0
- package/internal/rindo-core/index.d.ts +52 -0
- package/internal/rindo-core/index.js +16 -0
- package/internal/rindo-ext-modules.d.ts +41 -0
- package/internal/rindo-private.d.ts +2289 -0
- package/internal/rindo-public-compiler.d.ts +2273 -0
- package/internal/rindo-public-docs.d.ts +116 -0
- package/internal/rindo-public-runtime.d.ts +1565 -0
- package/internal/testing/index.js +1093 -0
- package/internal/testing/package.json +7 -0
- package/internal/testing/shadow-css.js +143 -0
- package/mock-doc/index.cjs +4658 -0
- package/mock-doc/index.d.ts +928 -0
- package/mock-doc/index.js +4622 -0
- package/mock-doc/package.json +15 -0
- package/package.json +151 -0
- package/readme.md +74 -0
- package/screenshot/compare/assets/favicon.ico +0 -0
- package/screenshot/compare/assets/logo.png +0 -0
- package/screenshot/compare/build/app.css +1 -0
- package/screenshot/compare/build/app.esm.js +1 -0
- package/screenshot/compare/build/app.js +33 -0
- package/screenshot/compare/build/index.esm.js +0 -0
- package/screenshot/compare/build/p-081b0641.js +1 -0
- package/screenshot/compare/build/p-227a1e18.entry.js +1 -0
- package/screenshot/compare/build/p-2c298727.entry.js +1 -0
- package/screenshot/compare/build/p-5479268c.entry.js +1 -0
- package/screenshot/compare/build/p-573ec8a4.entry.js +1 -0
- package/screenshot/compare/build/p-6ba08604.entry.js +1 -0
- package/screenshot/compare/build/p-6bc63295.entry.js +1 -0
- package/screenshot/compare/build/p-7a3759fd.entry.js +1 -0
- package/screenshot/compare/build/p-7b4e3ba7.js +1 -0
- package/screenshot/compare/build/p-988eb362.css +1 -0
- package/screenshot/compare/build/p-9b6a9315.js +1 -0
- package/screenshot/compare/build/p-b4cc611c.entry.js +1 -0
- package/screenshot/compare/build/p-d1bf53f5.entry.js +1 -0
- package/screenshot/compare/build/p-e2efe0df.js +1 -0
- package/screenshot/compare/build/p-e8ca6d97.entry.js +1 -0
- package/screenshot/compare/build/p-ec2f13e0.entry.js +1 -0
- package/screenshot/compare/build/p-f0b99977.entry.js +1 -0
- package/screenshot/compare/build/p-f4745c2f.entry.js +1 -0
- package/screenshot/compare/build/p-fbbae598.js +1 -0
- package/screenshot/compare/host.config.json +15 -0
- package/screenshot/compare/index.html +1 -0
- package/screenshot/compare/manifest.json +13 -0
- package/screenshot/connector-base.d.ts +42 -0
- package/screenshot/connector-local.d.ts +7 -0
- package/screenshot/connector.js +2 -0
- package/screenshot/index.d.ts +3 -0
- package/screenshot/index.js +659 -0
- package/screenshot/local-connector.js +2 -0
- package/screenshot/package.json +15 -0
- package/screenshot/pixel-match.d.ts +1 -0
- package/screenshot/pixel-match.js +2673 -0
- package/screenshot/screenshot-compare.d.ts +3 -0
- package/screenshot/screenshot-fs.d.ts +15 -0
- package/sys/node/autoprefixer.js +8 -0
- package/sys/node/glob.js +1 -0
- package/sys/node/graceful-fs.js +1 -0
- package/sys/node/index.d.ts +18 -0
- package/sys/node/index.js +6124 -0
- package/sys/node/node-fetch.js +1 -0
- package/sys/node/package.json +8 -0
- package/sys/node/prompts.js +1 -0
- package/sys/node/worker.js +52 -0
- package/testing/index.d.ts +12 -0
- package/testing/index.js +4232 -0
- package/testing/jest/jest-config.d.ts +16 -0
- package/testing/jest/jest-environment.d.ts +15 -0
- package/testing/jest/jest-preprocessor.d.ts +59 -0
- package/testing/jest/jest-runner.d.ts +9 -0
- package/testing/jest/jest-screenshot.d.ts +2 -0
- package/testing/jest/jest-serializer.d.ts +5 -0
- package/testing/jest/jest-setup-test-framework.d.ts +1 -0
- package/testing/jest-environment.js +3 -0
- package/testing/jest-preprocessor.js +3 -0
- package/testing/jest-preset.js +37 -0
- package/testing/jest-runner.js +3 -0
- package/testing/jest-setuptestframework.js +3 -0
- package/testing/matchers/attributes.d.ts +14 -0
- package/testing/matchers/class-list.d.ts +12 -0
- package/testing/matchers/events.d.ts +21 -0
- package/testing/matchers/html.d.ts +12 -0
- package/testing/matchers/index.d.ts +23 -0
- package/testing/matchers/screenshot.d.ts +5 -0
- package/testing/matchers/text.d.ts +4 -0
- package/testing/mock-fetch.d.ts +11 -0
- package/testing/mocks.d.ts +10 -0
- package/testing/package.json +8 -0
- package/testing/puppeteer/index.d.ts +2 -0
- package/testing/puppeteer/puppeteer-browser.d.ts +6 -0
- package/testing/puppeteer/puppeteer-declarations.d.ts +429 -0
- package/testing/puppeteer/puppeteer-element.d.ts +67 -0
- package/testing/puppeteer/puppeteer-emulate.d.ts +2 -0
- package/testing/puppeteer/puppeteer-events.d.ts +21 -0
- package/testing/puppeteer/puppeteer-page.d.ts +2 -0
- package/testing/puppeteer/puppeteer-screenshot.d.ts +4 -0
- package/testing/reset-build-conditionals.d.ts +2 -0
- package/testing/spec-page.d.ts +2 -0
- package/testing/test-transpile.d.ts +2 -0
- package/testing/testing-logger.d.ts +25 -0
- package/testing/testing-sys.d.ts +6 -0
- package/testing/testing-utils.d.ts +6 -0
- package/testing/testing.d.ts +2 -0
|
@@ -0,0 +1,659 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const fs = require('fs');
|
|
4
|
+
const path = require('path');
|
|
5
|
+
const os = require('os');
|
|
6
|
+
|
|
7
|
+
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
|
|
8
|
+
|
|
9
|
+
const fs__default = /*#__PURE__*/_interopDefaultLegacy(fs);
|
|
10
|
+
const path__default = /*#__PURE__*/_interopDefaultLegacy(path);
|
|
11
|
+
|
|
12
|
+
function fileExists(filePath) {
|
|
13
|
+
return new Promise((resolve) => {
|
|
14
|
+
fs__default['default'].access(filePath, (err) => resolve(!err));
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
function readFile(filePath) {
|
|
18
|
+
return new Promise((resolve, reject) => {
|
|
19
|
+
fs__default['default'].readFile(filePath, 'utf-8', (err, data) => {
|
|
20
|
+
if (err) {
|
|
21
|
+
reject(err);
|
|
22
|
+
}
|
|
23
|
+
else {
|
|
24
|
+
resolve(data);
|
|
25
|
+
}
|
|
26
|
+
});
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
function readFileBuffer(filePath) {
|
|
30
|
+
return new Promise((resolve, reject) => {
|
|
31
|
+
fs__default['default'].readFile(filePath, (err, data) => {
|
|
32
|
+
if (err) {
|
|
33
|
+
reject(err);
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
resolve(data);
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
function writeFile(filePath, data) {
|
|
42
|
+
return new Promise((resolve, reject) => {
|
|
43
|
+
fs__default['default'].writeFile(filePath, data, (err) => {
|
|
44
|
+
if (err) {
|
|
45
|
+
reject(err);
|
|
46
|
+
}
|
|
47
|
+
else {
|
|
48
|
+
resolve();
|
|
49
|
+
}
|
|
50
|
+
});
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
function mkDir(filePath) {
|
|
54
|
+
return new Promise((resolve) => {
|
|
55
|
+
fs__default['default'].mkdir(filePath, () => {
|
|
56
|
+
resolve();
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
function rmDir(filePath) {
|
|
61
|
+
return new Promise((resolve) => {
|
|
62
|
+
fs__default['default'].rmdir(filePath, () => {
|
|
63
|
+
resolve();
|
|
64
|
+
});
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
async function emptyDir(dir) {
|
|
68
|
+
const files = await readDir(dir);
|
|
69
|
+
const promises = files.map(async (fileName) => {
|
|
70
|
+
const filePath = path__default['default'].join(dir, fileName);
|
|
71
|
+
const isDirFile = await isFile(filePath);
|
|
72
|
+
if (isDirFile) {
|
|
73
|
+
await unlink(filePath);
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
await Promise.all(promises);
|
|
77
|
+
}
|
|
78
|
+
async function readDir(dir) {
|
|
79
|
+
return new Promise((resolve) => {
|
|
80
|
+
fs__default['default'].readdir(dir, (err, files) => {
|
|
81
|
+
if (err) {
|
|
82
|
+
resolve([]);
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
resolve(files);
|
|
86
|
+
}
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
async function isFile(itemPath) {
|
|
91
|
+
return new Promise((resolve) => {
|
|
92
|
+
fs__default['default'].stat(itemPath, (err, stat) => {
|
|
93
|
+
if (err) {
|
|
94
|
+
resolve(false);
|
|
95
|
+
}
|
|
96
|
+
else {
|
|
97
|
+
resolve(stat.isFile());
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
async function unlink(filePath) {
|
|
103
|
+
return new Promise((resolve) => {
|
|
104
|
+
fs__default['default'].unlink(filePath, () => {
|
|
105
|
+
resolve();
|
|
106
|
+
});
|
|
107
|
+
});
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
class ScreenshotConnector {
|
|
111
|
+
constructor() {
|
|
112
|
+
this.screenshotDirName = 'screenshot';
|
|
113
|
+
this.imagesDirName = 'images';
|
|
114
|
+
this.buildsDirName = 'builds';
|
|
115
|
+
this.masterBuildFileName = 'master.json';
|
|
116
|
+
this.screenshotCacheFileName = 'screenshot-cache.json';
|
|
117
|
+
}
|
|
118
|
+
async initBuild(opts) {
|
|
119
|
+
this.logger = opts.logger;
|
|
120
|
+
this.buildId = opts.buildId;
|
|
121
|
+
this.buildMessage = opts.buildMessage || '';
|
|
122
|
+
this.buildAuthor = opts.buildAuthor;
|
|
123
|
+
this.buildUrl = opts.buildUrl;
|
|
124
|
+
this.previewUrl = opts.previewUrl;
|
|
125
|
+
(this.buildTimestamp = typeof opts.buildTimestamp === 'number' ? opts.buildTimestamp : Date.now()),
|
|
126
|
+
(this.cacheDir = opts.cacheDir);
|
|
127
|
+
this.packageDir = opts.packageDir;
|
|
128
|
+
this.rootDir = opts.rootDir;
|
|
129
|
+
this.appNamespace = opts.appNamespace;
|
|
130
|
+
this.waitBeforeScreenshot = opts.waitBeforeScreenshot;
|
|
131
|
+
this.pixelmatchModulePath = opts.pixelmatchModulePath;
|
|
132
|
+
if (!opts.logger) {
|
|
133
|
+
throw new Error(`logger option required`);
|
|
134
|
+
}
|
|
135
|
+
if (typeof opts.buildId !== 'string') {
|
|
136
|
+
throw new Error(`buildId option required`);
|
|
137
|
+
}
|
|
138
|
+
if (typeof opts.cacheDir !== 'string') {
|
|
139
|
+
throw new Error(`cacheDir option required`);
|
|
140
|
+
}
|
|
141
|
+
if (typeof opts.packageDir !== 'string') {
|
|
142
|
+
throw new Error(`packageDir option required`);
|
|
143
|
+
}
|
|
144
|
+
if (typeof opts.rootDir !== 'string') {
|
|
145
|
+
throw new Error(`rootDir option required`);
|
|
146
|
+
}
|
|
147
|
+
this.updateMaster = !!opts.updateMaster;
|
|
148
|
+
this.allowableMismatchedPixels = opts.allowableMismatchedPixels;
|
|
149
|
+
this.allowableMismatchedRatio = opts.allowableMismatchedRatio;
|
|
150
|
+
this.pixelmatchThreshold = opts.pixelmatchThreshold;
|
|
151
|
+
this.logger.debug(`screenshot build: ${this.buildId}, ${this.buildMessage}, updateMaster: ${this.updateMaster}`);
|
|
152
|
+
this.logger.debug(`screenshot, allowableMismatchedPixels: ${this.allowableMismatchedPixels}, allowableMismatchedRatio: ${this.allowableMismatchedRatio}, pixelmatchThreshold: ${this.pixelmatchThreshold}`);
|
|
153
|
+
if (typeof opts.screenshotDirName === 'string') {
|
|
154
|
+
this.screenshotDirName = opts.screenshotDirName;
|
|
155
|
+
}
|
|
156
|
+
if (typeof opts.imagesDirName === 'string') {
|
|
157
|
+
this.imagesDirName = opts.imagesDirName;
|
|
158
|
+
}
|
|
159
|
+
if (typeof opts.buildsDirName === 'string') {
|
|
160
|
+
this.buildsDirName = opts.buildsDirName;
|
|
161
|
+
}
|
|
162
|
+
this.screenshotDir = path.join(this.rootDir, this.screenshotDirName);
|
|
163
|
+
this.imagesDir = path.join(this.screenshotDir, this.imagesDirName);
|
|
164
|
+
this.buildsDir = path.join(this.screenshotDir, this.buildsDirName);
|
|
165
|
+
this.masterBuildFilePath = path.join(this.buildsDir, this.masterBuildFileName);
|
|
166
|
+
this.screenshotCacheFilePath = path.join(this.cacheDir, this.screenshotCacheFileName);
|
|
167
|
+
this.currentBuildDir = path.join(os.tmpdir(), 'screenshot-build-' + this.buildId);
|
|
168
|
+
this.logger.debug(`screenshotDirPath: ${this.screenshotDir}`);
|
|
169
|
+
this.logger.debug(`imagesDirPath: ${this.imagesDir}`);
|
|
170
|
+
this.logger.debug(`buildsDirPath: ${this.buildsDir}`);
|
|
171
|
+
this.logger.debug(`currentBuildDir: ${this.currentBuildDir}`);
|
|
172
|
+
this.logger.debug(`cacheDir: ${this.cacheDir}`);
|
|
173
|
+
await mkDir(this.screenshotDir);
|
|
174
|
+
await Promise.all([
|
|
175
|
+
mkDir(this.imagesDir),
|
|
176
|
+
mkDir(this.buildsDir),
|
|
177
|
+
mkDir(this.currentBuildDir),
|
|
178
|
+
mkDir(this.cacheDir),
|
|
179
|
+
]);
|
|
180
|
+
}
|
|
181
|
+
async pullMasterBuild() {
|
|
182
|
+
/**/
|
|
183
|
+
}
|
|
184
|
+
async getMasterBuild() {
|
|
185
|
+
let masterBuild = null;
|
|
186
|
+
try {
|
|
187
|
+
masterBuild = JSON.parse(await readFile(this.masterBuildFilePath));
|
|
188
|
+
}
|
|
189
|
+
catch (e) { }
|
|
190
|
+
return masterBuild;
|
|
191
|
+
}
|
|
192
|
+
async completeBuild(masterBuild) {
|
|
193
|
+
const filePaths = (await readDir(this.currentBuildDir))
|
|
194
|
+
.map((f) => path.join(this.currentBuildDir, f))
|
|
195
|
+
.filter((f) => f.endsWith('.json'));
|
|
196
|
+
const screenshots = await Promise.all(filePaths.map(async (f) => JSON.parse(await readFile(f))));
|
|
197
|
+
this.sortScreenshots(screenshots);
|
|
198
|
+
if (!masterBuild) {
|
|
199
|
+
masterBuild = {
|
|
200
|
+
id: this.buildId,
|
|
201
|
+
message: this.buildMessage,
|
|
202
|
+
author: this.buildAuthor,
|
|
203
|
+
url: this.buildUrl,
|
|
204
|
+
previewUrl: this.previewUrl,
|
|
205
|
+
appNamespace: this.appNamespace,
|
|
206
|
+
timestamp: this.buildTimestamp,
|
|
207
|
+
screenshots: screenshots,
|
|
208
|
+
};
|
|
209
|
+
}
|
|
210
|
+
const results = {
|
|
211
|
+
appNamespace: this.appNamespace,
|
|
212
|
+
masterBuild: masterBuild,
|
|
213
|
+
currentBuild: {
|
|
214
|
+
id: this.buildId,
|
|
215
|
+
message: this.buildMessage,
|
|
216
|
+
author: this.buildAuthor,
|
|
217
|
+
url: this.buildUrl,
|
|
218
|
+
previewUrl: this.previewUrl,
|
|
219
|
+
appNamespace: this.appNamespace,
|
|
220
|
+
timestamp: this.buildTimestamp,
|
|
221
|
+
screenshots: screenshots,
|
|
222
|
+
},
|
|
223
|
+
compare: {
|
|
224
|
+
id: `${masterBuild.id}-${this.buildId}`,
|
|
225
|
+
a: {
|
|
226
|
+
id: masterBuild.id,
|
|
227
|
+
message: masterBuild.message,
|
|
228
|
+
author: masterBuild.author,
|
|
229
|
+
url: masterBuild.url,
|
|
230
|
+
previewUrl: masterBuild.previewUrl,
|
|
231
|
+
},
|
|
232
|
+
b: {
|
|
233
|
+
id: this.buildId,
|
|
234
|
+
message: this.buildMessage,
|
|
235
|
+
author: this.buildAuthor,
|
|
236
|
+
url: this.buildUrl,
|
|
237
|
+
previewUrl: this.previewUrl,
|
|
238
|
+
},
|
|
239
|
+
url: null,
|
|
240
|
+
appNamespace: this.appNamespace,
|
|
241
|
+
timestamp: this.buildTimestamp,
|
|
242
|
+
diffs: [],
|
|
243
|
+
},
|
|
244
|
+
};
|
|
245
|
+
results.currentBuild.screenshots.forEach((screenshot) => {
|
|
246
|
+
screenshot.diff.device = screenshot.diff.device || screenshot.diff.userAgent;
|
|
247
|
+
results.compare.diffs.push(screenshot.diff);
|
|
248
|
+
delete screenshot.diff;
|
|
249
|
+
});
|
|
250
|
+
this.sortCompares(results.compare.diffs);
|
|
251
|
+
await emptyDir(this.currentBuildDir);
|
|
252
|
+
await rmDir(this.currentBuildDir);
|
|
253
|
+
return results;
|
|
254
|
+
}
|
|
255
|
+
async publishBuild(results) {
|
|
256
|
+
return results;
|
|
257
|
+
}
|
|
258
|
+
async generateJsonpDataUris(build) {
|
|
259
|
+
if (build && Array.isArray(build.screenshots)) {
|
|
260
|
+
for (let i = 0; i < build.screenshots.length; i++) {
|
|
261
|
+
const screenshot = build.screenshots[i];
|
|
262
|
+
const jsonpFileName = `screenshot_${screenshot.image}.js`;
|
|
263
|
+
const jsonFilePath = path.join(this.cacheDir, jsonpFileName);
|
|
264
|
+
const jsonpExists = await fileExists(jsonFilePath);
|
|
265
|
+
if (!jsonpExists) {
|
|
266
|
+
const imageFilePath = path.join(this.imagesDir, screenshot.image);
|
|
267
|
+
const imageBuf = await readFileBuffer(imageFilePath);
|
|
268
|
+
const jsonpContent = `loadScreenshot("${screenshot.image}","data:image/png;base64,${imageBuf.toString('base64')}");`;
|
|
269
|
+
await writeFile(jsonFilePath, jsonpContent);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
async getScreenshotCache() {
|
|
275
|
+
return null;
|
|
276
|
+
}
|
|
277
|
+
async updateScreenshotCache(screenshotCache, buildResults) {
|
|
278
|
+
screenshotCache = screenshotCache || {};
|
|
279
|
+
screenshotCache.timestamp = this.buildTimestamp;
|
|
280
|
+
screenshotCache.lastBuildId = this.buildId;
|
|
281
|
+
screenshotCache.size = 0;
|
|
282
|
+
screenshotCache.items = screenshotCache.items || [];
|
|
283
|
+
if (buildResults && buildResults.compare && Array.isArray(buildResults.compare.diffs)) {
|
|
284
|
+
buildResults.compare.diffs.forEach((diff) => {
|
|
285
|
+
if (typeof diff.cacheKey !== 'string') {
|
|
286
|
+
return;
|
|
287
|
+
}
|
|
288
|
+
if (diff.imageA === diff.imageB) {
|
|
289
|
+
// no need to cache identical matches
|
|
290
|
+
return;
|
|
291
|
+
}
|
|
292
|
+
const existingItem = screenshotCache.items.find((i) => i.key === diff.cacheKey);
|
|
293
|
+
if (existingItem) {
|
|
294
|
+
// already have this cached, but update its timestamp
|
|
295
|
+
existingItem.ts = this.buildTimestamp;
|
|
296
|
+
}
|
|
297
|
+
else {
|
|
298
|
+
// add this item to the cache
|
|
299
|
+
screenshotCache.items.push({
|
|
300
|
+
key: diff.cacheKey,
|
|
301
|
+
ts: this.buildTimestamp,
|
|
302
|
+
mp: diff.mismatchedPixels,
|
|
303
|
+
});
|
|
304
|
+
}
|
|
305
|
+
});
|
|
306
|
+
}
|
|
307
|
+
// sort so the newest items are on top
|
|
308
|
+
screenshotCache.items.sort((a, b) => {
|
|
309
|
+
if (a.ts > b.ts)
|
|
310
|
+
return -1;
|
|
311
|
+
if (a.ts < b.ts)
|
|
312
|
+
return 1;
|
|
313
|
+
if (a.mp > b.mp)
|
|
314
|
+
return -1;
|
|
315
|
+
if (a.mp < b.mp)
|
|
316
|
+
return 1;
|
|
317
|
+
return 0;
|
|
318
|
+
});
|
|
319
|
+
// keep only the most recent items
|
|
320
|
+
screenshotCache.items = screenshotCache.items.slice(0, 1000);
|
|
321
|
+
screenshotCache.size = screenshotCache.items.length;
|
|
322
|
+
return screenshotCache;
|
|
323
|
+
}
|
|
324
|
+
toJson(masterBuild, screenshotCache) {
|
|
325
|
+
const masterScreenshots = {};
|
|
326
|
+
if (masterBuild && Array.isArray(masterBuild.screenshots)) {
|
|
327
|
+
masterBuild.screenshots.forEach((masterScreenshot) => {
|
|
328
|
+
masterScreenshots[masterScreenshot.id] = masterScreenshot.image;
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
const mismatchCache = {};
|
|
332
|
+
if (screenshotCache && Array.isArray(screenshotCache.items)) {
|
|
333
|
+
screenshotCache.items.forEach((cacheItem) => {
|
|
334
|
+
mismatchCache[cacheItem.key] = cacheItem.mp;
|
|
335
|
+
});
|
|
336
|
+
}
|
|
337
|
+
const screenshotBuild = {
|
|
338
|
+
buildId: this.buildId,
|
|
339
|
+
rootDir: this.rootDir,
|
|
340
|
+
screenshotDir: this.screenshotDir,
|
|
341
|
+
imagesDir: this.imagesDir,
|
|
342
|
+
buildsDir: this.buildsDir,
|
|
343
|
+
masterScreenshots: masterScreenshots,
|
|
344
|
+
cache: mismatchCache,
|
|
345
|
+
currentBuildDir: this.currentBuildDir,
|
|
346
|
+
updateMaster: this.updateMaster,
|
|
347
|
+
allowableMismatchedPixels: this.allowableMismatchedPixels,
|
|
348
|
+
allowableMismatchedRatio: this.allowableMismatchedRatio,
|
|
349
|
+
pixelmatchThreshold: this.pixelmatchThreshold,
|
|
350
|
+
timeoutBeforeScreenshot: this.waitBeforeScreenshot,
|
|
351
|
+
pixelmatchModulePath: this.pixelmatchModulePath,
|
|
352
|
+
};
|
|
353
|
+
return JSON.stringify(screenshotBuild);
|
|
354
|
+
}
|
|
355
|
+
sortScreenshots(screenshots) {
|
|
356
|
+
return screenshots.sort((a, b) => {
|
|
357
|
+
if (a.desc && b.desc) {
|
|
358
|
+
if (a.desc.toLowerCase() < b.desc.toLowerCase())
|
|
359
|
+
return -1;
|
|
360
|
+
if (a.desc.toLowerCase() > b.desc.toLowerCase())
|
|
361
|
+
return 1;
|
|
362
|
+
}
|
|
363
|
+
if (a.device && b.device) {
|
|
364
|
+
if (a.device.toLowerCase() < b.device.toLowerCase())
|
|
365
|
+
return -1;
|
|
366
|
+
if (a.device.toLowerCase() > b.device.toLowerCase())
|
|
367
|
+
return 1;
|
|
368
|
+
}
|
|
369
|
+
if (a.userAgent && b.userAgent) {
|
|
370
|
+
if (a.userAgent.toLowerCase() < b.userAgent.toLowerCase())
|
|
371
|
+
return -1;
|
|
372
|
+
if (a.userAgent.toLowerCase() > b.userAgent.toLowerCase())
|
|
373
|
+
return 1;
|
|
374
|
+
}
|
|
375
|
+
if (a.width < b.width)
|
|
376
|
+
return -1;
|
|
377
|
+
if (a.width > b.width)
|
|
378
|
+
return 1;
|
|
379
|
+
if (a.height < b.height)
|
|
380
|
+
return -1;
|
|
381
|
+
if (a.height > b.height)
|
|
382
|
+
return 1;
|
|
383
|
+
if (a.id < b.id)
|
|
384
|
+
return -1;
|
|
385
|
+
if (a.id > b.id)
|
|
386
|
+
return 1;
|
|
387
|
+
return 0;
|
|
388
|
+
});
|
|
389
|
+
}
|
|
390
|
+
sortCompares(compares) {
|
|
391
|
+
return compares.sort((a, b) => {
|
|
392
|
+
if (a.allowableMismatchedPixels > b.allowableMismatchedPixels)
|
|
393
|
+
return -1;
|
|
394
|
+
if (a.allowableMismatchedPixels < b.allowableMismatchedPixels)
|
|
395
|
+
return 1;
|
|
396
|
+
if (a.allowableMismatchedRatio > b.allowableMismatchedRatio)
|
|
397
|
+
return -1;
|
|
398
|
+
if (a.allowableMismatchedRatio < b.allowableMismatchedRatio)
|
|
399
|
+
return 1;
|
|
400
|
+
if (a.desc && b.desc) {
|
|
401
|
+
if (a.desc.toLowerCase() < b.desc.toLowerCase())
|
|
402
|
+
return -1;
|
|
403
|
+
if (a.desc.toLowerCase() > b.desc.toLowerCase())
|
|
404
|
+
return 1;
|
|
405
|
+
}
|
|
406
|
+
if (a.device && b.device) {
|
|
407
|
+
if (a.device.toLowerCase() < b.device.toLowerCase())
|
|
408
|
+
return -1;
|
|
409
|
+
if (a.device.toLowerCase() > b.device.toLowerCase())
|
|
410
|
+
return 1;
|
|
411
|
+
}
|
|
412
|
+
if (a.userAgent && b.userAgent) {
|
|
413
|
+
if (a.userAgent.toLowerCase() < b.userAgent.toLowerCase())
|
|
414
|
+
return -1;
|
|
415
|
+
if (a.userAgent.toLowerCase() > b.userAgent.toLowerCase())
|
|
416
|
+
return 1;
|
|
417
|
+
}
|
|
418
|
+
if (a.width < b.width)
|
|
419
|
+
return -1;
|
|
420
|
+
if (a.width > b.width)
|
|
421
|
+
return 1;
|
|
422
|
+
if (a.height < b.height)
|
|
423
|
+
return -1;
|
|
424
|
+
if (a.height > b.height)
|
|
425
|
+
return 1;
|
|
426
|
+
if (a.id < b.id)
|
|
427
|
+
return -1;
|
|
428
|
+
if (a.id > b.id)
|
|
429
|
+
return 1;
|
|
430
|
+
return 0;
|
|
431
|
+
});
|
|
432
|
+
}
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
/**
|
|
436
|
+
* Convert Windows backslash paths to slash paths: foo\\bar ➔ foo/bar
|
|
437
|
+
* Forward-slash paths can be used in Windows as long as they're not
|
|
438
|
+
* extended-length paths and don't contain any non-ascii characters.
|
|
439
|
+
* This was created since the path methods in Node.js outputs \\ paths on Windows.
|
|
440
|
+
*/
|
|
441
|
+
const normalizePath = (path) => {
|
|
442
|
+
if (typeof path !== 'string') {
|
|
443
|
+
throw new Error(`invalid path to normalize`);
|
|
444
|
+
}
|
|
445
|
+
path = normalizeSlashes(path.trim());
|
|
446
|
+
const components = pathComponents(path, getRootLength(path));
|
|
447
|
+
const reducedComponents = reducePathComponents(components);
|
|
448
|
+
const rootPart = reducedComponents[0];
|
|
449
|
+
const secondPart = reducedComponents[1];
|
|
450
|
+
const normalized = rootPart + reducedComponents.slice(1).join('/');
|
|
451
|
+
if (normalized === '') {
|
|
452
|
+
return '.';
|
|
453
|
+
}
|
|
454
|
+
if (rootPart === '' &&
|
|
455
|
+
secondPart &&
|
|
456
|
+
path.includes('/') &&
|
|
457
|
+
!secondPart.startsWith('.') &&
|
|
458
|
+
!secondPart.startsWith('@')) {
|
|
459
|
+
return './' + normalized;
|
|
460
|
+
}
|
|
461
|
+
return normalized;
|
|
462
|
+
};
|
|
463
|
+
const normalizeSlashes = (path) => path.replace(backslashRegExp, '/');
|
|
464
|
+
const altDirectorySeparator = '\\';
|
|
465
|
+
const urlSchemeSeparator = '://';
|
|
466
|
+
const backslashRegExp = /\\/g;
|
|
467
|
+
const reducePathComponents = (components) => {
|
|
468
|
+
if (!Array.isArray(components) || components.length === 0) {
|
|
469
|
+
return [];
|
|
470
|
+
}
|
|
471
|
+
const reduced = [components[0]];
|
|
472
|
+
for (let i = 1; i < components.length; i++) {
|
|
473
|
+
const component = components[i];
|
|
474
|
+
if (!component)
|
|
475
|
+
continue;
|
|
476
|
+
if (component === '.')
|
|
477
|
+
continue;
|
|
478
|
+
if (component === '..') {
|
|
479
|
+
if (reduced.length > 1) {
|
|
480
|
+
if (reduced[reduced.length - 1] !== '..') {
|
|
481
|
+
reduced.pop();
|
|
482
|
+
continue;
|
|
483
|
+
}
|
|
484
|
+
}
|
|
485
|
+
else if (reduced[0])
|
|
486
|
+
continue;
|
|
487
|
+
}
|
|
488
|
+
reduced.push(component);
|
|
489
|
+
}
|
|
490
|
+
return reduced;
|
|
491
|
+
};
|
|
492
|
+
const getRootLength = (path) => {
|
|
493
|
+
const rootLength = getEncodedRootLength(path);
|
|
494
|
+
return rootLength < 0 ? ~rootLength : rootLength;
|
|
495
|
+
};
|
|
496
|
+
const getEncodedRootLength = (path) => {
|
|
497
|
+
if (!path)
|
|
498
|
+
return 0;
|
|
499
|
+
const ch0 = path.charCodeAt(0);
|
|
500
|
+
// POSIX or UNC
|
|
501
|
+
if (ch0 === 47 /* slash */ || ch0 === 92 /* backslash */) {
|
|
502
|
+
if (path.charCodeAt(1) !== ch0)
|
|
503
|
+
return 1; // POSIX: "/" (or non-normalized "\")
|
|
504
|
+
const p1 = path.indexOf(ch0 === 47 /* slash */ ? '/' : altDirectorySeparator, 2);
|
|
505
|
+
if (p1 < 0)
|
|
506
|
+
return path.length; // UNC: "//server" or "\\server"
|
|
507
|
+
return p1 + 1; // UNC: "//server/" or "\\server\"
|
|
508
|
+
}
|
|
509
|
+
// DOS
|
|
510
|
+
if (isVolumeCharacter(ch0) && path.charCodeAt(1) === 58 /* colon */) {
|
|
511
|
+
const ch2 = path.charCodeAt(2);
|
|
512
|
+
if (ch2 === 47 /* slash */ || ch2 === 92 /* backslash */)
|
|
513
|
+
return 3; // DOS: "c:/" or "c:\"
|
|
514
|
+
if (path.length === 2)
|
|
515
|
+
return 2; // DOS: "c:" (but not "c:d")
|
|
516
|
+
}
|
|
517
|
+
// URL
|
|
518
|
+
const schemeEnd = path.indexOf(urlSchemeSeparator);
|
|
519
|
+
if (schemeEnd !== -1) {
|
|
520
|
+
const authorityStart = schemeEnd + urlSchemeSeparator.length;
|
|
521
|
+
const authorityEnd = path.indexOf('/', authorityStart);
|
|
522
|
+
if (authorityEnd !== -1) {
|
|
523
|
+
// URL: "file:///", "file://server/", "file://server/path"
|
|
524
|
+
// For local "file" URLs, include the leading DOS volume (if present).
|
|
525
|
+
// Per https://www.ietf.org/rfc/rfc1738.txt, a host of "" or "localhost" is a
|
|
526
|
+
// special case interpreted as "the machine from which the URL is being interpreted".
|
|
527
|
+
const scheme = path.slice(0, schemeEnd);
|
|
528
|
+
const authority = path.slice(authorityStart, authorityEnd);
|
|
529
|
+
if (scheme === 'file' &&
|
|
530
|
+
(authority === '' || authority === 'localhost') &&
|
|
531
|
+
isVolumeCharacter(path.charCodeAt(authorityEnd + 1))) {
|
|
532
|
+
const volumeSeparatorEnd = getFileUrlVolumeSeparatorEnd(path, authorityEnd + 2);
|
|
533
|
+
if (volumeSeparatorEnd !== -1) {
|
|
534
|
+
if (path.charCodeAt(volumeSeparatorEnd) === 47 /* slash */) {
|
|
535
|
+
// URL: "file:///c:/", "file://localhost/c:/", "file:///c%3a/", "file://localhost/c%3a/"
|
|
536
|
+
return ~(volumeSeparatorEnd + 1);
|
|
537
|
+
}
|
|
538
|
+
if (volumeSeparatorEnd === path.length) {
|
|
539
|
+
// URL: "file:///c:", "file://localhost/c:", "file:///c$3a", "file://localhost/c%3a"
|
|
540
|
+
// but not "file:///c:d" or "file:///c%3ad"
|
|
541
|
+
return ~volumeSeparatorEnd;
|
|
542
|
+
}
|
|
543
|
+
}
|
|
544
|
+
}
|
|
545
|
+
return ~(authorityEnd + 1); // URL: "file://server/", "http://server/"
|
|
546
|
+
}
|
|
547
|
+
return ~path.length; // URL: "file://server", "http://server"
|
|
548
|
+
}
|
|
549
|
+
// relative
|
|
550
|
+
return 0;
|
|
551
|
+
};
|
|
552
|
+
const isVolumeCharacter = (charCode) => (charCode >= 97 /* a */ && charCode <= 122 /* z */) ||
|
|
553
|
+
(charCode >= 65 /* A */ && charCode <= 90 /* Z */);
|
|
554
|
+
const getFileUrlVolumeSeparatorEnd = (url, start) => {
|
|
555
|
+
const ch0 = url.charCodeAt(start);
|
|
556
|
+
if (ch0 === 58 /* colon */)
|
|
557
|
+
return start + 1;
|
|
558
|
+
if (ch0 === 37 /* percent */ && url.charCodeAt(start + 1) === 51 /* _3 */) {
|
|
559
|
+
const ch2 = url.charCodeAt(start + 2);
|
|
560
|
+
if (ch2 === 97 /* a */ || ch2 === 65 /* A */)
|
|
561
|
+
return start + 3;
|
|
562
|
+
}
|
|
563
|
+
return -1;
|
|
564
|
+
};
|
|
565
|
+
const pathComponents = (path, rootLength) => {
|
|
566
|
+
const root = path.substring(0, rootLength);
|
|
567
|
+
const rest = path.substring(rootLength).split('/');
|
|
568
|
+
const restLen = rest.length;
|
|
569
|
+
if (restLen > 0 && !rest[restLen - 1]) {
|
|
570
|
+
rest.pop();
|
|
571
|
+
}
|
|
572
|
+
return [root, ...rest];
|
|
573
|
+
};
|
|
574
|
+
|
|
575
|
+
class ScreenshotLocalConnector extends ScreenshotConnector {
|
|
576
|
+
async publishBuild(results) {
|
|
577
|
+
if (this.updateMaster || !results.masterBuild) {
|
|
578
|
+
results.masterBuild = {
|
|
579
|
+
id: 'master',
|
|
580
|
+
message: 'Master',
|
|
581
|
+
appNamespace: this.appNamespace,
|
|
582
|
+
timestamp: Date.now(),
|
|
583
|
+
screenshots: [],
|
|
584
|
+
};
|
|
585
|
+
}
|
|
586
|
+
results.currentBuild.screenshots.forEach((currentScreenshot) => {
|
|
587
|
+
const masterHasScreenshot = results.masterBuild.screenshots.some((masterScreenshot) => {
|
|
588
|
+
return currentScreenshot.id === masterScreenshot.id;
|
|
589
|
+
});
|
|
590
|
+
if (!masterHasScreenshot) {
|
|
591
|
+
results.masterBuild.screenshots.push(Object.assign({}, currentScreenshot));
|
|
592
|
+
}
|
|
593
|
+
});
|
|
594
|
+
this.sortScreenshots(results.masterBuild.screenshots);
|
|
595
|
+
await writeFile(this.masterBuildFilePath, JSON.stringify(results.masterBuild, null, 2));
|
|
596
|
+
await this.generateJsonpDataUris(results.currentBuild);
|
|
597
|
+
const compareAppSourceDir = path.join(this.packageDir, 'screenshot', 'compare');
|
|
598
|
+
const appSrcUrl = normalizePath(path.relative(this.screenshotDir, compareAppSourceDir));
|
|
599
|
+
const imagesUrl = normalizePath(path.relative(this.screenshotDir, this.imagesDir));
|
|
600
|
+
const jsonpUrl = normalizePath(path.relative(this.screenshotDir, this.cacheDir));
|
|
601
|
+
const compareAppHtml = createLocalCompareApp(this.appNamespace, appSrcUrl, imagesUrl, jsonpUrl, results.masterBuild, results.currentBuild);
|
|
602
|
+
const compareAppFileName = 'compare.html';
|
|
603
|
+
const compareAppFilePath = path.join(this.screenshotDir, compareAppFileName);
|
|
604
|
+
await writeFile(compareAppFilePath, compareAppHtml);
|
|
605
|
+
const gitIgnorePath = path.join(this.screenshotDir, '.gitignore');
|
|
606
|
+
const gitIgnoreExists = await fileExists(gitIgnorePath);
|
|
607
|
+
if (!gitIgnoreExists) {
|
|
608
|
+
const content = [this.imagesDirName, this.buildsDirName, compareAppFileName];
|
|
609
|
+
await writeFile(gitIgnorePath, content.join('\n'));
|
|
610
|
+
}
|
|
611
|
+
const url = new URL(`file://${compareAppFilePath}`);
|
|
612
|
+
results.compare.url = url.href;
|
|
613
|
+
return results;
|
|
614
|
+
}
|
|
615
|
+
async getScreenshotCache() {
|
|
616
|
+
let screenshotCache = null;
|
|
617
|
+
try {
|
|
618
|
+
screenshotCache = JSON.parse(await readFile(this.screenshotCacheFilePath));
|
|
619
|
+
}
|
|
620
|
+
catch (e) { }
|
|
621
|
+
return screenshotCache;
|
|
622
|
+
}
|
|
623
|
+
async updateScreenshotCache(cache, buildResults) {
|
|
624
|
+
cache = await super.updateScreenshotCache(cache, buildResults);
|
|
625
|
+
await writeFile(this.screenshotCacheFilePath, JSON.stringify(cache, null, 2));
|
|
626
|
+
return cache;
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
function createLocalCompareApp(namespace, appSrcUrl, imagesUrl, jsonpUrl, a, b) {
|
|
630
|
+
return `<!doctype html>
|
|
631
|
+
<html dir="ltr" lang="en">
|
|
632
|
+
<head>
|
|
633
|
+
<meta charset="utf-8">
|
|
634
|
+
<title>Local ${namespace || ''} - Rindo Screenshot Visual Diff</title>
|
|
635
|
+
<meta name="viewport" content="viewport-fit=cover, width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=no">
|
|
636
|
+
<meta http-equiv="x-ua-compatible" content="IE=Edge">
|
|
637
|
+
<link href="${appSrcUrl}/build/app.css" rel="stylesheet">
|
|
638
|
+
<script type="module" src="${appSrcUrl}/build/app.esm.js"></script>
|
|
639
|
+
<script nomodule src="${appSrcUrl}/build/app.js"></script>
|
|
640
|
+
<link rel="icon" type="image/x-icon" href="${appSrcUrl}/assets/favicon.ico">
|
|
641
|
+
</head>
|
|
642
|
+
<body>
|
|
643
|
+
<script>
|
|
644
|
+
(function() {
|
|
645
|
+
var app = document.createElement('screenshot-compare');
|
|
646
|
+
app.appSrcUrl = '${appSrcUrl}';
|
|
647
|
+
app.imagesUrl = '${imagesUrl}/';
|
|
648
|
+
app.jsonpUrl = '${jsonpUrl}/';
|
|
649
|
+
app.a = ${JSON.stringify(a)};
|
|
650
|
+
app.b = ${JSON.stringify(b)};
|
|
651
|
+
document.body.appendChild(app);
|
|
652
|
+
})();
|
|
653
|
+
</script>
|
|
654
|
+
</body>
|
|
655
|
+
</html>`;
|
|
656
|
+
}
|
|
657
|
+
|
|
658
|
+
exports.ScreenshotConnector = ScreenshotConnector;
|
|
659
|
+
exports.ScreenshotLocalConnector = ScreenshotLocalConnector;
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@rindo/core/screenshot",
|
|
3
|
+
"version": "0.0.0-dev.20220810110141",
|
|
4
|
+
"description": "Rindo Screenshot.",
|
|
5
|
+
"main": "./index.js",
|
|
6
|
+
"types": "./index.d.ts",
|
|
7
|
+
"files": [
|
|
8
|
+
"compare/",
|
|
9
|
+
"index.js",
|
|
10
|
+
"connector.js",
|
|
11
|
+
"local-connector.js",
|
|
12
|
+
"pixel-match.js"
|
|
13
|
+
],
|
|
14
|
+
"private": true
|
|
15
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|