happo 2.8.1 → 2.8.2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/lib/happo/public/026c001beac0620bb23a.worker.js +50 -0
- data/lib/happo/public/05ad45a945afc6882286.worker.js +56 -0
- data/lib/happo/public/0dd329b74ed90a207896.worker.js +56 -0
- data/lib/happo/public/174b944ab9e9053c766d.worker.js +56 -0
- data/lib/happo/public/196b75660e2791fda25a.worker.js +56 -0
- data/lib/happo/public/1d5c0eb9c479f44f0ec2.worker.js +56 -0
- data/lib/happo/public/1d6c41b4828b191ae315.worker.js +56 -0
- data/lib/happo/public/212269d558f1f8b1bc63.worker.js +56 -0
- data/lib/happo/public/25426637d793045ccb77.worker.js +56 -0
- data/lib/happo/public/28537ac53e64479763ba.worker.js +56 -0
- data/lib/happo/public/30e3af0e8f974da2885c.worker.js +56 -0
- data/lib/happo/public/32fc2a09f98d30cb2a06.worker.js +56 -0
- data/lib/happo/public/33c43f9f1296a6c3b236.worker.js +56 -0
- data/lib/happo/public/3a6592504d8ee1810017.worker.js +56 -0
- data/lib/happo/public/3a8bedd3121a2cdacb6d.worker.js +56 -0
- data/lib/happo/public/3d9ed68ca315896c8ec7.worker.js +56 -0
- data/lib/happo/public/3da88471c60703398f95.worker.js +56 -0
- data/lib/happo/public/40c1aadf458bda245630.worker.js +1 -0
- data/lib/happo/public/41749e1ff9db63f87a1e.worker.js +56 -0
- data/lib/happo/public/42bc2c20d61979a23a2f.worker.js +56 -0
- data/lib/happo/public/447f3bfe79be8a67b9cc.worker.js +56 -0
- data/lib/happo/public/4580dd75ba5a064f4366.worker.js +56 -0
- data/lib/happo/public/4975b25fe791d85e9827.worker.js +56 -0
- data/lib/happo/public/4b23fa5d7875a8fba618.worker.js +56 -0
- data/lib/happo/public/4ecbb720788698ce7a9b.worker.js +56 -0
- data/lib/happo/public/4ef2e26b666b08b7411a.worker.js +56 -0
- data/lib/happo/public/51ac7ca571ae07cd9d7a.worker.js +56 -0
- data/lib/happo/public/531d99f56a3343cee5f8.worker.js +56 -0
- data/lib/happo/public/53e6a60c2cf0558157a7.worker.js +56 -0
- data/lib/happo/public/55d488167040d5350534.worker.js +56 -0
- data/lib/happo/public/588f7a6b784ef2b87932.worker.js +56 -0
- data/lib/happo/public/59d03b8990473ae492e0.worker.js +56 -0
- data/lib/happo/public/5aded6f9af816e819a2b.worker.js +56 -0
- data/lib/happo/public/5e6771e4a41f597c26a9.worker.js +56 -0
- data/lib/happo/public/61e924aed1761ec05ca0.worker.js +56 -0
- data/lib/happo/public/65e7608bb10d0f2250d1.worker.js +56 -0
- data/lib/happo/public/6a71a1273a0cf4065e1e.worker.js +56 -0
- data/lib/happo/public/727ca4589d56ddcb9c7f.worker.js +56 -0
- data/lib/happo/public/769883a4d5abfa7384d4.worker.js +56 -0
- data/lib/happo/public/78fea81d29830297df57.worker.js +56 -0
- data/lib/happo/public/7bf45d52cf57b1949851.worker.js +56 -0
- data/lib/happo/public/7d4a4086de8530abe101.worker.js +56 -0
- data/lib/happo/public/7e330bc3ee246ab39910.worker.js +56 -0
- data/lib/happo/public/84f3a66517b77798456d.worker.js +56 -0
- data/lib/happo/public/856dbe98c40d8f4bd452.worker.js +56 -0
- data/lib/happo/public/90a4b81e4d7c2da63973.worker.js +56 -0
- data/lib/happo/public/92dd03e7cbe827a2df45.worker.js +56 -0
- data/lib/happo/public/94976fe48214f2bc68d6.worker.js +56 -0
- data/lib/happo/public/95f3e6e7eec59c0af3e5.worker.js +56 -0
- data/lib/happo/public/96c4c2f5df8175649a67.worker.js +56 -0
- data/lib/happo/public/973a6ae106ac4e664858.worker.js +56 -0
- data/lib/happo/public/989e9ba236891bdfcf46.worker.js +56 -0
- data/lib/happo/public/HappoApp.bundle.js +1 -2
- data/lib/happo/public/a50e9af74a422839a15c.worker.js +56 -0
- data/lib/happo/public/ad4fdfa82741585ce164.worker.js +56 -0
- data/lib/happo/public/ae0622a81101951ac112.worker.js +56 -0
- data/lib/happo/public/ae223678cdbfac88d9be.worker.js +56 -0
- data/lib/happo/public/b441a6c1393a5d39b5ff.worker.js +56 -0
- data/lib/happo/public/b4c84506199d5745d950.worker.js +56 -0
- data/lib/happo/public/b5176c936b696653a407.worker.js +56 -0
- data/lib/happo/public/b86b963a72892d8f0fd1.worker.js +56 -0
- data/lib/happo/public/ba50214d2b6ac001493b.worker.js +56 -0
- data/lib/happo/public/bb6413d1cea030ba7a1e.worker.js +56 -0
- data/lib/happo/public/bc084ad60fb7dbbe0818.worker.js +56 -0
- data/lib/happo/public/c15e6b44ae496af6af9b.worker.js +56 -0
- data/lib/happo/public/c4393394848f5242932e.worker.js +56 -0
- data/lib/happo/public/c60ceaff7849b394e27a.worker.js +56 -0
- data/lib/happo/public/c67c76de1e4def5019d5.worker.js +56 -0
- data/lib/happo/public/c7969739b50b79043b86.worker.js +56 -0
- data/lib/happo/public/cc82d7609935eb1d2738.worker.js +56 -0
- data/lib/happo/public/d135dda9b0318be75e3c.worker.js +56 -0
- data/lib/happo/public/d2aded5d63912f5b7e1b.worker.js +56 -0
- data/lib/happo/public/d2be644581f8f67b207d.worker.js +56 -0
- data/lib/happo/public/d6e037d3891cfb95fbca.worker.js +50 -0
- data/lib/happo/public/d7440908d264133044bb.worker.js +56 -0
- data/lib/happo/public/da071c51e9a641ed805c.worker.js +56 -0
- data/lib/happo/public/dbe8eef9555ef0d43469.worker.js +56 -0
- data/lib/happo/public/e1cbbfd7a3d705049eb0.worker.js +56 -0
- data/lib/happo/public/e3882591e6710cf03ed4.worker.js +56 -0
- data/lib/happo/public/e414c768d66b9bcf86bb.worker.js +56 -0
- data/lib/happo/public/e85bc4bb461158f5561d.worker.js +56 -0
- data/lib/happo/public/ed6256235d0a2b11aaf6.worker.js +56 -0
- data/lib/happo/public/edcc84bf4483670ba23e.worker.js +56 -0
- data/lib/happo/public/f3d33d482a4326121c45.worker.js +56 -0
- data/lib/happo/public/f67e23a954230ea878be.worker.js +56 -0
- data/lib/happo/public/f800c3a1d9200926bde6.worker.js +56 -0
- data/lib/happo/public/f8954fa2849952175e2c.worker.js +56 -0
- data/lib/happo/public/fde5a206ff86c3d7ab10.worker.js +56 -0
- data/lib/happo/public/globe-current.png +0 -0
- data/lib/happo/public/globe-previous.png +0 -0
- data/lib/happo/server.rb +7 -0
- data/lib/happo/version.rb +1 -1
- metadata +91 -3
- data/lib/happo/public/happo-logo.svg +0 -1
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n if (memo[aLength][bLength] === 0) {\n // No soltuion found\n return null;\n }\n return solution;\n}\n\nfunction placeholders(count) {\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n var changes = 0;\n\n function applyChange() {\n if (changes < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n } else if (changes > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(changes))));\n }\n changes = 0;\n }\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n applyChange();\n ai--;\n bi--;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n changes++;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n changes--;\n }\n movement = solution[ai][bi];\n }\n if (ai - bi !== 0) {\n b.splice.apply(b, [0, 0].concat(_toConsumableArray(placeholders(bi))));\n a.splice.apply(a, [0, 0].concat(_toConsumableArray(placeholders(ai))));\n }\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n var aLength = a.length;\n var bLength = b.length;\n var shorterArray = aLength > bLength ? b : a;\n shorterArray.push.apply(shorterArray, _toConsumableArray(placeholders(Math.abs(aLength - bLength))));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n if (lcsSolution) {\n applySolution(lcsSolution, a, b);\n } else {\n applyPadding(a, b);\n }\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/alignArrays.js?01b7"],"names":["alignArrays","MOVEMENT","none","diagonal","vertical","horizontal","initMatrix","height","width","rows","Array","i","length","Int32Array","longestCommonSubsequence","a","b","aLength","bLength","memo","solution","j","Math","max","placeholders","count","result","applySolution","movement","ai","bi","changes","applyChange","splice","abs","applyPadding","shorterArray","push","lcsSolution"],"mappings":"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","file":"1.js","sourcesContent":["const MOVEMENT = {\n  none: 0,\n  diagonal: 1,\n  vertical: 2,\n  horizontal: 3,\n};\n\nfunction initMatrix(height, width) {\n  const rows = new Array(height);\n  for (let i = 0; i < rows.length; i++) {\n    rows[i] = new Int32Array(width);\n  }\n  return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n  // adapted from\n  // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n  const aLength = a.length;\n  const bLength = b.length;\n  const memo = initMatrix(aLength + 1, bLength + 1);\n  const solution = initMatrix(aLength + 1, bLength + 1);\n\n  // Loop and find the solution\n  for (let i = 1; i <= aLength; i++) {\n    for (let j = 1; j <= bLength; j++) {\n      if (a[i - 1] === b[j - 1]) {\n        // diagonal\n        memo[i][j] = memo[i - 1][j - 1] + 1;\n        solution[i][j] = MOVEMENT.diagonal;\n      } else {\n        memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n        if (memo[i][j] === memo[i - 1][j]) {\n          solution[i][j] = MOVEMENT.vertical;\n        } else {\n          solution[i][j] = MOVEMENT.horizontal;\n        }\n      }\n    }\n  }\n  if (memo[aLength][bLength] === 0) {\n    // No soltuion found\n    return null;\n  }\n  return solution;\n}\n\nfunction placeholders(count) {\n  const result = new Array(count);\n  for (let i = 0; i < count; i++) {\n    result[i] = '+';\n  }\n  return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n  let movement = solution[a.length][b.length];\n  let ai = a.length;\n  let bi = b.length;\n  let changes = 0;\n\n  function applyChange() {\n    if (changes < 0) {\n      b.splice(bi, 0, ...placeholders(Math.abs(changes)));\n    } else if (changes > 0) {\n      a.splice(ai, 0, ...placeholders(changes));\n    }\n    changes = 0;\n  }\n\n  while (movement !== MOVEMENT.none) {\n    if (movement === MOVEMENT.diagonal) {\n      applyChange();\n      ai--;\n      bi--;\n    } else if (movement === MOVEMENT.horizontal) {\n      bi--;\n      changes++;\n    } else if (movement === MOVEMENT.vertical) {\n      ai--;\n      changes--;\n    }\n    movement = solution[ai][bi];\n  }\n  if (ai - bi !== 0) {\n    b.splice(0, 0, ...placeholders(bi));\n    a.splice(0, 0, ...placeholders(ai));\n  }\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n  const aLength = a.length;\n  const bLength = b.length;\n  const shorterArray = aLength > bLength ? b : a;\n  shorterArray.push(...placeholders(Math.abs(aLength - bLength)));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nexport default function alignArrays(a, b) {\n  const lcsSolution = longestCommonSubsequence(a, b);\n  if (lcsSolution) {\n    applySolution(lcsSolution, a, b);\n  } else {\n    applyPadding(a, b);\n  }\n}\n\n\n\n/** WEBPACK FOOTER **\n ** ./js/src/alignArrays.js\n **/"],"sourceRoot":""}");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n console.log(memo);\n return solution;\n}\n\nfunction placeholders(count) {\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n var diffI = ai - bi;\n console.log(ai, bi, diffI);\n if (diffI < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(diffI)))));\n } else if (diffI > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(Math.abs(diffI)))));\n }\n ai--;\n bi--;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n }\n movement = solution[ai][bi];\n }\n}\n\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n applySolution(lcsSolution, a, b);\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvYWxpZ25BcnJheXMuanM/MDFiNyJdLCJuYW1lcyI6WyJhbGlnbkFycmF5cyIsIk1PVkVNRU5UIiwibm9uZSIsImRpYWdvbmFsIiwidmVydGljYWwiLCJob3Jpem9udGFsIiwiaW5pdE1hdHJpeCIsImhlaWdodCIsIndpZHRoIiwicm93cyIsIkFycmF5IiwiaSIsImxlbmd0aCIsIkludDMyQXJyYXkiLCJsb25nZXN0Q29tbW9uU3Vic2VxdWVuY2UiLCJhIiwiYiIsImFMZW5ndGgiLCJiTGVuZ3RoIiwibWVtbyIsInNvbHV0aW9uIiwiaiIsIk1hdGgiLCJtYXgiLCJjb25zb2xlIiwibG9nIiwicGxhY2Vob2xkZXJzIiwiY291bnQiLCJyZXN1bHQiLCJhcHBseVNvbHV0aW9uIiwibW92ZW1lbnQiLCJhaSIsImJpIiwiZGlmZkkiLCJzcGxpY2UiLCJhYnMiLCJsY3NTb2x1dGlvbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7a0JBNEZ3QkEsVzs7OztBQTVGeEIsSUFBTUMsV0FBVztBQUNmQyxRQUFNLENBRFM7QUFFZkMsWUFBVSxDQUZLO0FBR2ZDLFlBQVUsQ0FISztBQUlmQyxjQUFZO0FBSkcsQ0FBakI7O0FBT0EsU0FBU0MsVUFBVCxDQUFvQkMsTUFBcEIsRUFBNEJDLEtBQTVCLEVBQW1DO0FBQ2pDLE1BQU1DLE9BQU8sSUFBSUMsS0FBSixDQUFVSCxNQUFWLENBQWI7QUFDQSxPQUFLLElBQUlJLElBQUksQ0FBYixFQUFnQkEsSUFBSUYsS0FBS0csTUFBekIsRUFBaUNELEdBQWpDLEVBQXNDO0FBQ3BDRixTQUFLRSxDQUFMLElBQVUsSUFBSUUsVUFBSixDQUFlTCxLQUFmLENBQVY7QUFDRDtBQUNELFNBQU9DLElBQVA7QUFDRDs7QUFFRCxTQUFTSyx3QkFBVCxDQUFrQ0MsQ0FBbEMsRUFBcUNDLENBQXJDLEVBQXdDO0FBQ3RDO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixFQUFFSCxNQUFsQjtBQUNBLE1BQU1NLFVBQVVGLEVBQUVKLE1B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LEVBQVQsRUFBYUMsRUFBYixDQUFYO0FBQ0Q7QUFDRjs7QUFFRDs7Ozs7Ozs7QUFRZSxTQUFTaEMsV0FBVCxDQUFxQmUsQ0FBckIsRUFBd0JDLENBQXhCLEVBQTJCO0FBQ3hDLE1BQU1vQixjQUFjdEIseUJBQXlCQyxDQUF6QixFQUE0QkMsQ0FBNUIsQ0FBcEI7QUFDQWEsZ0JBQWNPLFdBQWQsRUFBMkJyQixDQUEzQixFQUE4QkMsQ0FBOUI7QUFDRCIsImZpbGUiOiIxLmpzIiwic291cmNlc0NvbnRlbnQiOlsiY29uc3QgTU9WRU1FTlQgPSB7XG4gIG5vbmU6IDAsXG4gIGRpYWdvbmFsOiAxLFxuICB2ZXJ0aWNhbDogMixcbiAgaG9yaXpvbnRhbDogMyxcbn07XG5cbmZ1bmN0aW9uIGluaXRNYXRyaXgoaGVpZ2h0LCB3aWR0aCkge1xuICBjb25zdCByb3dzID0gbmV3IEFycmF5KGhlaWdodCk7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgcm93cy5sZW5ndGg7IGkrKykge1xuICAgIHJvd3NbaV0gPSBuZXcgSW50MzJBcnJheSh3aWR0aCk7XG4gIH1cbiAgcmV0dXJuIHJvd3M7XG59XG5cbmZ1bmN0aW9uIGxvbmdlc3RDb21tb25TdWJzZXF1ZW5jZShhLCBiKSB7XG4gIC8vIGFkYXB0ZWQgZnJvbVxuICAvLyBodHRwOi8vYWxnb3JpdGhtcy50dXRvcmlhbGhvcml6b24uY29tL2R5bmFtaWMtcHJvZ3JhbW1pbmctbG9uZ2VzdC1jb21tb24tc3Vic2VxdWVuY2UvXG4gIGNvbnN0IGFMZW5ndGggPSBhLmxlbmd0aDtcbiAgY29uc3QgYkxlbmd0aCA9IGIubGVuZ3RoO1xuICBjb25zdCBtZW1vID0gaW5pdE1hdHJpeChhTGVuZ3RoICsgMSwgYkxlbmd0aCArIDEpO1xuICBjb25zdCBzb2x1dGlvbiA9IGluaXRNYXRyaXgoYUxlbmd0aCArIDEsIGJMZW5ndGggKyAxKTtcblxuICAvLyBMb29wIGFuZCBmaW5kIHRoZSBzb2x1dGlvblxuICBmb3IgKGxldCBpID0gMTsgaSA8PSBhTGVuZ3RoOyBpKyspIHtcbiAgICBmb3IgKGxldCBqID0gMTsgaiA8PSBiTGVuZ3RoOyBqKyspIHtcbiAgICAgIGlmIChhW2kgLSAxXSA9PT0gYltqIC0gMV0pIHtcbiAgICAgICAgLy8gZGlhZ29uYWxcbiAgICAgICAgbWVtb1tpXVtqXSA9IG1lbW9baSAtIDFdW2ogLSAxXSArIDE7XG4gICAgICAgIHNvbHV0aW9uW2ldW2pdID0gTU9WRU1FTlQuZGlhZ29uYWw7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBtZW1vW2ldW2pdID0gTWF0aC5tYXgobWVtb1tpIC0gMV1bal0sIG1lbW9baV1baiAtIDFdKTtcbiAgICAgICAgaWYgKG1lbW9baV1bal0gPT09IG1lbW9baSAtIDFdW2pdKSB7XG4gICAgICAgICAgc29sdXRpb25baV1bal0gPSBNT1ZFTUVOVC52ZXJ0aWNhbDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBzb2x1dGlvbltpXVtqXSA9IE1PVkVNRU5ULmhvcml6b250YWw7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cbiAgY29uc29sZS5sb2cobWVtbylcbiAgcmV0dXJuIHNvbHV0aW9uO1xufVxuXG5mdW5jdGlvbiBwbGFjZWhvbGRlcnMoY291bnQpIHtcbiAgY29uc3QgcmVzdWx0ID0gbmV3IEFycmF5KGNvdW50KTtcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBjb3VudDsgaSsrKSB7XG4gICAgcmVzdWx0W2ldID0gJysnO1xuICB9XG4gIHJldHVybiByZXN1bHQ7XG59XG4vKipcbiAqIEFwcGx5IGFuIGxjcyBzb2x1dGlvbiB0byBhcnJheXMuIE5vdGUgdGhhdCB0aGlzIHdpbGwgTVVUQVRFIHRoZSBhcnJheXMsXG4gKiBpbmplY3RpbmcgXCIrXCIgd2hlcmUgZ2FwcyBhcmUgbmVlZGVkLlxuICpcbiAqIEBwYXJhbSB7QXJyYXk8QXJyYXk+fSBzb2x1dGlvbiBhcyBjb21wdXRlZCBieSBgbG9uZ2VzdENvbW1vblN1YnNlcXVlbmNlYFxuICogQHBhcmFtIHtBcnJheX0gYVxuICogQHBhcmFtIHtBcnJheX0gYlxuICovXG5mdW5jdGlvbiBhcHBseVNvbHV0aW9uKHNvbHV0aW9uLCBhLCBiKSB7XG4gIGxldCBtb3ZlbWVudCA9IHNvbHV0aW9uW2EubGVuZ3RoXVtiLmxlbmd0aF07XG4gIGxldCBhaSA9IGEubGVuZ3RoO1xuICBsZXQgYmkgPSBiLmxlbmd0aDtcblxuICB3aGlsZSAobW92ZW1lbnQgIT09IE1PVkVNRU5ULm5vbmUpIHtcbiAgICBpZiAobW92ZW1lbnQgPT09IE1PVkVNRU5ULmRpYWdvbmFsKSB7XG4gICAgICBjb25zdCBkaWZmSSA9IGFpIC0gYmk7XG4gICAgICBjb25zb2xlLmxvZyhhaSwgYmksIGRpZmZJKVxuICAgICAgaWYgKGRpZmZJIDwgMCkge1xuICAgICAgICBiLnNwbGljZShiaSwgMCwgLi4ucGxhY2Vob2xkZXJzKE1hdGguYWJzKGRpZmZJKSkpO1xuICAgICAgfSBlbHNlIGlmIChkaWZmSSA+IDApIHtcbiAgICAgICAgYS5zcGxpY2UoYWksIDAsIC4uLnBsYWNlaG9sZGVycyhNYXRoLmFicyhkaWZmSSkpKTtcbiAgICAgIH1cbiAgICAgIGFpLS07XG4gICAgICBiaS0tO1xuICAgIH0gZWxzZSBpZiAobW92ZW1lbnQgPT09IE1PVkVNRU5ULmhvcml6b250YWwpIHtcbiAgICAgIGJpLS07XG4gICAgfSBlbHNlIGlmIChtb3ZlbWVudCA9PT0gTU9WRU1FTlQudmVydGljYWwpIHtcbiAgICAgIGFpLS07XG4gICAgfVxuICAgIG1vdmVtZW50ID0gc29sdXRpb25bYWldW2JpXTtcbiAgfVxufVxuXG4vKipcbiAqIENvbXB1dGVzIHRoZSBsb25nZXN0IGNvbW1vbiBzdWJzZXF1ZW5jZSBvZiB0d28gYXJyYXlzLCB0aGVuIHVzZXMgdGhhdFxuICogc29sdXRpb24gdG8gaW5qZWN0IGdhcHMgaW50byB0aGUgYXJyYXlzLCBtYWtpbmcgdGhlbSBhbGlnbiBvbiBjb21tb25cbiAqIHN1YnNlcXVlbmNlcy5cbiAqXG4gKiBAcGFyYW0ge0FycmF5fSBhXG4gKiBAcGFyYW0ge0FycmF5fSBiXG4gKi9cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIGFsaWduQXJyYXlzKGEsIGIpIHtcbiAgY29uc3QgbGNzU29sdXRpb24gPSBsb25nZXN0Q29tbW9uU3Vic2VxdWVuY2UoYSwgYik7XG4gIGFwcGx5U29sdXRpb24obGNzU29sdXRpb24sIGEsIGIpO1xufVxuXG5cblxuLyoqIFdFQlBBQ0sgRk9PVEVSICoqXG4gKiogLi9qcy9zcmMvYWxpZ25BcnJheXMuanNcbiAqKi8iXSwic291cmNlUm9vdCI6IiJ9");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n if (memo[aLength][bLength] === 0) {\n // No soltuion found\n return null;\n }\n return solution;\n}\n\nfunction placeholders(count) {\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n var changes = 0;\n\n function applyChange() {\n if (changes < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n } else if (changes > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(changes))));\n }\n changes = 0;\n }\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n applyChange();\n ai--;\n bi--;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n changes++;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n changes--;\n }\n movement = solution[ai][bi];\n }\n changes = ai - bi;\n ai = 0;\n bi = 0;\n applyChange();\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n var aLength = a.length;\n var bLength = b.length;\n var shorterArray = aLength > bLength ? b : a;\n shorterArray.push.apply(shorterArray, _toConsumableArray(placeholders(Math.abs(aLength - bLength))));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n if (lcsSolution) {\n applySolution(lcsSolution, a, b);\n } else {\n applyPadding(a, b);\n }\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/alignArrays.js?01b7"],"names":["alignArrays","MOVEMENT","none","diagonal","vertical","horizontal","initMatrix","height","width","rows","Array","i","length","Int32Array","longestCommonSubsequence","a","b","aLength","bLength","memo","solution","j","Math","max","placeholders","count","result","applySolution","movement","ai","bi","changes","applyChange","splice","abs","applyPadding","shorterArray","push","lcsSolution"],"mappings":"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","file":"1.js","sourcesContent":["const MOVEMENT = {\n  none: 0,\n  diagonal: 1,\n  vertical: 2,\n  horizontal: 3,\n};\n\nfunction initMatrix(height, width) {\n  const rows = new Array(height);\n  for (let i = 0; i < rows.length; i++) {\n    rows[i] = new Int32Array(width);\n  }\n  return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n  // adapted from\n  // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n  const aLength = a.length;\n  const bLength = b.length;\n  const memo = initMatrix(aLength + 1, bLength + 1);\n  const solution = initMatrix(aLength + 1, bLength + 1);\n\n  // Loop and find the solution\n  for (let i = 1; i <= aLength; i++) {\n    for (let j = 1; j <= bLength; j++) {\n      if (a[i - 1] === b[j - 1]) {\n        // diagonal\n        memo[i][j] = memo[i - 1][j - 1] + 1;\n        solution[i][j] = MOVEMENT.diagonal;\n      } else {\n        memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n        if (memo[i][j] === memo[i - 1][j]) {\n          solution[i][j] = MOVEMENT.vertical;\n        } else {\n          solution[i][j] = MOVEMENT.horizontal;\n        }\n      }\n    }\n  }\n  if (memo[aLength][bLength] === 0) {\n    // No soltuion found\n    return null;\n  }\n  return solution;\n}\n\nfunction placeholders(count) {\n  const result = new Array(count);\n  for (let i = 0; i < count; i++) {\n    result[i] = '+';\n  }\n  return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n  let movement = solution[a.length][b.length];\n  let ai = a.length;\n  let bi = b.length;\n  let changes = 0;\n\n  function applyChange() {\n    if (changes < 0) {\n      b.splice(bi, 0, ...placeholders(Math.abs(changes)));\n    } else if (changes > 0) {\n      a.splice(ai, 0, ...placeholders(changes));\n    }\n    changes = 0;\n  }\n\n  while (movement !== MOVEMENT.none) {\n    if (movement === MOVEMENT.diagonal) {\n      applyChange();\n      ai--;\n      bi--;\n    } else if (movement === MOVEMENT.horizontal) {\n      bi--;\n      changes++;\n    } else if (movement === MOVEMENT.vertical) {\n      ai--;\n      changes--;\n    }\n    movement = solution[ai][bi];\n  }\n  changes = ai - bi;\n  ai = 0;\n  bi = 0;\n  applyChange();\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n  const aLength = a.length;\n  const bLength = b.length;\n  const shorterArray = aLength > bLength ? b : a;\n  shorterArray.push(...placeholders(Math.abs(aLength - bLength)));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nexport default function alignArrays(a, b) {\n  const lcsSolution = longestCommonSubsequence(a, b);\n  if (lcsSolution) {\n    applySolution(lcsSolution, a, b);\n  } else {\n    applyPadding(a, b);\n  }\n}\n\n\n\n/** WEBPACK FOOTER **\n ** ./js/src/alignArrays.js\n **/"],"sourceRoot":""}");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n if (memo[aLength][bLength] === 0) {\n // No solution found\n return null;\n }\n return solution;\n}\n\nfunction placeholders(count) {\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n var changes = 0;\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n if (changes < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n } else if (changes > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(changes))));\n }\n ai--;\n bi--;\n changes = 0;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n changes++;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n changes--;\n }\n movement = solution[ai][bi];\n }\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n var aLength = a.length;\n var bLength = b.length;\n var shorterArray = aLength > bLength ? b : a;\n shorterArray.push.apply(shorterArray, _toConsumableArray(placeholders(Math.abs(aLength - bLength))));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n if (lcsSolution) {\n applySolution(lcsSolution, a, b);\n }\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/alignArrays.js?01b7"],"names":["alignArrays","MOVEMENT","none","diagonal","vertical","horizontal","initMatrix","height","width","rows","Array","i","length","Int32Array","longestCommonSubsequence","a","b","aLength","bLength","memo","solution","j","Math","max","placeholders","count","result","applySolution","movement","ai","bi","changes","splice","abs","applyPadding","shorterArray","push","lcsSolution"],"mappings":"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","file":"1.js","sourcesContent":["const MOVEMENT = {\n  none: 0,\n  diagonal: 1,\n  vertical: 2,\n  horizontal: 3,\n};\n\nfunction initMatrix(height, width) {\n  const rows = new Array(height);\n  for (let i = 0; i < rows.length; i++) {\n    rows[i] = new Int32Array(width);\n  }\n  return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n  // adapted from\n  // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n  const aLength = a.length;\n  const bLength = b.length;\n  const memo = initMatrix(aLength + 1, bLength + 1);\n  const solution = initMatrix(aLength + 1, bLength + 1);\n\n  // Loop and find the solution\n  for (let i = 1; i <= aLength; i++) {\n    for (let j = 1; j <= bLength; j++) {\n      if (a[i - 1] === b[j - 1]) {\n        // diagonal\n        memo[i][j] = memo[i - 1][j - 1] + 1;\n        solution[i][j] = MOVEMENT.diagonal;\n      } else {\n        memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n        if (memo[i][j] === memo[i - 1][j]) {\n          solution[i][j] = MOVEMENT.vertical;\n        } else {\n          solution[i][j] = MOVEMENT.horizontal;\n        }\n      }\n    }\n  }\n  if (memo[aLength][bLength] === 0) {\n    // No solution found\n    return null;\n  }\n  return solution;\n}\n\nfunction placeholders(count) {\n  const result = new Array(count);\n  for (let i = 0; i < count; i++) {\n    result[i] = '+';\n  }\n  return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n  let movement = solution[a.length][b.length];\n  let ai = a.length;\n  let bi = b.length;\n  let changes = 0;\n\n  while (movement !== MOVEMENT.none) {\n    if (movement === MOVEMENT.diagonal) {\n      if (changes < 0) {\n        b.splice(bi, 0, ...placeholders(Math.abs(changes)));\n      } else if (changes > 0) {\n        a.splice(ai, 0, ...placeholders(changes));\n      }\n      ai--;\n      bi--;\n      changes = 0;\n    } else if (movement === MOVEMENT.horizontal) {\n      bi--;\n      changes++;\n    } else if (movement === MOVEMENT.vertical) {\n      ai--;\n      changes--;\n    }\n    movement = solution[ai][bi];\n  }\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n  const aLength = a.length;\n  const bLength = b.length;\n  const shorterArray = aLength > bLength ? b : a;\n  shorterArray.push(...placeholders(Math.abs(aLength - bLength)));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nexport default function alignArrays(a, b) {\n  const lcsSolution = longestCommonSubsequence(a, b);\n  if (lcsSolution) {\n    applySolution(lcsSolution, a, b);\n  }\n}\n\n\n\n/** WEBPACK FOOTER **\n ** ./js/src/alignArrays.js\n **/"],"sourceRoot":""}");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n if (memo[aLength][bLength] === 0) {\n // No soltuion found\n return null;\n }\n return solution;\n}\n\nfunction placeholders(count) {\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n var changes = 0;\n\n function applyChange() {\n if (changes < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n } else if (changes > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(changes))));\n }\n changes = 0;\n }\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n applyChange();\n ai--;\n bi--;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n changes++;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n changes--;\n }\n movement = solution[ai][bi];\n }\n applyChange();\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n var aLength = a.length;\n var bLength = b.length;\n var shorterArray = aLength > bLength ? b : a;\n shorterArray.push.apply(shorterArray, _toConsumableArray(placeholders(Math.abs(aLength - bLength))));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n if (lcsSolution) {\n applySolution(lcsSolution, a, b);\n } else {\n applyPadding(a, b);\n }\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/alignArrays.js?01b7"],"names":["alignArrays","MOVEMENT","none","diagonal","vertical","horizontal","initMatrix","height","width","rows","Array","i","length","Int32Array","longestCommonSubsequence","a","b","aLength","bLength","memo","solution","j","Math","max","placeholders","count","result","applySolution","movement","ai","bi","changes","applyChange","splice","abs","applyPadding","shorterArray","push","lcsSolution"],"mappings":"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","file":"1.js","sourcesContent":["const MOVEMENT = {\n  none: 0,\n  diagonal: 1,\n  vertical: 2,\n  horizontal: 3,\n};\n\nfunction initMatrix(height, width) {\n  const rows = new Array(height);\n  for (let i = 0; i < rows.length; i++) {\n    rows[i] = new Int32Array(width);\n  }\n  return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n  // adapted from\n  // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n  const aLength = a.length;\n  const bLength = b.length;\n  const memo = initMatrix(aLength + 1, bLength + 1);\n  const solution = initMatrix(aLength + 1, bLength + 1);\n\n  // Loop and find the solution\n  for (let i = 1; i <= aLength; i++) {\n    for (let j = 1; j <= bLength; j++) {\n      if (a[i - 1] === b[j - 1]) {\n        // diagonal\n        memo[i][j] = memo[i - 1][j - 1] + 1;\n        solution[i][j] = MOVEMENT.diagonal;\n      } else {\n        memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n        if (memo[i][j] === memo[i - 1][j]) {\n          solution[i][j] = MOVEMENT.vertical;\n        } else {\n          solution[i][j] = MOVEMENT.horizontal;\n        }\n      }\n    }\n  }\n  if (memo[aLength][bLength] === 0) {\n    // No soltuion found\n    return null;\n  }\n  return solution;\n}\n\nfunction placeholders(count) {\n  const result = new Array(count);\n  for (let i = 0; i < count; i++) {\n    result[i] = '+';\n  }\n  return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n  let movement = solution[a.length][b.length];\n  let ai = a.length;\n  let bi = b.length;\n  let changes = 0;\n\n  function applyChange() {\n    if (changes < 0) {\n      b.splice(bi, 0, ...placeholders(Math.abs(changes)));\n    } else if (changes > 0) {\n      a.splice(ai, 0, ...placeholders(changes));\n    }\n    changes = 0;\n  }\n\n  while (movement !== MOVEMENT.none) {\n    if (movement === MOVEMENT.diagonal) {\n      applyChange();\n      ai--;\n      bi--;\n    } else if (movement === MOVEMENT.horizontal) {\n      bi--;\n      changes++;\n    } else if (movement === MOVEMENT.vertical) {\n      ai--;\n      changes--;\n    }\n    movement = solution[ai][bi];\n  }\n  applyChange();\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n  const aLength = a.length;\n  const bLength = b.length;\n  const shorterArray = aLength > bLength ? b : a;\n  shorterArray.push(...placeholders(Math.abs(aLength - bLength)));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nexport default function alignArrays(a, b) {\n  const lcsSolution = longestCommonSubsequence(a, b);\n  if (lcsSolution) {\n    applySolution(lcsSolution, a, b);\n  } else {\n    applyPadding(a, b);\n  }\n}\n\n\n\n/** WEBPACK FOOTER **\n ** ./js/src/alignArrays.js\n **/"],"sourceRoot":""}");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n console.log(memo);\n return solution;\n}\n\nfunction placeholders(count) {\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n\n var changes = 0;\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n console.log(changes);\n if (changes < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n } else if (changes > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n }\n ai--;\n bi--;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n changes++;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n changes--;\n }\n movement = solution[ai][bi];\n }\n}\n\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n applySolution(lcsSolution, a, b);\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvYWxpZ25BcnJheXMuanM/MDFiNyJdLCJuYW1lcyI6WyJhbGlnbkFycmF5cyIsIk1PVkVNRU5UIiwibm9uZSIsImRpYWdvbmFsIiwidmVydGljYWwiLCJob3Jpem9udGFsIiwiaW5pdE1hdHJpeCIsImhlaWdodCIsIndpZHRoIiwicm93cyIsIkFycmF5IiwiaSIsImxlbmd0aCIsIkludDMyQXJyYXkiLCJsb25nZXN0Q29tbW9uU3Vic2VxdWVuY2UiLCJhIiwiYiIsImFMZW5ndGgiLCJiTGVuZ3RoIiwibWVtbyIsInNvbHV0aW9uIiwiaiIsIk1hdGgiLCJtYXgiLCJjb25zb2xlIiwibG9nIiwicGxhY2Vob2xkZXJzIiwiY291bnQiLCJyZXN1bHQiLCJhcHBseVNvbHV0aW9uIiwibW92ZW1lbnQiLCJhaSIsImJpIiwiY2hhbmdlcyIsInNwbGljZSIsImFicyIsImxjc1NvbHV0aW9uIl0sIm1hcHBpbmdzIjoiOzs7OztrQkErRndCQSxXOzs7O0FBL0Z4QixJQUFNQyxXQUFXO0FBQ2ZDLFFBQU0sQ0FEUztBQUVmQyxZQUFVLENBRks7QUFHZkMsWUFBVSxDQUhLO0FBSWZDLGNBQVk7QUFKRyxDQUFqQjs7QUFPQSxTQUFTQyxVQUFULENBQW9CQyxNQUFwQixFQUE0QkMsS0FBNUIsRUFBbUM7QUFDakMsTUFBTUMsT0FBTyxJQUFJQyxLQUFKLENBQVVILE1BQVYsQ0FBYjtBQUNBLE9BQUssSUFBSUksSUFBSSxDQUFiLEVBQWdCQSxJQUFJRixLQUFLRyxNQUF6QixFQUFpQ0QsR0FBakMsRUFBc0M7QUFDcENGLFNBQUtFLENBQUwsSUFBVSxJQUFJRSxVQUFKLENBQWVMLEtBQWYsQ0FBVjtBQUNEO0FBQ0QsU0FBT0MsSUFBUDtBQUNEOztBQUVELFNBQVNLLHdCQUFULENBQWtDQyxDQUFsQyxFQUFxQ0MsQ0FBckMsRUFBd0M7QUFDdEM7QUFDQTtBQUNBLE1BQU1DLFVBQVVGLEVBQUVILE1BQWxCO0FBQ0EsTUFBTU0sVUFBVUYsRUFBRUos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Q0FBWDtBQUNEO0FBQ0Y7O0FBRUQ7Ozs7Ozs7O0FBUWUsU0FBU2hDLFdBQVQsQ0FBcUJlLENBQXJCLEVBQXdCQyxDQUF4QixFQUEyQjtBQUN4QyxNQUFNb0IsY0FBY3RCLHlCQUF5QkMsQ0FBekIsRUFBNEJDLENBQTVCLENBQXBCO0FBQ0FhLGdCQUFjTyxXQUFkLEVBQTJCckIsQ0FBM0IsRUFBOEJDLENBQTlCO0FBQ0QiLCJmaWxlIjoiMS5qcyIsInNvdXJjZXNDb250ZW50IjpbImNvbnN0IE1PVkVNRU5UID0ge1xuICBub25lOiAwLFxuICBkaWFnb25hbDogMSxcbiAgdmVydGljYWw6IDIsXG4gIGhvcml6b250YWw6IDMsXG59O1xuXG5mdW5jdGlvbiBpbml0TWF0cml4KGhlaWdodCwgd2lkdGgpIHtcbiAgY29uc3Qgcm93cyA9IG5ldyBBcnJheShoZWlnaHQpO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IHJvd3MubGVuZ3RoOyBpKyspIHtcbiAgICByb3dzW2ldID0gbmV3IEludDMyQXJyYXkod2lkdGgpO1xuICB9XG4gIHJldHVybiByb3dzO1xufVxuXG5mdW5jdGlvbiBsb25nZXN0Q29tbW9uU3Vic2VxdWVuY2UoYSwgYikge1xuICAvLyBhZGFwdGVkIGZyb21cbiAgLy8gaHR0cDovL2FsZ29yaXRobXMudHV0b3JpYWxob3Jpem9uLmNvbS9keW5hbWljLXByb2dyYW1taW5nLWxvbmdlc3QtY29tbW9uLXN1YnNlcXVlbmNlL1xuICBjb25zdCBhTGVuZ3RoID0gYS5sZW5ndGg7XG4gIGNvbnN0IGJMZW5ndGggPSBiLmxlbmd0aDtcbiAgY29uc3QgbWVtbyA9IGluaXRNYXRyaXgoYUxlbmd0aCArIDEsIGJMZW5ndGggKyAxKTtcbiAgY29uc3Qgc29sdXRpb24gPSBpbml0TWF0cml4KGFMZW5ndGggKyAxLCBiTGVuZ3RoICsgMSk7XG5cbiAgLy8gTG9vcCBhbmQgZmluZCB0aGUgc29sdXRpb25cbiAgZm9yIChsZXQgaSA9IDE7IGkgPD0gYUxlbmd0aDsgaSsrKSB7XG4gICAgZm9yIChsZXQgaiA9IDE7IGogPD0gYkxlbmd0aDsgaisrKSB7XG4gICAgICBpZiAoYVtpIC0gMV0gPT09IGJbaiAtIDFdKSB7XG4gICAgICAgIC8vIGRpYWdvbmFsXG4gICAgICAgIG1lbW9baV1bal0gPSBtZW1vW2kgLSAxXVtqIC0gMV0gKyAxO1xuICAgICAgICBzb2x1dGlvbltpXVtqXSA9IE1PVkVNRU5ULmRpYWdvbmFsO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgbWVtb1tpXVtqXSA9IE1hdGgubWF4KG1lbW9baSAtIDFdW2pdLCBtZW1vW2ldW2ogLSAxXSk7XG4gICAgICAgIGlmIChtZW1vW2ldW2pdID09PSBtZW1vW2kgLSAxXVtqXSkge1xuICAgICAgICAgIHNvbHV0aW9uW2ldW2pdID0gTU9WRU1FTlQudmVydGljYWw7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgc29sdXRpb25baV1bal0gPSBNT1ZFTUVOVC5ob3Jpem9udGFsO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG4gIGNvbnNvbGUubG9nKG1lbW8pXG4gIHJldHVybiBzb2x1dGlvbjtcbn1cblxuZnVuY3Rpb24gcGxhY2Vob2xkZXJzKGNvdW50KSB7XG4gIGNvbnN0IHJlc3VsdCA9IG5ldyBBcnJheShjb3VudCk7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgY291bnQ7IGkrKykge1xuICAgIHJlc3VsdFtpXSA9ICcrJztcbiAgfVxuICByZXR1cm4gcmVzdWx0O1xufVxuLyoqXG4gKiBBcHBseSBhbiBsY3Mgc29sdXRpb24gdG8gYXJyYXlzLiBOb3RlIHRoYXQgdGhpcyB3aWxsIE1VVEFURSB0aGUgYXJyYXlzLFxuICogaW5qZWN0aW5nIFwiK1wiIHdoZXJlIGdhcHMgYXJlIG5lZWRlZC5cbiAqXG4gKiBAcGFyYW0ge0FycmF5PEFycmF5Pn0gc29sdXRpb24gYXMgY29tcHV0ZWQgYnkgYGxvbmdlc3RDb21tb25TdWJzZXF1ZW5jZWBcbiAqIEBwYXJhbSB7QXJyYXl9IGFcbiAqIEBwYXJhbSB7QXJyYXl9IGJcbiAqL1xuZnVuY3Rpb24gYXBwbHlTb2x1dGlvbihzb2x1dGlvbiwgYSwgYikge1xuICBsZXQgbW92ZW1lbnQgPSBzb2x1dGlvblthLmxlbmd0aF1bYi5sZW5ndGhdO1xuICBsZXQgYWkgPSBhLmxlbmd0aDtcbiAgbGV0IGJpID0gYi5sZW5ndGg7XG5cbiAgbGV0IGNoYW5nZXMgPSAwO1xuXG4gIHdoaWxlIChtb3ZlbWVudCAhPT0gTU9WRU1FTlQubm9uZSkge1xuICAgIGlmIChtb3ZlbWVudCA9PT0gTU9WRU1FTlQuZGlhZ29uYWwpIHtcbiAgICAgIGNvbnNvbGUubG9nKGNoYW5nZXMpXG4gICAgICBpZiAoY2hhbmdlcyA8IDApIHtcbiAgICAgICAgYi5zcGxpY2UoYmksIDAsIC4uLnBsYWNlaG9sZGVycyhNYXRoLmFicyhjaGFuZ2VzKSkpO1xuICAgICAgfSBlbHNlIGlmIChjaGFuZ2VzID4gMCkge1xuICAgICAgICBhLnNwbGljZShhaSwgMCwgLi4ucGxhY2Vob2xkZXJzKE1hdGguYWJzKGNoYW5nZXMpKSk7XG4gICAgICB9XG4gICAgICBhaS0tO1xuICAgICAgYmktLTtcbiAgICB9IGVsc2UgaWYgKG1vdmVtZW50ID09PSBNT1ZFTUVOVC5ob3Jpem9udGFsKSB7XG4gICAgICBiaS0tO1xuICAgICAgY2hhbmdlcysrO1xuICAgIH0gZWxzZSBpZiAobW92ZW1lbnQgPT09IE1PVkVNRU5ULnZlcnRpY2FsKSB7XG4gICAgICBhaS0tO1xuICAgICAgY2hhbmdlcy0tO1xuICAgIH1cbiAgICBtb3ZlbWVudCA9IHNvbHV0aW9uW2FpXVtiaV07XG4gIH1cbn1cblxuLyoqXG4gKiBDb21wdXRlcyB0aGUgbG9uZ2VzdCBjb21tb24gc3Vic2VxdWVuY2Ugb2YgdHdvIGFycmF5cywgdGhlbiB1c2VzIHRoYXRcbiAqIHNvbHV0aW9uIHRvIGluamVjdCBnYXBzIGludG8gdGhlIGFycmF5cywgbWFraW5nIHRoZW0gYWxpZ24gb24gY29tbW9uXG4gKiBzdWJzZXF1ZW5jZXMuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gYVxuICogQHBhcmFtIHtBcnJheX0gYlxuICovXG5leHBvcnQgZGVmYXVsdCBmdW5jdGlvbiBhbGlnbkFycmF5cyhhLCBiKSB7XG4gIGNvbnN0IGxjc1NvbHV0aW9uID0gbG9uZ2VzdENvbW1vblN1YnNlcXVlbmNlKGEsIGIpO1xuICBhcHBseVNvbHV0aW9uKGxjc1NvbHV0aW9uLCBhLCBiKTtcbn1cblxuXG5cbi8qKiBXRUJQQUNLIEZPT1RFUiAqKlxuICoqIC4vanMvc3JjL2FsaWduQXJyYXlzLmpzXG4gKiovIl0sInNvdXJjZVJvb3QiOiIifQ==");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _adiff = __webpack_require__(1);\n\nvar _adiff2 = _interopRequireDefault(_adiff);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n alignArrays(hashedPreviousData, hashedCurrentData);\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n var adiffResults = getAdiffResults({\n previousData: previousData,\n currentData: currentData,\n previousImageData: previousImageData,\n currentImageData: currentImageData\n });\n\n self.postMessage({ progress: 85 });\n\n // iterate and apply changes to previous data\n adiffResults.forEach(function (instruction) {\n var atIndex = instruction[0];\n var deletedItems = instruction[1];\n var addedItems = instruction.length - 2;\n\n for (var y = 0; y < Math.max(deletedItems, addedItems); y++) {\n if (y < deletedItems) {\n // ignore, we just keep the old line\n } else {\n previousImageData.splice(atIndex + y, 0, transparentLine);\n }\n }\n });\n self.postMessage({ progress: 95 });\n\n // iterate backwards and apply changes to current data\n for (var i = adiffResults.length - 1; i >= 0; i--) {\n var instruction = adiffResults[i];\n var atIndex = instruction[0];\n var deletedItems = instruction[1];\n var addedItems = instruction.length - 2;\n\n for (var y = 0; y < Math.max(deletedItems, addedItems); y++) {\n if (y < addedItems) {\n // ignore, we just keep the old line\n } else {\n currentImageData.splice(atIndex + y, 0, transparentLine);\n }\n }\n }\n self.postMessage({ progress: 98 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/workers/ComputeAndInjectDiffsWorker.js?ddb5"],"names":["imageTo2DArray","paddingRight","data","width","height","rowSize","newData","row","pixelsInRow","Uint8ClampedArray","location","push","align","previousImageData","currentImageData","hashedPreviousData","map","JSON","stringify","self","postMessage","progress","hashedCurrentData","alignArrays","computeAndInjectDiffs","previousData","currentData","maxWidth","Math","max","transparentLine","adiffResults","getAdiffResults","forEach","instruction","atIndex","deletedItems","addedItems","length","y","splice","i","addEventListener","result","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\nfunction imageTo2DArray({ data, width, height }, paddingRight) {\n  // The imageData is a 1D array. Each element in the array corresponds to a\n  // decimal value that represents one of the RGBA channels for that pixel.\n  const rowSize = width * 4;\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = new Uint8ClampedArray(rowSize + (paddingRight * 4));\n    for (let location = 0; location < rowSize; location++) {\n      pixelsInRow[location] = data[(row * rowSize) + location];\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction align({\n  previousImageData,\n  currentImageData,\n}) {\n  const hashedPreviousData = previousImageData.map(JSON.stringify);\n  self.postMessage({ progress: 40 });\n  const hashedCurrentData = currentImageData.map(JSON.stringify);\n  self.postMessage({ progress: 60 });\n\n  alignArrays(\n    hashedPreviousData,\n    hashedCurrentData\n  );\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs({ previousData, currentData }) {\n  const maxWidth = Math.max(previousData.width, currentData.width);\n\n  const transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n  const previousImageData = imageTo2DArray(\n    previousData, maxWidth - previousData.width);\n\n  const currentImageData = imageTo2DArray(\n    currentData, maxWidth - currentData.width);\n\n  self.postMessage({ progress: 20 });\n\n  const adiffResults = getAdiffResults({\n    previousData,\n    currentData,\n    previousImageData,\n    currentImageData,\n  });\n\n  self.postMessage({ progress: 85 });\n\n  // iterate and apply changes to previous data\n  adiffResults.forEach((instruction) => {\n    const atIndex = instruction[0];\n    const deletedItems = instruction[1];\n    const addedItems = instruction.length - 2;\n\n    for (let y = 0; y < Math.max(deletedItems, addedItems); y++) {\n      if (y < deletedItems) {\n        // ignore, we just keep the old line\n      } else {\n        previousImageData.splice(atIndex + y, 0, transparentLine);\n      }\n    }\n  });\n  self.postMessage({ progress: 95 });\n\n  // iterate backwards and apply changes to current data\n  for (let i = adiffResults.length - 1; i >= 0; i--) {\n    const instruction = adiffResults[i];\n    const atIndex = instruction[0];\n    const deletedItems = instruction[1];\n    const addedItems = instruction.length - 2;\n\n    for (let y = 0; y < Math.max(deletedItems, addedItems); y++) {\n      if (y < addedItems) {\n        // ignore, we just keep the old line\n      } else {\n        currentImageData.splice(atIndex + y, 0, transparentLine);\n      }\n    }\n  }\n  self.postMessage({ progress: 98 });\n\n  return {\n    currentData: {\n      data: currentImageData,\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: previousImageData,\n      height: previousImageData.length,\n      width: maxWidth,\n    },\n  };\n}\n\nself.addEventListener('message', ({ data: { previousData, currentData } }) => {\n  const result = computeAndInjectDiffs({ previousData, currentData });\n  self.postMessage(result);\n  self.close();\n});\n\n\n\n/** WEBPACK FOOTER **\n ** ./js/src/workers/ComputeAndInjectDiffsWorker.js\n **/"],"sourceRoot":""}");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("function head (a) {\n return a[0]\n}\n\nfunction last (a) {\n return a[a.length - 1]\n}\n\nfunction tail(a) {\n return a.slice(1)\n}\n\nfunction retreat (e) {\n return e.pop()\n}\n\nfunction hasLength (e) {\n return e.length\n}\n\nfunction any(ary, test) {\n for(var i=0;i<ary.length;i++)\n if(test(ary[i]))\n return true\n return false\n}\n\nfunction score (a) {\n return a.reduce(function (s, a) {\n return s + a.length + a[1] + 1\n }, 0)\n}\n\nfunction best (a, b) {\n return score(a) <= score(b) ? a : b\n}\n\n\nvar _rules // set at the bottom \n\n// note, naive implementation. will break on circular objects.\n\nfunction _equal(a, b) {\n if(a && !b) return false\n if(Array.isArray(a))\n if(a.length != b.length) return false\n if(a && 'object' == typeof a) {\n for(var i in a)\n if(!_equal(a[i], b[i])) return false\n for(var i in b)\n if(!_equal(a[i], b[i])) return false\n return true\n }\n return a == b\n}\n\nfunction getArgs(args) {\n return args.length == 1 ? args[0] : [].slice.call(args)\n}\n\n// return the index of the element not like the others, or -1\nfunction oddElement(ary, cmp) {\n var c\n function guess(a) {\n var odd = -1\n c = 0\n for (var i = a; i < ary.length; i ++) {\n if(!cmp(ary[a], ary[i])) {\n odd = i, c++\n }\n }\n return c > 1 ? -1 : odd\n }\n //assume that it is the first element.\n var g = guess(0)\n if(-1 != g) return g\n //0 was the odd one, then all the other elements are equal\n //else there more than one different element\n guess(1)\n return c == 0 ? 0 : -1\n}\nvar exports = module.exports = function (deps, exports) {\n var equal = (deps && deps.equal) || _equal\n exports = exports || {} \n exports.lcs = \n function lcs() {\n var cache = {}\n var args = getArgs(arguments)\n var a = args[0], b = args[1]\n\n function key (a,b){\n return a.length + ':' + b.length\n }\n\n //find length that matches at the head\n\n if(args.length > 2) {\n //if called with multiple sequences\n //recurse, since lcs(a, b, c, d) == lcs(lcs(a,b), lcs(c,d))\n args.push(lcs(args.shift(), args.shift()))\n return lcs(args)\n }\n \n //this would be improved by truncating input first\n //and not returning an lcs as an intermediate step.\n //untill that is a performance problem.\n\n var start = 0, end = 0\n for(var i = 0; i < a.length && i < b.length \n && equal(a[i], b[i])\n ; i ++\n )\n start = i + 1\n\n if(a.length === start)\n return a.slice()\n\n for(var i = 0; i < a.length - start && i < b.length - start\n && equal(a[a.length - 1 - i], b[b.length - 1 - i])\n ; i ++\n )\n end = i\n\n function recurse (a, b) {\n if(!a.length || !b.length) return []\n //avoid exponential time by caching the results\n if(cache[key(a, b)]) return cache[key(a, b)]\n\n if(equal(a[0], b[0]))\n return [head(a)].concat(recurse(tail(a), tail(b)))\n else { \n var _a = recurse(tail(a), b)\n var _b = recurse(a, tail(b))\n return cache[key(a,b)] = _a.length > _b.length ? _a : _b \n }\n }\n \n var middleA = a.slice(start, a.length - end)\n var middleB = b.slice(start, b.length - end)\n\n return (\n a.slice(0, start).concat(\n recurse(middleA, middleB)\n ).concat(a.slice(a.length - end))\n )\n }\n\n // given n sequences, calc the lcs, and then chunk strings into stable and unstable sections.\n // unstable chunks are passed to build\n exports.chunk =\n function (q, build) {\n var q = q.map(function (e) { return e.slice() })\n var lcs = exports.lcs.apply(null, q)\n var all = [lcs].concat(q)\n\n function matchLcs (e) {\n if(e.length && !lcs.length || !e.length && lcs.length)\n return false //incase the last item is null\n return equal(last(e), last(lcs)) || ((e.length + lcs.length) === 0)\n }\n\n while(any(q, hasLength)) {\n //if each element is at the lcs then this chunk is stable.\n while(q.every(matchLcs) && q.every(hasLength))\n all.forEach(retreat)\n //collect the changes in each array upto the next match with the lcs\n var c = false\n var unstable = q.map(function (e) {\n var change = []\n while(!matchLcs(e)) {\n change.unshift(retreat(e))\n c = true\n }\n return change\n })\n if(c) build(q[0].length, unstable)\n }\n }\n\n //calculate a diff this is only updates\n exports.optimisticDiff =\n function (a, b) {\n var M = Math.max(a.length, b.length)\n var m = Math.min(a.length, b.length)\n var patch = []\n for(var i = 0; i < M; i++)\n if(a[i] !== b[i]) {\n var cur = [i,0], deletes = 0\n while(a[i] !== b[i] && i < m) {\n cur[1] = ++deletes\n cur.push(b[i++])\n }\n //the rest are deletes or inserts\n if(i >= m) {\n //the rest are deletes\n if(a.length > b.length)\n cur[1] += a.length - b.length\n //the rest are inserts\n else if(a.length < b.length)\n cur = cur.concat(b.slice(a.length))\n }\n patch.push(cur)\n }\n\n return patch\n }\n\n exports.diff =\n function (a, b) {\n var optimistic = exports.optimisticDiff(a, b)\n var changes = []\n exports.chunk([a, b], function (index, unstable) {\n var del = unstable.shift().length\n var insert = unstable.shift()\n changes.push([index, del].concat(insert))\n })\n return best(optimistic, changes)\n }\n\n exports.patch = function (a, changes, mutate) {\n if(mutate !== true) a = a.slice(a)//copy a\n changes.forEach(function (change) {\n [].splice.apply(a, change)\n })\n return a\n }\n\n // http://en.wikipedia.org/wiki/Concestor\n // me, concestor, you...\n exports.merge = function () {\n var args = getArgs(arguments)\n var patch = exports.diff3(args)\n return exports.patch(args[0], patch)\n }\n\n exports.diff3 = function () {\n var args = getArgs(arguments)\n var r = []\n exports.chunk(args, function (index, unstable) {\n var mine = unstable[0]\n var insert = resolve(unstable)\n if(equal(mine, insert)) return \n r.push([index, mine.length].concat(insert)) \n })\n return r\n }\n exports.oddOneOut =\n function oddOneOut (changes) {\n changes = changes.slice()\n //put the concestor first\n changes.unshift(changes.splice(1,1)[0])\n var i = oddElement(changes, equal)\n if(i == 0) // concestor was different, 'false conflict'\n return changes[1]\n if (~i)\n return changes[i] \n }\n exports.insertMergeOverDelete = \n //i've implemented this as a seperate rule,\n //because I had second thoughts about this.\n function insertMergeOverDelete (changes) {\n changes = changes.slice()\n changes.splice(1,1)// remove concestor\n \n //if there is only one non empty change thats okay.\n //else full confilct\n for (var i = 0, nonempty; i < changes.length; i++)\n if(changes[i].length) \n if(!nonempty) nonempty = changes[i]\n else return // full conflict\n return nonempty\n }\n\n var rules = (deps && deps.rules) || [exports.oddOneOut, exports.insertMergeOverDelete]\n\n function resolve (changes) {\n var l = rules.length\n for (var i in rules) { // first\n \n var c = rules[i] && rules[i](changes)\n if(c) return c\n }\n changes.splice(1,1) // remove concestor\n //returning the conflicts as an object is a really bad idea,\n // because == will not detect they are the same. and conflicts build.\n // better to use\n // '<<<<<<<<<<<<<'\n // of course, i wrote this before i started on snob, so i didn't know that then.\n /*var conflict = ['>>>>>>>>>>>>>>>>']\n while(changes.length)\n conflict = conflict.concat(changes.shift()).concat('============')\n conflict.pop()\n conflict.push ('<<<<<<<<<<<<<<<')\n changes.unshift ('>>>>>>>>>>>>>>>')\n return conflict*/\n //nah, better is just to use an equal can handle objects\n return {'?': changes}\n }\n return exports\n}\nexports(null, exports)\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./~/adiff/index.js?04d6"],"names":[],"mappings":"AAAA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,cAAc,aAAa;AAC3B;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;;;AAGA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,gBAAgB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,kBAAkB;AAClB;AACA,OAAO;AACP;AACA;;AAEA;AACA;;AAEA,kBAAkB;AAClB;AACA,OAAO;AACP;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,Y;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,gCAAgC,mBAAmB;AACnD;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,OAAO;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,+BAA+B,oBAAoB;AACnD;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA,0BAA0B;;AAE1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA","file":"1.js","sourcesContent":["function head (a) {\n  return a[0]\n}\n\nfunction last (a) {\n  return a[a.length - 1]\n}\n\nfunction tail(a) {\n  return a.slice(1)\n}\n\nfunction retreat (e) {\n  return e.pop()\n}\n\nfunction hasLength (e) {\n  return e.length\n}\n\nfunction any(ary, test) {\n  for(var i=0;i<ary.length;i++)\n    if(test(ary[i]))\n      return true\n  return false\n}\n\nfunction score (a) {\n  return a.reduce(function (s, a) {\n      return s + a.length + a[1] + 1\n  }, 0)\n}\n\nfunction best (a, b) {\n  return score(a) <= score(b) ? a : b\n}\n\n\nvar _rules // set at the bottom  \n\n// note, naive implementation. will break on circular objects.\n\nfunction _equal(a, b) {\n  if(a && !b) return false\n  if(Array.isArray(a))\n    if(a.length != b.length) return false\n  if(a && 'object' == typeof a) {\n    for(var i in a)\n      if(!_equal(a[i], b[i])) return false\n    for(var i in b)\n      if(!_equal(a[i], b[i])) return false\n    return true\n  }\n  return a == b\n}\n\nfunction getArgs(args) {\n  return args.length == 1 ? args[0] : [].slice.call(args)\n}\n\n// return the index of the element not like the others, or -1\nfunction oddElement(ary, cmp) {\n  var c\n  function guess(a) {\n    var odd = -1\n    c = 0\n    for (var i = a; i < ary.length; i ++) {\n      if(!cmp(ary[a], ary[i])) {\n        odd = i, c++\n      }\n    }\n    return c > 1 ? -1 : odd\n  }\n  //assume that it is the first element.\n  var g = guess(0)\n  if(-1 != g) return g\n  //0 was the odd one, then all the other elements are equal\n  //else there more than one different element\n  guess(1)\n  return c == 0 ? 0 : -1\n}\nvar exports = module.exports = function (deps, exports) {\n  var equal = (deps && deps.equal) || _equal\n  exports = exports || {} \n  exports.lcs = \n  function lcs() {\n    var cache = {}\n    var args = getArgs(arguments)\n    var a = args[0], b = args[1]\n\n    function key (a,b){\n      return a.length + ':' + b.length\n    }\n\n    //find length that matches at the head\n\n    if(args.length > 2) {\n      //if called with multiple sequences\n      //recurse, since lcs(a, b, c, d) == lcs(lcs(a,b), lcs(c,d))\n      args.push(lcs(args.shift(), args.shift()))\n      return lcs(args)\n    }\n    \n    //this would be improved by truncating input first\n    //and not returning an lcs as an intermediate step.\n    //untill that is a performance problem.\n\n    var start = 0, end = 0\n    for(var i = 0; i < a.length && i < b.length \n      && equal(a[i], b[i])\n      ; i ++\n    )\n      start = i + 1\n\n    if(a.length === start)\n      return a.slice()\n\n    for(var i = 0;  i < a.length - start && i < b.length - start\n      && equal(a[a.length - 1 - i], b[b.length - 1 - i])\n      ; i ++\n    )\n      end = i\n\n    function recurse (a, b) {\n      if(!a.length || !b.length) return []\n      //avoid exponential time by caching the results\n      if(cache[key(a, b)]) return cache[key(a, b)]\n\n      if(equal(a[0], b[0]))\n        return [head(a)].concat(recurse(tail(a), tail(b)))\n      else { \n        var _a = recurse(tail(a), b)\n        var _b = recurse(a, tail(b))\n        return cache[key(a,b)] = _a.length > _b.length ? _a : _b  \n      }\n    }\n    \n    var middleA = a.slice(start, a.length - end)\n    var middleB = b.slice(start, b.length - end)\n\n    return (\n      a.slice(0, start).concat(\n        recurse(middleA, middleB)\n      ).concat(a.slice(a.length - end))\n    )\n  }\n\n  // given n sequences, calc the lcs, and then chunk strings into stable and unstable sections.\n  // unstable chunks are passed to build\n  exports.chunk =\n  function (q, build) {\n    var q = q.map(function (e) { return e.slice() })\n    var lcs = exports.lcs.apply(null, q)\n    var all = [lcs].concat(q)\n\n    function matchLcs (e) {\n      if(e.length && !lcs.length || !e.length && lcs.length)\n        return false //incase the last item is null\n      return equal(last(e), last(lcs)) || ((e.length + lcs.length) === 0)\n    }\n\n    while(any(q, hasLength)) {\n      //if each element is at the lcs then this chunk is stable.\n      while(q.every(matchLcs) && q.every(hasLength))\n        all.forEach(retreat)\n      //collect the changes in each array upto the next match with the lcs\n      var c = false\n      var unstable = q.map(function (e) {\n        var change = []\n        while(!matchLcs(e)) {\n          change.unshift(retreat(e))\n          c = true\n        }\n        return change\n      })\n      if(c) build(q[0].length, unstable)\n    }\n  }\n\n  //calculate a diff this is only updates\n  exports.optimisticDiff =\n  function (a, b) {\n    var M = Math.max(a.length, b.length)\n    var m = Math.min(a.length, b.length)\n    var patch = []\n    for(var i = 0; i < M; i++)\n      if(a[i] !== b[i]) {\n        var cur = [i,0], deletes = 0\n        while(a[i] !== b[i] && i < m) {\n          cur[1] = ++deletes\n          cur.push(b[i++])\n        }\n        //the rest are deletes or inserts\n        if(i >= m) {\n          //the rest are deletes\n          if(a.length > b.length)\n            cur[1] += a.length - b.length\n          //the rest are inserts\n          else if(a.length < b.length)\n            cur = cur.concat(b.slice(a.length))\n        }\n        patch.push(cur)\n      }\n\n    return patch\n  }\n\n  exports.diff =\n  function (a, b) {\n    var optimistic = exports.optimisticDiff(a, b)\n    var changes = []\n    exports.chunk([a, b], function (index, unstable) {\n      var del = unstable.shift().length\n      var insert = unstable.shift()\n      changes.push([index, del].concat(insert))\n    })\n    return best(optimistic, changes)\n  }\n\n  exports.patch = function (a, changes, mutate) {\n    if(mutate !== true) a = a.slice(a)//copy a\n    changes.forEach(function (change) {\n      [].splice.apply(a, change)\n    })\n    return a\n  }\n\n  // http://en.wikipedia.org/wiki/Concestor\n  // me, concestor, you...\n  exports.merge = function () {\n    var args = getArgs(arguments)\n    var patch = exports.diff3(args)\n    return exports.patch(args[0], patch)\n  }\n\n  exports.diff3 = function () {\n    var args = getArgs(arguments)\n    var r = []\n    exports.chunk(args, function (index, unstable) {\n      var mine = unstable[0]\n      var insert = resolve(unstable)\n      if(equal(mine, insert)) return \n      r.push([index, mine.length].concat(insert)) \n    })\n    return r\n  }\n  exports.oddOneOut =\n    function oddOneOut (changes) {\n      changes = changes.slice()\n      //put the concestor first\n      changes.unshift(changes.splice(1,1)[0])\n      var i = oddElement(changes, equal)\n      if(i == 0) // concestor was different, 'false conflict'\n        return changes[1]\n      if (~i)\n        return changes[i] \n    }\n  exports.insertMergeOverDelete = \n    //i've implemented this as a seperate rule,\n    //because I had second thoughts about this.\n    function insertMergeOverDelete (changes) {\n      changes = changes.slice()\n      changes.splice(1,1)// remove concestor\n      \n      //if there is only one non empty change thats okay.\n      //else full confilct\n      for (var i = 0, nonempty; i < changes.length; i++)\n        if(changes[i].length) \n          if(!nonempty) nonempty = changes[i]\n          else return // full conflict\n      return nonempty\n    }\n\n  var rules = (deps && deps.rules) || [exports.oddOneOut, exports.insertMergeOverDelete]\n\n  function resolve (changes) {\n    var l = rules.length\n    for (var i in rules) { // first\n      \n      var c = rules[i] && rules[i](changes)\n      if(c) return c\n    }\n    changes.splice(1,1) // remove concestor\n    //returning the conflicts as an object is a really bad idea,\n    // because == will not detect they are the same. and conflicts build.\n    // better to use\n    // '<<<<<<<<<<<<<'\n    // of course, i wrote this before i started on snob, so i didn't know that then.\n    /*var conflict = ['>>>>>>>>>>>>>>>>']\n    while(changes.length)\n      conflict = conflict.concat(changes.shift()).concat('============')\n    conflict.pop()\n    conflict.push          ('<<<<<<<<<<<<<<<')\n    changes.unshift       ('>>>>>>>>>>>>>>>')\n    return conflict*/\n    //nah, better is just to use an equal can handle objects\n    return {'?': changes}\n  }\n  return exports\n}\nexports(null, exports)\n\n\n\n/*****************\n ** WEBPACK FOOTER\n ** ./~/adiff/index.js\n ** module id = 1\n ** module chunks = 0\n **/"],"sourceRoot":""}");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|
@@ -0,0 +1,56 @@
|
|
1
|
+
/******/ (function(modules) { // webpackBootstrap
|
2
|
+
/******/ // The module cache
|
3
|
+
/******/ var installedModules = {};
|
4
|
+
|
5
|
+
/******/ // The require function
|
6
|
+
/******/ function __webpack_require__(moduleId) {
|
7
|
+
|
8
|
+
/******/ // Check if module is in cache
|
9
|
+
/******/ if(installedModules[moduleId])
|
10
|
+
/******/ return installedModules[moduleId].exports;
|
11
|
+
|
12
|
+
/******/ // Create a new module (and put it into the cache)
|
13
|
+
/******/ var module = installedModules[moduleId] = {
|
14
|
+
/******/ exports: {},
|
15
|
+
/******/ id: moduleId,
|
16
|
+
/******/ loaded: false
|
17
|
+
/******/ };
|
18
|
+
|
19
|
+
/******/ // Execute the module function
|
20
|
+
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
21
|
+
|
22
|
+
/******/ // Flag the module as loaded
|
23
|
+
/******/ module.loaded = true;
|
24
|
+
|
25
|
+
/******/ // Return the exports of the module
|
26
|
+
/******/ return module.exports;
|
27
|
+
/******/ }
|
28
|
+
|
29
|
+
|
30
|
+
/******/ // expose the modules object (__webpack_modules__)
|
31
|
+
/******/ __webpack_require__.m = modules;
|
32
|
+
|
33
|
+
/******/ // expose the module cache
|
34
|
+
/******/ __webpack_require__.c = installedModules;
|
35
|
+
|
36
|
+
/******/ // __webpack_public_path__
|
37
|
+
/******/ __webpack_require__.p = "";
|
38
|
+
|
39
|
+
/******/ // Load entry module and return exports
|
40
|
+
/******/ return __webpack_require__(0);
|
41
|
+
/******/ })
|
42
|
+
/************************************************************************/
|
43
|
+
/******/ ([
|
44
|
+
/* 0 */
|
45
|
+
/***/ function(module, exports, __webpack_require__) {
|
46
|
+
|
47
|
+
eval("'use strict';\n\nvar _alignArrays = __webpack_require__(1);\n\nvar _alignArrays2 = _interopRequireDefault(_alignArrays);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction imageTo2DArray(_ref, paddingRight) {\n var data = _ref.data;\n var width = _ref.width;\n var height = _ref.height;\n\n // The imageData is a 1D array. Each element in the array corresponds to a\n // decimal value that represents one of the RGBA channels for that pixel.\n var rowSize = width * 4;\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = new Uint8ClampedArray(rowSize + paddingRight * 4);\n for (var location = 0; location < rowSize; location++) {\n pixelsInRow[location] = data[row * rowSize + location];\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction align(_ref2) {\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n var maxWidth = _ref2.maxWidth;\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n self.postMessage({ progress: 40 });\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n self.postMessage({ progress: 60 });\n\n (0, _alignArrays2.default)(hashedPreviousData, hashedCurrentData);\n\n var transparentLine = new Uint8ClampedArray(maxWidth * 4);\n\n hashedPreviousData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n previousImageData.splice(i, 0, transparentLine);\n }\n });\n\n hashedCurrentData.forEach(function (hashedLine, i) {\n if (hashedLine === '+') {\n currentImageData.splice(i, 0, transparentLine);\n }\n });\n}\n\n/**\n * Takes two 2d images, computes the diff between the two, and injects pixels to\n * both in order to:\n * a) make both images the same height\n * b) properly visualize differences\n *\n * Please note that this method MUTATES data.\n *\n * @param {Array} previousData\n * @param {Array} currentData\n */\nfunction computeAndInjectDiffs(_ref3) {\n var previousData = _ref3.previousData;\n var currentData = _ref3.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 20 });\n\n align({\n previousImageData: previousImageData,\n currentImageData: currentImageData,\n maxWidth: maxWidth\n });\n\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: currentImageData,\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: previousImageData,\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref4) {\n var _ref4$data = _ref4.data;\n var previousData = _ref4$data.previousData;\n var currentData = _ref4$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage(result);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9Db21wdXRlQW5kSW5qZWN0RGlmZnNXb3JrZXIuanM/ZGRiNSJdLCJuYW1lcyI6WyJpbWFnZVRvMkRBcnJheSIsInBhZGRpbmdSaWdodCIsImRhdGEiLCJ3aWR0aCIsImhlaWdodCIsInJvd1NpemUiLCJuZXdEYXRhIiwicm93IiwicGl4ZWxzSW5Sb3ciLCJVaW50OENsYW1wZWRBcnJheSIsImxvY2F0aW9uIiwicHVzaCIsImFsaWduIiwicHJldmlvdXNJbWFnZURhdGEiLCJjdXJyZW50SW1hZ2VEYXRhIiwibWF4V2lkdGgiLCJoYXNoZWRQcmV2aW91c0RhdGEiLCJtYXAiLCJKU09OIiwic3RyaW5naWZ5Iiwic2VsZiIsInBvc3RNZXNzYWdlIiwicHJvZ3Jlc3MiLCJoYXNoZWRDdXJyZW50RGF0YSIsInRyYW5zcGFyZW50TGluZSIsImZvckVhY2giLCJoYXNoZWRMaW5lIiwiaSIsInNwbGljZSIsImNvbXB1dGVBbmRJbmplY3REaWZmcyIsInByZXZpb3VzRGF0YSIsImN1cnJlbnREYXRhIiwiTWF0aCIsIm1heCIsImxlbmd0aCIsImFkZEV2ZW50TGlzdGVuZXIiLCJyZXN1bHQiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUEsU0FBU0EsY0FBVCxPQUFpREMsWUFBakQsRUFBK0Q7QUFBQSxNQUFyQ0MsSUFBcUMsUUFBckNBLElBQXFDO0FBQUEsTUFBL0JDLEtBQStCLFFBQS9CQSxLQUErQjtBQUFBLE1BQXhCQyxNQUF3QixRQUF4QkEsTUFBd0I7O0FBQzdEO0FBQ0E7QUFDQSxNQUFNQyxVQUFVRixRQUFRLENBQXhCOztBQUVBLE1BQU1HLFVBQVUsRUFBaEI7QUFDQSxPQUFLLElBQUlDLE1BQU0sQ0FBZixFQUFrQkEsTUFBTUgsTUFBeEIsRUFBZ0NHLEtBQWhDLEVBQXVDO0FBQ3JDLFFBQU1DLGNBQWMsSUFBSUMsaUJBQUosQ0FBc0JKLFVBQVdKLGVBQWUsQ0FBaEQsQ0FBcEI7QUFDQSxT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cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGFsaWduQXJyYXlzIGZyb20gJy4uL2FsaWduQXJyYXlzJztcblxuZnVuY3Rpb24gaW1hZ2VUbzJEQXJyYXkoeyBkYXRhLCB3aWR0aCwgaGVpZ2h0IH0sIHBhZGRpbmdSaWdodCkge1xuICAvLyBUaGUgaW1hZ2VEYXRhIGlzIGEgMUQgYXJyYXkuIEVhY2ggZWxlbWVudCBpbiB0aGUgYXJyYXkgY29ycmVzcG9uZHMgdG8gYVxuICAvLyBkZWNpbWFsIHZhbHVlIHRoYXQgcmVwcmVzZW50cyBvbmUgb2YgdGhlIFJHQkEgY2hhbm5lbHMgZm9yIHRoYXQgcGl4ZWwuXG4gIGNvbnN0IHJvd1NpemUgPSB3aWR0aCAqIDQ7XG5cbiAgY29uc3QgbmV3RGF0YSA9IFtdO1xuICBmb3IgKGxldCByb3cgPSAwOyByb3cgPCBoZWlnaHQ7IHJvdysrKSB7XG4gICAgY29uc3QgcGl4ZWxzSW5Sb3cgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocm93U2l6ZSArIChwYWRkaW5nUmlnaHQgKiA0KSk7XG4gICAgZm9yIChsZXQgbG9jYXRpb24gPSAwOyBsb2NhdGlvbiA8IHJvd1NpemU7IGxvY2F0aW9uKyspIHtcbiAgICAgIHBpeGVsc0luUm93W2xvY2F0aW9uXSA9IGRhdGFbKHJvdyAqIHJvd1NpemUpICsgbG9jYXRpb25dO1xuICAgIH1cbiAgICBuZXdEYXRhLnB1c2gocGl4ZWxzSW5Sb3cpO1xuICB9XG4gIHJldHVybiBuZXdEYXRhO1xufVxuXG5mdW5jdGlvbiBhbGlnbih7XG4gIHByZXZpb3VzSW1hZ2VEYXRhLFxuICBjdXJyZW50SW1hZ2VEYXRhLFxuICBtYXhXaWR0aCxcbn0pIHtcbiAgY29uc3QgaGFzaGVkUHJldmlvdXNEYXRhID0gcHJldmlvdXNJbWFnZURhdGEubWFwKEpTT04uc3RyaW5naWZ5KTtcbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA0MCB9KTtcbiAgY29uc3QgaGFzaGVkQ3VycmVudERhdGEgPSBjdXJyZW50SW1hZ2VEYXRhLm1hcChKU09OLnN0cmluZ2lmeSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UoeyBwcm9ncmVzczogNjAgfSk7XG5cbiAgYWxpZ25BcnJheXMoXG4gICAgaGFzaGVkUHJldmlvdXNEYXRhLFxuICAgIGhhc2hlZEN1cnJlbnREYXRhXG4gICk7XG5cbiAgY29uc3QgdHJhbnNwYXJlbnRMaW5lID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KG1heFdpZHRoICogNCk7XG5cbiAgaGFzaGVkUHJldmlvdXNEYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBwcmV2aW91c0ltYWdlRGF0YS5zcGxpY2UoaSwgMCwgdHJhbnNwYXJlbnRMaW5lKTtcbiAgICB9XG4gIH0pO1xuXG4gIGhhc2hlZEN1cnJlbnREYXRhLmZvckVhY2goKGhhc2hlZExpbmUsIGkpID0+IHtcbiAgICBpZiAoaGFzaGVkTGluZSA9PT0gJysnKSB7XG4gICAgICBjdXJyZW50SW1hZ2VEYXRhLnNwbGljZShpLCAwLCB0cmFuc3BhcmVudExpbmUpO1xuICAgIH1cbiAgfSk7XG59XG5cbi8qKlxuICogVGFrZXMgdHdvIDJkIGltYWdlcywgY29tcHV0ZXMgdGhlIGRpZmYgYmV0d2VlbiB0aGUgdHdvLCBhbmQgaW5qZWN0cyBwaXhlbHMgdG9cbiAqIGJvdGggaW4gb3JkZXIgdG86XG4gKiBhKSBtYWtlIGJvdGggaW1hZ2VzIHRoZSBzYW1lIGhlaWdodFxuICogYikgcHJvcGVybHkgdmlzdWFsaXplIGRpZmZlcmVuY2VzXG4gKlxuICogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIG1ldGhvZCBNVVRBVEVTIGRhdGEuXG4gKlxuICogQHBhcmFtIHtBcnJheX0gcHJldmlvdXNEYXRhXG4gKiBAcGFyYW0ge0FycmF5fSBjdXJyZW50RGF0YVxuICovXG5mdW5jdGlvbiBjb21wdXRlQW5kSW5qZWN0RGlmZnMoeyBwcmV2aW91c0RhdGEsIGN1cnJlbnREYXRhIH0pIHtcbiAgY29uc3QgbWF4V2lkdGggPSBNYXRoLm1heChwcmV2aW91c0RhdGEud2lkdGgsIGN1cnJlbnREYXRhLndpZHRoKTtcblxuICBjb25zdCBwcmV2aW91c0ltYWdlRGF0YSA9IGltYWdlVG8yREFycmF5KFxuICAgIHByZXZpb3VzRGF0YSwgbWF4V2lkdGggLSBwcmV2aW91c0RhdGEud2lkdGgpO1xuXG4gIGNvbnN0IGN1cnJlbnRJbWFnZURhdGEgPSBpbWFnZVRvMkRBcnJheShcbiAgICBjdXJyZW50RGF0YSwgbWF4V2lkdGggLSBjdXJyZW50RGF0YS53aWR0aCk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiAyMCB9KTtcblxuICBhbGlnbih7XG4gICAgcHJldmlvdXNJbWFnZURhdGEsXG4gICAgY3VycmVudEltYWdlRGF0YSxcbiAgICBtYXhXaWR0aCxcbiAgfSk7XG5cbiAgc2VsZi5wb3N0TWVzc2FnZSh7IHByb2dyZXNzOiA4NSB9KTtcblxuICByZXR1cm4ge1xuICAgIGN1cnJlbnREYXRhOiB7XG4gICAgICBkYXRhOiBjdXJyZW50SW1hZ2VEYXRhLFxuICAgICAgaGVpZ2h0OiBjdXJyZW50SW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICAgIHByZXZpb3VzRGF0YToge1xuICAgICAgZGF0YTogcHJldmlvdXNJbWFnZURhdGEsXG4gICAgICBoZWlnaHQ6IHByZXZpb3VzSW1hZ2VEYXRhLmxlbmd0aCxcbiAgICAgIHdpZHRoOiBtYXhXaWR0aCxcbiAgICB9LFxuICB9O1xufVxuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSB9KSA9PiB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbXB1dGVBbmRJbmplY3REaWZmcyh7IHByZXZpb3VzRGF0YSwgY3VycmVudERhdGEgfSk7XG4gIHNlbGYucG9zdE1lc3NhZ2UocmVzdWx0KTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0NvbXB1dGVBbmRJbmplY3REaWZmc1dvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
|
48
|
+
|
49
|
+
/***/ },
|
50
|
+
/* 1 */
|
51
|
+
/***/ function(module, exports) {
|
52
|
+
|
53
|
+
eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = alignArrays;\n\nfunction _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nvar MOVEMENT = {\n none: 0,\n diagonal: 1,\n vertical: 2,\n horizontal: 3\n};\n\nfunction initMatrix(height, width) {\n var rows = new Array(height);\n for (var i = 0; i < rows.length; i++) {\n rows[i] = new Int32Array(width);\n }\n return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n // adapted from\n // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n var aLength = a.length;\n var bLength = b.length;\n var memo = initMatrix(aLength + 1, bLength + 1);\n var solution = initMatrix(aLength + 1, bLength + 1);\n\n // Loop and find the solution\n for (var i = 1; i <= aLength; i++) {\n for (var j = 1; j <= bLength; j++) {\n if (a[i - 1] === b[j - 1]) {\n // diagonal\n memo[i][j] = memo[i - 1][j - 1] + 1;\n solution[i][j] = MOVEMENT.diagonal;\n } else {\n memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n if (memo[i][j] === memo[i - 1][j]) {\n solution[i][j] = MOVEMENT.vertical;\n } else {\n solution[i][j] = MOVEMENT.horizontal;\n }\n }\n }\n }\n if (memo[aLength][bLength] === 0) {\n // No soltuion found\n return null;\n }\n return solution;\n}\n\nfunction placeholders(count) {\n console.log(count);\n var result = new Array(count);\n for (var i = 0; i < count; i++) {\n result[i] = '+';\n }\n return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n var movement = solution[a.length][b.length];\n var ai = a.length;\n var bi = b.length;\n\n var changes = 0;\n\n while (movement !== MOVEMENT.none) {\n if (movement === MOVEMENT.diagonal) {\n if (changes < 0) {\n b.splice.apply(b, [bi, 0].concat(_toConsumableArray(placeholders(Math.abs(changes)))));\n } else if (changes > 0) {\n a.splice.apply(a, [ai, 0].concat(_toConsumableArray(placeholders(changes))));\n }\n ai--;\n bi--;\n changes = 0;\n } else if (movement === MOVEMENT.horizontal) {\n bi--;\n changes++;\n } else if (movement === MOVEMENT.vertical) {\n ai--;\n changes--;\n }\n movement = solution[ai][bi];\n }\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n var aLength = a.length;\n var bLength = b.length;\n var shorterArray = aLength > bLength ? b : a;\n shorterArray.push.apply(shorterArray, _toConsumableArray(placeholders(Math.abs(aLength - bLength))));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction alignArrays(a, b) {\n var lcsSolution = longestCommonSubsequence(a, b);\n if (lcsSolution) {\n applySolution(lcsSolution, a, b);\n } else {\n applyPadding(a, b);\n }\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/alignArrays.js?01b7"],"names":["alignArrays","MOVEMENT","none","diagonal","vertical","horizontal","initMatrix","height","width","rows","Array","i","length","Int32Array","longestCommonSubsequence","a","b","aLength","bLength","memo","solution","j","Math","max","placeholders","count","console","log","result","applySolution","movement","ai","bi","changes","splice","abs","applyPadding","shorterArray","push","lcsSolution"],"mappings":"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","file":"1.js","sourcesContent":["const MOVEMENT = {\n  none: 0,\n  diagonal: 1,\n  vertical: 2,\n  horizontal: 3,\n};\n\nfunction initMatrix(height, width) {\n  const rows = new Array(height);\n  for (let i = 0; i < rows.length; i++) {\n    rows[i] = new Int32Array(width);\n  }\n  return rows;\n}\n\nfunction longestCommonSubsequence(a, b) {\n  // adapted from\n  // http://algorithms.tutorialhorizon.com/dynamic-programming-longest-common-subsequence/\n  const aLength = a.length;\n  const bLength = b.length;\n  const memo = initMatrix(aLength + 1, bLength + 1);\n  const solution = initMatrix(aLength + 1, bLength + 1);\n\n  // Loop and find the solution\n  for (let i = 1; i <= aLength; i++) {\n    for (let j = 1; j <= bLength; j++) {\n      if (a[i - 1] === b[j - 1]) {\n        // diagonal\n        memo[i][j] = memo[i - 1][j - 1] + 1;\n        solution[i][j] = MOVEMENT.diagonal;\n      } else {\n        memo[i][j] = Math.max(memo[i - 1][j], memo[i][j - 1]);\n        if (memo[i][j] === memo[i - 1][j]) {\n          solution[i][j] = MOVEMENT.vertical;\n        } else {\n          solution[i][j] = MOVEMENT.horizontal;\n        }\n      }\n    }\n  }\n  if (memo[aLength][bLength] === 0) {\n    // No soltuion found\n    return null;\n  }\n  return solution;\n}\n\nfunction placeholders(count) {\n  console.log(count)\n  const result = new Array(count);\n  for (let i = 0; i < count; i++) {\n    result[i] = '+';\n  }\n  return result;\n}\n/**\n * Apply an lcs solution to arrays. Note that this will MUTATE the arrays,\n * injecting \"+\" where gaps are needed.\n *\n * @param {Array<Array>} solution as computed by `longestCommonSubsequence`\n * @param {Array} a\n * @param {Array} b\n */\nfunction applySolution(solution, a, b) {\n  let movement = solution[a.length][b.length];\n  let ai = a.length;\n  let bi = b.length;\n\n  let changes = 0;\n\n  while (movement !== MOVEMENT.none) {\n    if (movement === MOVEMENT.diagonal) {\n      if (changes < 0) {\n        b.splice(bi, 0, ...placeholders(Math.abs(changes)));\n      } else if (changes > 0) {\n        a.splice(ai, 0, ...placeholders(changes));\n      }\n      ai--;\n      bi--;\n      changes = 0;\n    } else if (movement === MOVEMENT.horizontal) {\n      bi--;\n      changes++;\n    } else if (movement === MOVEMENT.vertical) {\n      ai--;\n      changes--;\n    }\n    movement = solution[ai][bi];\n  }\n}\n\n/**\n * To be called for arrays that share nothing in common.\n *\n * Note that this will MUTATE the arrays,\n *\n * @param {Array} a\n * @param {Array} b\n */\nfunction applyPadding(a, b) {\n  const aLength = a.length;\n  const bLength = b.length;\n  const shorterArray = aLength > bLength ? b : a;\n  shorterArray.push(...placeholders(Math.abs(aLength - bLength)));\n}\n/**\n * Computes the longest common subsequence of two arrays, then uses that\n * solution to inject gaps into the arrays, making them align on common\n * subsequences.\n *\n * @param {Array} a\n * @param {Array} b\n */\nexport default function alignArrays(a, b) {\n  const lcsSolution = longestCommonSubsequence(a, b);\n  if (lcsSolution) {\n    applySolution(lcsSolution, a, b);\n  } else {\n    applyPadding(a, b);\n  }\n}\n\n\n\n/** WEBPACK FOOTER **\n ** ./js/src/alignArrays.js\n **/"],"sourceRoot":""}");
|
54
|
+
|
55
|
+
/***/ }
|
56
|
+
/******/ ]);
|