happo 2.8.0 → 2.8.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (175) hide show
  1. checksums.yaml +4 -4
  2. data/lib/happo/public/05bf9fb3d7f0192e7194.worker.js +56 -0
  3. data/lib/happo/public/06bf22e3fa348f77a82f.worker.js +56 -0
  4. data/lib/happo/public/08ccdc92f4d8ed3cff76.worker.js +62 -0
  5. data/lib/happo/public/0b94c9757c981bfc9fd7.worker.js +62 -0
  6. data/lib/happo/public/0d036738a62c01299743.worker.js +56 -0
  7. data/lib/happo/public/{203272bf00c349325c60.worker.js → 0e32a5b5219342a9aa1e.worker.js} +1 -1
  8. data/lib/happo/public/105bb978b49beac37583.worker.js +62 -0
  9. data/lib/happo/public/1172b74983f510409e2d.worker.js +62 -0
  10. data/lib/happo/public/13d0677d081eefd8c8bb.worker.js +62 -0
  11. data/lib/happo/public/1635684653e1a1979bea.worker.js +56 -0
  12. data/lib/happo/public/{4f6b0faf1b43e93a1404.worker.js → 1bb6a3a6bba3c02df491.worker.js} +1 -1
  13. data/lib/happo/public/1c48c5a95c9c5331c65f.worker.js +56 -0
  14. data/lib/happo/public/{1a86029ce53b1aa81f68.worker.js → 1d1049629f51e72af69f.worker.js} +1 -1
  15. data/lib/happo/public/{423ba9f549f3829873e6.worker.js → 1fd046493403de128de1.worker.js} +1 -1
  16. data/lib/happo/public/20c34bdb9a226c485df8.worker.js +56 -0
  17. data/lib/happo/public/23686d1b2fc3f4f86d12.worker.js +56 -0
  18. data/lib/happo/public/238307cd4525e692bc79.worker.js +56 -0
  19. data/lib/happo/public/2c643331ceb20b73e255.worker.js +56 -0
  20. data/lib/happo/public/2db6b1d7219028bac961.worker.js +62 -0
  21. data/lib/happo/public/2fbe9cf56ab2c3ab9a4a.worker.js +62 -0
  22. data/lib/happo/public/30ee387dd21d6b20a927.worker.js +62 -0
  23. data/lib/happo/public/34c08770d9dc4981b2a2.worker.js +62 -0
  24. data/lib/happo/public/35d8e219a4200e27adf9.worker.js +62 -0
  25. data/lib/happo/public/360e0acb872760f43b07.worker.js +1 -1
  26. data/lib/happo/public/385cc56a00d991287244.worker.js +62 -0
  27. data/lib/happo/public/38e90e6329f74e2d681a.worker.js +56 -0
  28. data/lib/happo/public/3a1ce5368a835b999d55.worker.js +62 -0
  29. data/lib/happo/public/3a7567174ad2bd91c107.worker.js +56 -0
  30. data/lib/happo/public/3b63d7abc667a23e0a2b.worker.js +56 -0
  31. data/lib/happo/public/3bdf5dcbba8864a7dcae.worker.js +62 -0
  32. data/lib/happo/public/{463eb0ecadbb11365ca4.worker.js → 3ee7268990c5d4183ad1.worker.js} +1 -1
  33. data/lib/happo/public/41d90eebc5449581b92a.worker.js +56 -0
  34. data/lib/happo/public/46a8fcf803fae37aa259.worker.js +62 -0
  35. data/lib/happo/public/46ef1d4f416d92b92cab.worker.js +62 -0
  36. data/lib/happo/public/4960a52eba13a73eb069.worker.js +56 -0
  37. data/lib/happo/public/49c762dec74fb3e7c638.worker.js +62 -0
  38. data/lib/happo/public/507ef3ea49332d42f582.worker.js +62 -0
  39. data/lib/happo/public/511d7e03fd683c618a22.worker.js +56 -0
  40. data/lib/happo/public/527516eb649429b011f7.worker.js +68 -0
  41. data/lib/happo/public/52e42bbfa68714fd1b26.worker.js +62 -0
  42. data/lib/happo/public/53e970da89474270e914.worker.js +56 -0
  43. data/lib/happo/public/572846c1173405e57d7d.worker.js +62 -0
  44. data/lib/happo/public/578cb8199b43c9f746ec.worker.js +62 -0
  45. data/lib/happo/public/58ae755db5700fc141d3.worker.js +62 -0
  46. data/lib/happo/public/5b7ffa7e0cfb69ac4952.worker.js +56 -0
  47. data/lib/happo/public/{62c7585d1d23297b316f.worker.js → 5cbf9957eb776724a5b6.worker.js} +1 -1
  48. data/lib/happo/public/612cb8b078897547b091.worker.js +62 -0
  49. data/lib/happo/public/{9cdd6f0763fd866a5118.worker.js → 6368c448d11a203d506b.worker.js} +1 -1
  50. data/lib/happo/public/69d16a4997e4170af4ee.worker.js +1 -0
  51. data/lib/happo/public/69fdd0bf1915b19115d6.worker.js +62 -0
  52. data/lib/happo/public/74309d94f574207e4e79.worker.js +62 -0
  53. data/lib/happo/public/74882ef02b7b28391aac.worker.js +62 -0
  54. data/lib/happo/public/785285eac660735ef3e0.worker.js +56 -0
  55. data/lib/happo/public/{730c035f5c2b404ff225.worker.js → 7d7551366f97adda7b5b.worker.js} +2 -2
  56. data/lib/happo/public/7e53176d18c890a5424c.worker.js +56 -0
  57. data/lib/happo/public/{b20a32590dfe259f4fe3.worker.js → 818ac008a4a7d62aa55c.worker.js} +2 -2
  58. data/lib/happo/public/823340592678aeab5317.worker.js +56 -0
  59. data/lib/happo/public/8cc13e3a447ea341f962.worker.js +56 -0
  60. data/lib/happo/public/8d0d79f0c81652bfb3f1.worker.js +1 -0
  61. data/lib/happo/public/9037f52c590941a75440.worker.js +62 -0
  62. data/lib/happo/public/{9472102d9a6d2577f988.worker.js → 935b279f09cf3c4046ec.worker.js} +1 -1
  63. data/lib/happo/public/{4e685162de1d8a9b102d.worker.js → 945f54dcc60d8b7abc9d.worker.js} +1 -1
  64. data/lib/happo/public/{f81ed0eec7edfa2dfadb.worker.js → 94ff43f0e47ff2759712.worker.js} +1 -1
  65. data/lib/happo/public/95a6641585f90432e282.worker.js +1 -0
  66. data/lib/happo/public/95ae334beb5125d1b4ce.worker.js +62 -0
  67. data/lib/happo/public/95b52fcff8f5732b1902.worker.js +62 -0
  68. data/lib/happo/public/984f0e2d0e896c2f4048.worker.js +62 -0
  69. data/lib/happo/public/988f633191766eef664a.worker.js +56 -0
  70. data/lib/happo/public/99af2b663636fee65291.worker.js +62 -0
  71. data/lib/happo/public/9c8f5ee8dfa77750be8c.worker.js +62 -0
  72. data/lib/happo/public/9d57ea9c2af229a03119.worker.js +56 -0
  73. data/lib/happo/public/9d7be8cf3aadb0b77624.worker.js +62 -0
  74. data/lib/happo/public/9e6b804a33c6799809cf.worker.js +62 -0
  75. data/lib/happo/public/HappoApp.bundle.js.map +1 -0
  76. data/lib/happo/public/a3574ff725014ef26602.worker.js +56 -0
  77. data/lib/happo/public/a5564ec6a5def9e7a95a.worker.js +1 -0
  78. data/lib/happo/public/a59e639dbf2bc03792c2.worker.js +56 -0
  79. data/lib/happo/public/a6de7e3f4866c4d4af09.worker.js +62 -0
  80. data/lib/happo/public/a79d520c7469920c10a2.worker.js +62 -0
  81. data/lib/happo/public/a7da9c5111db811b9535.worker.js +62 -0
  82. data/lib/happo/public/a8d9429cf3dd094aa190.worker.js +62 -0
  83. data/lib/happo/public/ab952692e7c4df03d3b8.worker.js +56 -0
  84. data/lib/happo/public/{80009e8bc0ffd29d3390.worker.js → aca1442707a8e088ec9e.worker.js} +1 -1
  85. data/lib/happo/public/{28237f0668f12dfb456c.worker.js → adc1f6a613d0d2eda4b8.worker.js} +1 -7
  86. data/lib/happo/public/b2a65c76321af7bc6d89.worker.js +62 -0
  87. data/lib/happo/public/b2a81bbc8e6f0ddb3432.worker.js +62 -0
  88. data/lib/happo/public/{e247eaa7506f44b5b847.worker.js → b41bd9a6b659bfe3096f.worker.js} +2 -2
  89. data/lib/happo/public/b66da676446fdda1a58c.worker.js +56 -0
  90. data/lib/happo/public/c1dd7c953827ff9578f0.worker.js +68 -0
  91. data/lib/happo/public/{3818f1c91ce4d4d8b724.worker.js → c556b6ede3bbd230cebd.worker.js} +1 -1
  92. data/lib/happo/public/ca172f55a8f34b526853.worker.js +62 -0
  93. data/lib/happo/public/{830db6cc9c99c2e8e9c1.worker.js → cbc31c846bebca5538bf.worker.js} +1 -1
  94. data/lib/happo/public/ce2440f84aa6708e51d4.worker.js +56 -0
  95. data/lib/happo/public/d4d18f5ac12fd3309c03.worker.js +56 -0
  96. data/lib/happo/public/d5ec9230a58d0929fe2d.worker.js +62 -0
  97. data/lib/happo/public/d650ec5f5ee0203de80a.worker.js +1 -0
  98. data/lib/happo/public/d6dbbb2aeed75685efa1.worker.js +62 -0
  99. data/lib/happo/public/d7403b48b365a76148a2.worker.js +56 -0
  100. data/lib/happo/public/d75350c7043f58e46d0c.worker.js +62 -0
  101. data/lib/happo/public/d899e9318500e6115f21.worker.js +62 -0
  102. data/lib/happo/public/df0d03d918e87308e3e3.worker.js +62 -0
  103. data/lib/happo/public/e39e8636909998d394e8.worker.js +56 -0
  104. data/lib/happo/public/e5c91be22eaad28e7e82.worker.js +56 -0
  105. data/lib/happo/public/e65a11af21a63b0c66df.worker.js +62 -0
  106. data/lib/happo/public/e8d18bed99776d3eabfd.worker.js +62 -0
  107. data/lib/happo/public/f0a047027dca5832b680.worker.js +62 -0
  108. data/lib/happo/public/f18c5838c7ec92d4a291.worker.js +62 -0
  109. data/lib/happo/public/f1b8838f395f31ff14a0.worker.js +62 -0
  110. data/lib/happo/public/fb955bccd11761291311.worker.js +62 -0
  111. data/lib/happo/public/happo-logo.svg +1 -0
  112. data/lib/happo/uploader.rb +3 -1
  113. data/lib/happo/version.rb +1 -1
  114. metadata +111 -80
  115. data/lib/happo/public/0622a1eb2a5b4dab0d7d.worker.js +0 -62
  116. data/lib/happo/public/088cc9d4615d51d37661.worker.js +0 -68
  117. data/lib/happo/public/11b439cedd6aa6a01e71.worker.js +0 -68
  118. data/lib/happo/public/167f0a5b59a8c27815fe.worker.js +0 -62
  119. data/lib/happo/public/19f5f7403fb4a2ba42e2.worker.js +0 -62
  120. data/lib/happo/public/1b6803fbc7b07f61cba2.worker.js +0 -74
  121. data/lib/happo/public/1d3dc560f1f39393dafe.worker.js +0 -62
  122. data/lib/happo/public/25ad7a1e52bd424ca81d.worker.js +0 -62
  123. data/lib/happo/public/26d95a76207ba0fe6c25.worker.js +0 -74
  124. data/lib/happo/public/2a7a935f3b9526c6f7ab.worker.js +0 -74
  125. data/lib/happo/public/2c00cb8d0a2eb5dd4078.worker.js +0 -68
  126. data/lib/happo/public/2c71ddb5b15a755d4644.worker.js +0 -68
  127. data/lib/happo/public/32648d3dfb22995f4066.worker.js +0 -68
  128. data/lib/happo/public/3452650686a8096b2bb7.worker.js +0 -68
  129. data/lib/happo/public/346cefcc1572890455b8.worker.js +0 -68
  130. data/lib/happo/public/35bd63f58bb69ac267ea.worker.js +0 -68
  131. data/lib/happo/public/3cc2e649d9b6ff62f4ab.worker.js +0 -62
  132. data/lib/happo/public/3deccedc8fbb96bb1aa7.worker.js +0 -68
  133. data/lib/happo/public/3e672be26a7a8864a342.worker.js +0 -68
  134. data/lib/happo/public/41aa7471344a79ff98f2.worker.js +0 -68
  135. data/lib/happo/public/47c52ff0e060fedb841f.worker.js +0 -62
  136. data/lib/happo/public/4eea1b4008ed68e41f82.worker.js +0 -62
  137. data/lib/happo/public/589c2f9547d240ac8a57.worker.js +0 -68
  138. data/lib/happo/public/5a8c0588ca745d7904f7.worker.js +0 -68
  139. data/lib/happo/public/5b3cabd81f7d8688f7a5.worker.js +0 -62
  140. data/lib/happo/public/5f5c9f07cb5117c523a2.worker.js +0 -68
  141. data/lib/happo/public/5fb8c7066659ea1b57e2.worker.js +0 -62
  142. data/lib/happo/public/5fb962cc191a60b42af0.worker.js +0 -68
  143. data/lib/happo/public/62e676d31cbf8aaa9359.worker.js +0 -68
  144. data/lib/happo/public/65efcf6aee5e3ef33539.worker.js +0 -68
  145. data/lib/happo/public/693d4918a5dae465c134.worker.js +0 -68
  146. data/lib/happo/public/69d106b071dd31ad86de.worker.js +0 -68
  147. data/lib/happo/public/7609088c49b73ee2e3e1.worker.js +0 -68
  148. data/lib/happo/public/76161d401db5bb36980a.worker.js +0 -62
  149. data/lib/happo/public/7d9febb46b37ddffa2d5.worker.js +0 -62
  150. data/lib/happo/public/7f29af56bd0ea82d77d9.worker.js +0 -68
  151. data/lib/happo/public/85ad69599657de02d20d.worker.js +0 -68
  152. data/lib/happo/public/875e790f3e64476b3aa0.worker.js +0 -68
  153. data/lib/happo/public/8fa9a7b2a5f19075eedc.worker.js +0 -74
  154. data/lib/happo/public/906ce908877052838b8e.worker.js +0 -62
  155. data/lib/happo/public/9ac309ab34f48e7af07c.worker.js +0 -68
  156. data/lib/happo/public/aa7ae8013f9780eb8acb.worker.js +0 -68
  157. data/lib/happo/public/b25ebdd4e2d2dee8c7b1.worker.js +0 -68
  158. data/lib/happo/public/b7a49d603b1d50f535b2.worker.js +0 -68
  159. data/lib/happo/public/b8e667564ad96f5c77f4.worker.js +0 -68
  160. data/lib/happo/public/b913c562805ec9a01635.worker.js +0 -68
  161. data/lib/happo/public/bab832c7110d15a5f43b.worker.js +0 -62
  162. data/lib/happo/public/bfcf99c9b2bf3d58ec7f.worker.js +0 -62
  163. data/lib/happo/public/c0aeff8b32c9911c8547.worker.js +0 -68
  164. data/lib/happo/public/c3d7b17359c21ff82281.worker.js +0 -68
  165. data/lib/happo/public/c9f7adcce3de80eb759a.worker.js +0 -68
  166. data/lib/happo/public/d7b87aec5f1e4f2a7a5d.worker.js +0 -68
  167. data/lib/happo/public/db617ae6458966f4acae.worker.js +0 -68
  168. data/lib/happo/public/e92bb4e1e06843fa771e.worker.js +0 -62
  169. data/lib/happo/public/e9e2d97848e0059e15c5.worker.js +0 -62
  170. data/lib/happo/public/ea87c48552c960ed7a1c.worker.js +0 -74
  171. data/lib/happo/public/eb5aafba8d16a397517a.worker.js +0 -68
  172. data/lib/happo/public/edc218b655b9ef694b3e.worker.js +0 -74
  173. data/lib/happo/public/eea64a6efa0620a576d7.worker.js +0 -68
  174. data/lib/happo/public/f269a8e7ffca13e9366a.worker.js +0 -68
  175. data/lib/happo/public/fcd20ee3e5a10e8667bb.worker.js +0 -68
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 9566789ffa0a31de9ff986e11424a10c222259eb
4
- data.tar.gz: 8e6ae034781cc9dd0258574ce9498d4411db594b
3
+ metadata.gz: 3f6c9d4bd5dd3b4c13623aefd31b0e44eb5667bd
4
+ data.tar.gz: 33eaf52012b3410c9f0be7163b7ba5cedab428a1
5
5
  SHA512:
6
- metadata.gz: 7d51844c05f2a24667c77f611910f98ace535b5ba7b1397aa764be4ff8ff5001111cc5fa0a49c567ec68912bc835411c81bdf7f244e0ee638990ef756b95093a
7
- data.tar.gz: 7d3176e11cef94c2d3f6a21fad84ff9be1a19de32e92698ba63bde4756a05f5f1a7f51a6f541e11db3b38f3a5067c86b2756aa8c4dbbbaab1cfc447affa52afa
6
+ metadata.gz: 61198c14b79ff9709c76decac7ca8aac49d89d489a5787210b15666945b3f539aecea92ca8af1f928d0eb2fe0c3581b99862212b19719c42210c5b55db00c8d4
7
+ data.tar.gz: 759e49006bdbafca900b0cccc18622ffff17aa6ff9142662584e15c4a8d2da100e537375f2f14d419237746da379111c64c85d009c019a831c5f5e6fef73384a
@@ -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 _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\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction flattenImageData(imageData) {\n var result = [];\n imageData.forEach(function (row) {\n row.forEach(function (pixel) {\n result.push.apply(result, _toConsumableArray(pixel));\n });\n });\n return Uint8ClampedArray.from(result);\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(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n var start = performance.now();\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n console.log('(internal) image creation took', performance.now() - start);\n\n start = performance.now();\n var adiffResults = _adiff2.default.diff(previousImageData.map(function (d) {\n return btoa(d);\n }), currentImageData.map(function (d) {\n return btoa(d);\n }));\n console.log('(internal) adiffing took', performance.now() - start);\n\n start = performance.now();\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\n console.log('(internal) image prep took', performance.now() - start);\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n\n return {\n currentData: {\n data: flattenImageData(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: flattenImageData(previousImageData),\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref3) {\n var _ref3$data = _ref3.data;\n var previousData = _ref3$data.previousData;\n var currentData = _ref3$data.currentData;\n\n var start = performance.now();\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n console.log('(internal) ComputeAndInjectDiffsWorker took', performance.now() - start);\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":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","flattenImageData","imageData","result","forEach","pixel","Uint8ClampedArray","from","computeAndInjectDiffs","previousData","currentData","maxWidth","Math","max","redLine","greenLine","start","performance","now","previousImageData","currentImageData","console","log","adiffResults","diff","map","btoa","d","instruction","atIndex","deletedItems","addedItems","length","splice","self","addEventListener","postMessage","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction flattenImageData(imageData) {\n  const result = [];\n  imageData.forEach((row) => {\n    row.forEach((pixel) => {\n      result.push(...pixel);\n    });\n  });\n  return Uint8ClampedArray.from(result);\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n  let start = performance.now();\n  const previousImageData = imageTo2DArray(\n    previousData, maxWidth - previousData.width);\n\n  const currentImageData = imageTo2DArray(\n    currentData, maxWidth - currentData.width);\n  console.log('(internal) image creation took', performance.now() - start);\n\n  start = performance.now();\n  const adiffResults = adiff.diff(\n    previousImageData.map(d => btoa(d)),\n    currentImageData.map(d => btoa(d))\n  );\n  console.log('(internal) adiffing took', performance.now() - start);\n\n  start = performance.now();\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\n  console.log('(internal) image prep took', performance.now() - start);\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(previousImageData),\n      height: previousImageData.length,\n      width: maxWidth,\n    },\n  };\n}\n\nself.addEventListener('message', ({ data: { previousData, currentData } }) => {\n  const start = performance.now();\n  const result = computeAndInjectDiffs({ previousData, currentData });\n  console.log('(internal) ComputeAndInjectDiffsWorker took', performance.now() - start);\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 _adiff = __webpack_require__(1);\n\nvar _adiff2 = _interopRequireDefault(_adiff);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\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\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction flattenImageData(imageData) {\n var result = [];\n imageData.forEach(function (row) {\n row.forEach(function (pixel) {\n result.push.apply(result, _toConsumableArray(pixel));\n });\n });\n return Uint8ClampedArray.from(result);\n}\n\nvar start = void 0;\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(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n\n var maxWidth = Math.max(previousData.width, currentData.width);\n\n var redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n start = performance.now();\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n console.log('(internal) image creation took', performance.now() - start);\n\n var hashedPreviousImageData = previousImageData.map(function (d) {\n return btoa(d);\n });\n var hashedCurrentImageData = currentImageData.map(function (d) {\n return btoa(d);\n });\n start = performance.now();\n var adiffResults = _adiff2.default.diff(hashedPreviousImageData, hashedCurrentImageData);\n console.log('(internal) adiffing took', performance.now() - start);\n\n start = performance.now();\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n console.log('(internal) image prep took', performance.now() - start);\n\n start = performance.now();\n return {\n currentData: {\n data: flattenImageData(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: flattenImageData(previousImageData),\n height: previousImageData.length,\n width: maxWidth\n }\n };\n}\n\nself.addEventListener('message', function (_ref3) {\n var _ref3$data = _ref3.data;\n var previousData = _ref3$data.previousData;\n var currentData = _ref3$data.currentData;\n\n var result = computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n console.log('(internal) flattening took', performance.now() - start);\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":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","flattenImageData","imageData","result","forEach","pixel","Uint8ClampedArray","from","start","computeAndInjectDiffs","previousData","currentData","maxWidth","Math","max","redLine","greenLine","performance","now","previousImageData","currentImageData","console","log","hashedPreviousImageData","map","btoa","d","hashedCurrentImageData","adiffResults","diff","instruction","atIndex","deletedItems","addedItems","length","splice","self","addEventListener","postMessage","close"],"mappings":";;AAAA;;;;;;;;AAEA;;;;;;;AAOA,SAASA,oBAAT,CAA8BC,IAA9B,EAAoCC,KAApC,EAA2C;AACzC,MAAMC,OAAO,EAAb;AACA,OAAK,IAAIC,IAAI,CAAb,EAAgBA,IAAIF,KAApB,EAA2BE,GAA3B,EAAgC;AAC9BD,SAAKE,IAAL,CAAUJ,IAAV;AACD;AACD,SAAOE,IAAP;AACD;;AAED,SAASG,cAAT,OAAiDC,YAAjD,EAA+D;AAAA,MAArCC,IAAqC,QAArCA,IAAqC;AAAA,MAA/BN,KAA+B,QAA/BA,KAA+B;AAAA,MAAxBO,MAAwB,QAAxBA,MAAwB;;AAC7D;AACA;AACA,MAAMC,UAAUR,QAAQ,CAAxB;AACA,MAAMS,aAAa,SAAbA,UAAa,CAACC,CAAD,EAAIC,CAAJ,EAAU;AAC3B,QAAMC,aAAcD,IAAIH,OAAL,GAAiBE,IAAI,CAAxC;AACA,WAAO,CACLJ,KAAKM,UAAL,CADK,EAELN,KAAKM,aAAa,CAAlB,CAFK,EAGLN,KAAKM,aAAa,CAAlB,CAHK,EAILN,KAAKM,aAAa,CAAlB,CAJK,CAAP;AAMD,GARD;;AAUA,MAAMC,UAAU,EAAhB;AACA,OAAK,IAAIC,MAAM,CAAf,EAAkBA,MAAMP,MAAxB,EAAgCO,KAAhC,EAAuC;AACrC,QAAMC,cAAc,EAApB;AACA,SAAK,IAAIC,MAAM,CAAf,EAAkBA,MAAMhB,KAAxB,EAA+BgB,KAA/B,EAAsC;AACpCD,kBAAYZ,IAAZ,CAAiBM,WAAWO,GAAX,EAAgBF,GAAhB,CAAjB;AACD;AACD,SAAK,IAAIG,MAAM,CAAf,EAAkBA,MAAMZ,YAAxB,EAAsCY,KAAtC,EAA6C;AAC3CF,kBAAYZ,IAAZ,CAAiB,CAAC,CAAD,EAAI,CAAJ,EAAO,CAAP,EAAU,CAAV,CAAjB;AACD;AACDU,YAAQV,IAAR,CAAaY,WAAb;AACD;AACD,SAAOF,OAAP;AACD;;AAED,SAASK,gBAAT,CAA0BC,SAA1B,EAAqC;AACnC,MAAMC,SAAS,EAAf;AACAD,YAAUE,OAAV,CAAkB,UAACP,GAAD,EAAS;AACzBA,QAAIO,OAAJ,CAAY,UAACC,KAAD,EAAW;AACrBF,aAAOjB,IAAP,kCAAemB,KAAf;AACD,KAFD;AAGD,GAJD;AAKA,SAAOC,kBAAkBC,IAAlB,CAAuBJ,MAAvB,CAAP;AACD;;AAED,IAAIK,cAAJ;AACA;;;;;;;;;;;AAWA,SAASC,qBAAT,QAA8D;AAAA,MAA7BC,YAA6B,SAA7BA,YAA6B;AAAA,MAAfC,WAAe,SAAfA,WAAe;;AAC5D,MAAMC,WAAWC,KAAKC,GAAL,CAASJ,aAAa3B,KAAtB,EAA6B4B,YAAY5B,KAAzC,CAAjB;;AAEA,MAAMgC,UAAUlC,qBAAqB,CAAC,GAAD,EAAM,CAAN,EAAS,CAAT,EAAY,GAAZ,CAArB,EAAuC+B,QAAvC,CAAhB;AACA,MAAMI,YAAYnC,qBAAqB,CAAC,CAAD,EAAI,GAAJ,EAAS,CAAT,EAAY,GAAZ,CAArB,EAAuC+B,QAAvC,CAAlB;;AAEAJ,UAAQS,YAAYC,GAAZ,EAAR;AACA,MAAMC,oBAAoBhC,eACxBuB,YADwB,EACVE,WAAWF,aAAa3B,KADd,CAA1B;;AAGA,MAAMqC,mBAAmBjC,eACvBwB,WADuB,EACVC,WAAWD,YAAY5B,KADb,CAAzB;AAEAsC,UAAQC,GAAR,CAAY,gCAAZ,EAA8CL,YAAYC,GAAZ,KAAoBV,KAAlE;;AAEA,MAAMe,0BAA0BJ,kBAAkBK,GAAlB,CAAsB;AAAA,WAAKC,KAAKC,CAAL,CAAL;AAAA,GAAtB,CAAhC;AACA,MAAMC,yBAAyBP,iBAAiBI,GAAjB,CAAqB;AAAA,WAAKC,KAAKC,CAAL,CAAL;AAAA,GAArB,CAA/B;AACAlB,UAAQS,YAAYC,GAAZ,EAAR;AACA,MAAMU,eAAe,gBAAMC,IAAN,CACnBN,uBADmB,EAEnBI,sBAFmB,CAArB;AAIAN,UAAQC,GAAR,CAAY,0BAAZ,EAAwCL,YAAYC,GAAZ,KAAoBV,KAA5D;;AAEAA,UAAQS,YAAYC,GAAZ,EAAR;AACA;AACAU,eAAaxB,OAAb,CAAqB,UAAC0B,WAAD,EAAiB;AACpC,QAAMC,UAAUD,YAAY,CAAZ,CAAhB;AACA,QAAME,eAAeF,YAAY,CAAZ,CAArB;AACA,QAAMG,aAAaH,YAAYI,MAAZ,GAAqB,CAAxC;;AAEA,SAAK,IAAIxC,IAAI,CAAb,EAAgBA,IAAImB,KAAKC,GAAL,CAASkB,YAAT,EAAuBC,UAAvB,CAApB,EAAwDvC,GAAxD,EAA6D;AAC3D,UAAIA,IAAIsC,YAAR,EAAsB;AACpB;AACD,OAFD,MAEO;AACL;AACAb,0BAAkBgB,MAAlB,CAAyBJ,UAAUrC,CAAnC,EAAsC,CAAtC,EAAyCsB,SAAzC;AACD;AACF;AACF,GAbD;;AAeA;AACA,OAAK,IAAI/B,IAAI2C,aAAaM,MAAb,GAAsB,CAAnC,EAAsCjD,KAAK,CAA3C,EAA8CA,GAA9C,EAAmD;AACjD,QAAM6C,cAAcF,aAAa3C,CAAb,CAApB;AACA,QAAM8C,UAAUD,YAAY,CAAZ,CAAhB;AACA,QAAME,eAAeF,YAAY,CAAZ,CAArB;AACA,QAAMG,aAAaH,YAAYI,MAAZ,GAAqB,CAAxC;;AAEA,SAAK,IAAIxC,IAAI,CAAb,EAAgBA,IAAImB,KAAKC,GAAL,CAASkB,YAAT,EAAuBC,UAAvB,CAApB,EAAwDvC,GAAxD,EAA6D;AAC3D,UAAIA,IAAIuC,UAAR,EAAoB;AAClB;AACD,OAFD,MAEO;AACL;AACAb,yBAAiBe,MAAjB,CAAwBJ,UAAUrC,CAAlC,EAAqC,CAArC,EAAwCqB,OAAxC;AACD;AACF;AACF;AACDM,UAAQC,GAAR,CAAY,4BAAZ,EAA0CL,YAAYC,GAAZ,KAAoBV,KAA9D;;AAEAA,UAAQS,YAAYC,GAAZ,EAAR;AACA,SAAO;AACLP,iBAAa;AACXtB,YAAMY,iBAAiBmB,gBAAjB,CADK;AAEX9B,cAAQ8B,iBAAiBc,MAFd;AAGXnD,aAAO6B;AAHI,KADR;AAMLF,kBAAc;AACZrB,YAAMY,iBAAiBkB,iBAAjB,CADM;AAEZ7B,cAAQ6B,kBAAkBe,MAFd;AAGZnD,aAAO6B;AAHK;AANT,GAAP;AAYD;;AAEDwB,KAAKC,gBAAL,CAAsB,SAAtB,EAAiC,iBAA6C;AAAA,yBAA1ChD,IAA0C;AAAA,MAAlCqB,YAAkC,cAAlCA,YAAkC;AAAA,MAApBC,WAAoB,cAApBA,WAAoB;;AAC5E,MAAMR,SAASM,sBAAsB,EAAEC,0BAAF,EAAgBC,wBAAhB,EAAtB,CAAf;AACAU,UAAQC,GAAR,CAAY,4BAAZ,EAA0CL,YAAYC,GAAZ,KAAoBV,KAA9D;AACA4B,OAAKE,WAAL,CAAiBnC,MAAjB;AACAiC,OAAKG,KAAL;AACD,CALD","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction flattenImageData(imageData) {\n  const result = [];\n  imageData.forEach((row) => {\n    row.forEach((pixel) => {\n      result.push(...pixel);\n    });\n  });\n  return Uint8ClampedArray.from(result);\n}\n\nlet start;\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n  start = performance.now();\n  const previousImageData = imageTo2DArray(\n    previousData, maxWidth - previousData.width);\n\n  const currentImageData = imageTo2DArray(\n    currentData, maxWidth - currentData.width);\n  console.log('(internal) image creation took', performance.now() - start);\n\n  const hashedPreviousImageData = previousImageData.map(d => btoa(d));\n  const hashedCurrentImageData = currentImageData.map(d => btoa(d));\n  start = performance.now();\n  const adiffResults = adiff.diff(\n    hashedPreviousImageData,\n    hashedCurrentImageData\n  );\n  console.log('(internal) adiffing took', performance.now() - start);\n\n  start = performance.now();\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n  console.log('(internal) image prep took', performance.now() - start);\n\n  start = performance.now();\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(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  console.log('(internal) flattening took', performance.now() - start);\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,62 @@
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 _getDiffPixel = __webpack_require__(1);\n\nvar _getDiffPixel2 = _interopRequireDefault(_getDiffPixel);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nself.addEventListener('message', function (_ref) {\n var _ref$data = _ref.data;\n var previous = _ref$data.previous;\n var current = _ref$data.current;\n\n var data = new Uint8ClampedArray(previous.length);\n\n console.log('max', previous.length);\n var i = void 0;\n for (i = 0; i < previous.length; i += 4) {\n var pixel = (0, _getDiffPixel2.default)(previous.slice(i, i + 4), current.slice(i, i + 4));\n data[i + 0] = pixel[0]; // r\n data[i + 1] = pixel[1]; // g\n data[i + 2] = pixel[2]; // b\n data[i + 3] = pixel[3]; // a\n }\n console.log('last', i);\n self.postMessage(data);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9JbWFnZURpZmZXb3JrZXIuanM/ZjExYiJdLCJuYW1lcyI6WyJzZWxmIiwiYWRkRXZlbnRMaXN0ZW5lciIsImRhdGEiLCJwcmV2aW91cyIsImN1cnJlbnQiLCJVaW50OENsYW1wZWRBcnJheSIsImxlbmd0aCIsImNvbnNvbGUiLCJsb2ciLCJpIiwicGl4ZWwiLCJzbGljZSIsInBvc3RNZXNzYWdlIiwiY2xvc2UiXSwibWFwcGluZ3MiOiI7O0FBQUE7Ozs7OztBQUVBQSxLQUFLQyxnQkFBTCxDQUFzQixTQUF0QixFQUFpQyxnQkFBcUM7QUFBQSx1QkFBbENDLElBQWtDO0FBQUEsTUFBMUJDLFFBQTBCLGFBQTFCQSxRQUEwQjtBQUFBLE1BQWhCQyxPQUFnQixhQUFoQkEsT0FBZ0I7O0FBQ3BFLE1BQU1GLE9BQU8sSUFBSUcsaUJBQUosQ0FBc0JGLFNBQVNHLE1BQS9CLENBQWI7O0FBRUFDLFVBQVFDLEdBQVIsQ0FBWSxLQUFaLEVBQW1CTCxTQUFTRyxNQUE1QjtBQUNBLE1BQUlHLFVBQUo7QUFDQSxPQUFLQSxJQUFJLENBQVQsRUFBWUEsSUFBSU4sU0FBU0csTUFBekIsRUFBaUNHLEtBQUssQ0FBdEMsRUFBeUM7QUFDdkMsUUFBTUMsUUFBUSw0QkFDWlAsU0FBU1EsS0FBVCxDQUFlRixDQUFmLEVBQWtCQSxJQUFJLENBQXRCLENBRFksRUFFWkwsUUFBUU8sS0FBUixDQUFjRixDQUFkLEVBQWlCQSxJQUFJLENBQXJCLENBRlksQ0FBZDtBQUlBUCxTQUFLTyxJQUFJLENBQVQsSUFBY0MsTUFBTSxDQUFOLENBQWQsQ0FMdUMsQ0FLZjtBQUN4QlIsU0FBS08sSUFBSSxDQUFULElBQWNDLE1BQU0sQ0FBTixDQUFkLENBTnVDLENBTWY7QUFDeEJSLFNBQUtPLElBQUksQ0FBVCxJQUFjQyxNQUFNLENBQU4sQ0FBZCxDQVB1QyxDQU9mO0FBQ3hCUixTQUFLTyxJQUFJLENBQVQsSUFBY0MsTUFBTSxDQUFOLENBQWQsQ0FSdUMsQ0FRZjtBQUN6QjtBQUNESCxVQUFRQyxHQUFSLENBQVksTUFBWixFQUFvQkMsQ0FBcEI7QUFDQVQsT0FBS1ksV0FBTCxDQUFpQlYsSUFBakI7QUFDQUYsT0FBS2EsS0FBTDtBQUNELENBbEJEIiwiZmlsZSI6IjAuanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgZ2V0RGlmZlBpeGVsIGZyb20gJy4uL2dldERpZmZQaXhlbCc7XG5cbnNlbGYuYWRkRXZlbnRMaXN0ZW5lcignbWVzc2FnZScsICh7IGRhdGE6IHsgcHJldmlvdXMsIGN1cnJlbnQgfSB9KSA9PiB7XG4gIGNvbnN0IGRhdGEgPSBuZXcgVWludDhDbGFtcGVkQXJyYXkocHJldmlvdXMubGVuZ3RoKTtcblxuICBjb25zb2xlLmxvZygnbWF4JywgcHJldmlvdXMubGVuZ3RoKVxuICBsZXQgaTtcbiAgZm9yIChpID0gMDsgaSA8IHByZXZpb3VzLmxlbmd0aDsgaSArPSA0KSB7XG4gICAgY29uc3QgcGl4ZWwgPSBnZXREaWZmUGl4ZWwoXG4gICAgICBwcmV2aW91cy5zbGljZShpLCBpICsgNCksXG4gICAgICBjdXJyZW50LnNsaWNlKGksIGkgKyA0KVxuICAgICk7XG4gICAgZGF0YVtpICsgMF0gPSBwaXhlbFswXTsgLy8gclxuICAgIGRhdGFbaSArIDFdID0gcGl4ZWxbMV07IC8vIGdcbiAgICBkYXRhW2kgKyAyXSA9IHBpeGVsWzJdOyAvLyBiXG4gICAgZGF0YVtpICsgM10gPSBwaXhlbFszXTsgLy8gYVxuICB9XG4gIGNvbnNvbGUubG9nKCdsYXN0JywgaSlcbiAgc2VsZi5wb3N0TWVzc2FnZShkYXRhKTtcbiAgc2VsZi5jbG9zZSgpO1xufSk7XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy93b3JrZXJzL0ltYWdlRGlmZldvcmtlci5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
48
+
49
+ /***/ },
50
+ /* 1 */
51
+ /***/ function(module, exports, __webpack_require__) {
52
+
53
+ eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = getDiffPixel;\n\nvar _euclideanDistance = __webpack_require__(2);\n\nvar _euclideanDistance2 = _interopRequireDefault(_euclideanDistance);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction getDiffPixel(previousPixel, currentPixel) {\n if (!previousPixel) {\n return currentPixel;\n }\n\n if (!currentPixel) {\n return previousPixel;\n }\n\n var diff = (0, _euclideanDistance2.default)(previousPixel, currentPixel);\n if (diff === 0) {\n return [currentPixel[0], currentPixel[1], currentPixel[2], 50];\n }\n\n if (diff < 0.2) {\n diff = 0.2;\n }\n return [255, 0, 0, 255 * diff]; // TODO don't use red here\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZ2V0RGlmZlBpeGVsLmpzPzRiNjgiXSwibmFtZXMiOlsiZ2V0RGlmZlBpeGVsIiwicHJldmlvdXNQaXhlbCIsImN1cnJlbnRQaXhlbCIsImRpZmYiXSwibWFwcGluZ3MiOiI7Ozs7O2tCQUV3QkEsWTs7QUFGeEI7Ozs7OztBQUVlLFNBQVNBLFlBQVQsQ0FBc0JDLGFBQXRCLEVBQXFDQyxZQUFyQyxFQUFtRDtBQUNoRSxNQUFJLENBQUNELGFBQUwsRUFBb0I7QUFDbEIsV0FBT0MsWUFBUDtBQUNEOztBQUVELE1BQUksQ0FBQ0EsWUFBTCxFQUFtQjtBQUNqQixXQUFPRCxhQUFQO0FBQ0Q7O0FBRUQsTUFBSUUsT0FBTyxpQ0FBa0JGLGFBQWxCLEVBQWlDQyxZQUFqQyxDQUFYO0FBQ0EsTUFBSUMsU0FBUyxDQUFiLEVBQWdCO0FBQ2QsV0FBTyxDQUNMRCxhQUFhLENBQWIsQ0FESyxFQUVMQSxhQUFhLENBQWIsQ0FGSyxFQUdMQSxhQUFhLENBQWIsQ0FISyxFQUlMLEVBSkssQ0FBUDtBQU1EOztBQUVELE1BQUlDLE9BQU8sR0FBWCxFQUFnQjtBQUNkQSxXQUFPLEdBQVA7QUFDRDtBQUNELFNBQU8sQ0FBQyxHQUFELEVBQU0sQ0FBTixFQUFTLENBQVQsRUFBWSxNQUFNQSxJQUFsQixDQUFQLENBdEJnRSxDQXNCaEM7QUFDakMiLCJmaWxlIjoiMS5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBldWNsaWRlYW5EaXN0YW5jZSBmcm9tICcuL2V1Y2xpZGVhbkRpc3RhbmNlJztcblxuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gZ2V0RGlmZlBpeGVsKHByZXZpb3VzUGl4ZWwsIGN1cnJlbnRQaXhlbCkge1xuICBpZiAoIXByZXZpb3VzUGl4ZWwpIHtcbiAgICByZXR1cm4gY3VycmVudFBpeGVsO1xuICB9XG5cbiAgaWYgKCFjdXJyZW50UGl4ZWwpIHtcbiAgICByZXR1cm4gcHJldmlvdXNQaXhlbDtcbiAgfVxuXG4gIGxldCBkaWZmID0gZXVjbGlkZWFuRGlzdGFuY2UocHJldmlvdXNQaXhlbCwgY3VycmVudFBpeGVsKTtcbiAgaWYgKGRpZmYgPT09IDApIHtcbiAgICByZXR1cm4gW1xuICAgICAgY3VycmVudFBpeGVsWzBdLFxuICAgICAgY3VycmVudFBpeGVsWzFdLFxuICAgICAgY3VycmVudFBpeGVsWzJdLFxuICAgICAgNTAsXG4gICAgXTtcbiAgfVxuXG4gIGlmIChkaWZmIDwgMC4yKSB7XG4gICAgZGlmZiA9IDAuMjtcbiAgfVxuICByZXR1cm4gWzI1NSwgMCwgMCwgMjU1ICogZGlmZl07IC8vIFRPRE8gZG9uJ3QgdXNlIHJlZCBoZXJlXG59XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy9nZXREaWZmUGl4ZWwuanNcbiAqKi8iXSwic291cmNlUm9vdCI6IiJ9");
54
+
55
+ /***/ },
56
+ /* 2 */
57
+ /***/ function(module, exports) {
58
+
59
+ eval("\"use strict\";\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = euclideanDistance;\n/**\n * Compute a score that represents the difference between 2 pixels\n *\n * This method simply takes the Euclidean distance between the RGBA channels\n * of 2 colors over the maximum possible Euclidean distance. This gives us a\n * percentage of how different the two colors are.\n *\n * Although it would be more perceptually accurate to calculate a proper\n * Delta E in Lab colorspace, we probably don't need perceptual accuracy for\n * this application, and it is nice to avoid the overhead of converting RGBA\n * to Lab.\n *\n * Returns a float number between 0 and 1 where 1 is completely different\n * and 0 is no difference\n */\nfunction euclideanDistance(rgba1, rgba2) {\n var distance = (rgba1[0] - rgba2[0]) * (rgba1[0] - rgba2[0]) + (rgba1[1] - rgba2[1]) * (rgba1[1] - rgba2[1]) + (rgba1[2] - rgba2[2]) * (rgba1[2] - rgba2[2]) + (rgba1[3] - rgba2[3]) * (rgba1[3] - rgba2[3]);\n\n return Math.sqrt(distance) / 4 / 255;\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZXVjbGlkZWFuRGlzdGFuY2UuanM/ZDgzYiJdLCJuYW1lcyI6WyJldWNsaWRlYW5EaXN0YW5jZSIsInJnYmExIiwicmdiYTIiLCJkaXN0YW5jZSIsIk1hdGgiLCJzcXJ0Il0sIm1hcHBpbmdzIjoiOzs7OztrQkFld0JBLGlCO0FBZnhCOzs7Ozs7Ozs7Ozs7Ozs7QUFlZSxTQUFTQSxpQkFBVCxDQUEyQkMsS0FBM0IsRUFBa0NDLEtBQWxDLEVBQXlDO0FBQ3RELE1BQU1DLFdBQVksQ0FBQ0YsTUFBTSxDQUFOLElBQVdDLE1BQU0sQ0FBTixDQUFaLEtBQXlCRCxNQUFNLENBQU4sSUFBV0MsTUFBTSxDQUFOLENBQXBDLENBQUQsR0FDQyxDQUFDRCxNQUFNLENBQU4sSUFBV0MsTUFBTSxDQUFOLENBQVosS0FBeUJELE1BQU0sQ0FBTixJQUFXQyxNQUFNLENBQU4sQ0FBcEMsQ0FERCxHQUVDLENBQUNELE1BQU0sQ0FBTixJQUFXQyxNQUFNLENBQU4sQ0FBWixLQUF5QkQsTUFBTSxDQUFOLElBQVdDLE1BQU0sQ0FBTixDQUFwQyxDQUZELEdBR0MsQ0FBQ0QsTUFBTSxDQUFOLElBQVdDLE1BQU0sQ0FBTixDQUFaLEtBQXlCRCxNQUFNLENBQU4sSUFBV0MsTUFBTSxDQUFOLENBQXBDLENBSGxCOztBQUtBLFNBQVFFLEtBQUtDLElBQUwsQ0FBVUYsUUFBVixJQUFzQixDQUF2QixHQUE0QixHQUFuQztBQUNEIiwiZmlsZSI6IjIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvbXB1dGUgYSBzY29yZSB0aGF0IHJlcHJlc2VudHMgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiAyIHBpeGVsc1xuICpcbiAqIFRoaXMgbWV0aG9kIHNpbXBseSB0YWtlcyB0aGUgRXVjbGlkZWFuIGRpc3RhbmNlIGJldHdlZW4gdGhlIFJHQkEgY2hhbm5lbHNcbiAqIG9mIDIgY29sb3JzIG92ZXIgdGhlIG1heGltdW0gcG9zc2libGUgRXVjbGlkZWFuIGRpc3RhbmNlLiBUaGlzIGdpdmVzIHVzIGFcbiAqIHBlcmNlbnRhZ2Ugb2YgaG93IGRpZmZlcmVudCB0aGUgdHdvIGNvbG9ycyBhcmUuXG4gKlxuICogQWx0aG91Z2ggaXQgd291bGQgYmUgbW9yZSBwZXJjZXB0dWFsbHkgYWNjdXJhdGUgdG8gY2FsY3VsYXRlIGEgcHJvcGVyXG4gKiBEZWx0YSBFIGluIExhYiBjb2xvcnNwYWNlLCB3ZSBwcm9iYWJseSBkb24ndCBuZWVkIHBlcmNlcHR1YWwgYWNjdXJhY3kgZm9yXG4gKiB0aGlzIGFwcGxpY2F0aW9uLCBhbmQgaXQgaXMgbmljZSB0byBhdm9pZCB0aGUgb3ZlcmhlYWQgb2YgY29udmVydGluZyBSR0JBXG4gKiB0byBMYWIuXG4gKlxuICogUmV0dXJucyBhIGZsb2F0IG51bWJlciBiZXR3ZWVuIDAgYW5kIDEgd2hlcmUgMSBpcyBjb21wbGV0ZWx5IGRpZmZlcmVudFxuICogYW5kIDAgaXMgbm8gZGlmZmVyZW5jZVxuICovXG5leHBvcnQgZGVmYXVsdCBmdW5jdGlvbiBldWNsaWRlYW5EaXN0YW5jZShyZ2JhMSwgcmdiYTIpIHtcbiAgY29uc3QgZGlzdGFuY2UgPSAoKHJnYmExWzBdIC0gcmdiYTJbMF0pICogKHJnYmExWzBdIC0gcmdiYTJbMF0pKVxuICAgICAgICAgICAgICAgICArICgocmdiYTFbMV0gLSByZ2JhMlsxXSkgKiAocmdiYTFbMV0gLSByZ2JhMlsxXSkpXG4gICAgICAgICAgICAgICAgICsgKChyZ2JhMVsyXSAtIHJnYmEyWzJdKSAqIChyZ2JhMVsyXSAtIHJnYmEyWzJdKSlcbiAgICAgICAgICAgICAgICAgKyAoKHJnYmExWzNdIC0gcmdiYTJbM10pICogKHJnYmExWzNdIC0gcmdiYTJbM10pKTtcblxuICByZXR1cm4gKE1hdGguc3FydChkaXN0YW5jZSkgLyA0KSAvIDI1NTtcbn1cblxuXG5cbi8qKiBXRUJQQUNLIEZPT1RFUiAqKlxuICoqIC4vanMvc3JjL2V1Y2xpZGVhbkRpc3RhbmNlLmpzXG4gKiovIl0sInNvdXJjZVJvb3QiOiIifQ==");
60
+
61
+ /***/ }
62
+ /******/ ]);
@@ -0,0 +1,62 @@
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\nvar _flattenImageData = __webpack_require__(2);\n\nvar _flattenImageData2 = _interopRequireDefault(_flattenImageData);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction getAdiffResults(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n\n if (previousData.width !== currentData.width) {\n // we know that all rows will be different here, so we can take a shortcut\n var diff = [0, // diff starts at index 0\n previousData.height];\n diff.length = currentData.height + 2; // number of additions\n return [diff];\n }\n\n var hashedPreviousData = previousImageData.map(JSON.stringify);\n var hashedCurrentData = currentImageData.map(JSON.stringify);\n\n return _adiff2.default.diff(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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n self.postMessage({ progress: 30 });\n\n var adiffResults = getAdiffResults({\n previousData: previousData,\n currentData: currentData,\n previousImageData: previousImageData,\n currentImageData: currentImageData\n });\n\n self.postMessage({ progress: 75 });\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\n self.postMessage({ progress: 80 });\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n self.postMessage({ progress: 85 });\n\n return {\n currentData: {\n data: (0, _flattenImageData2.default)(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: (0, _flattenImageData2.default)(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":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","getAdiffResults","previousData","currentData","previousImageData","currentImageData","diff","length","hashedPreviousData","map","JSON","stringify","hashedCurrentData","computeAndInjectDiffs","maxWidth","Math","max","redLine","greenLine","self","postMessage","progress","adiffResults","forEach","instruction","atIndex","deletedItems","addedItems","splice","addEventListener","result","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\nimport flattenImageData from '../flattenImageData';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction getAdiffResults({\n  previousData,\n  currentData,\n  previousImageData,\n  currentImageData,\n}) {\n  if (previousData.width !== currentData.width) {\n    // we know that all rows will be different here, so we can take a shortcut\n    const diff = [\n      0, // diff starts at index 0\n      previousData.height, // number of deletions\n    ];\n    diff.length = currentData.height + 2; // number of additions\n    return [diff];\n  }\n\n  const hashedPreviousData = previousImageData.map(JSON.stringify);\n  const hashedCurrentData = currentImageData.map(JSON.stringify);\n\n  return adiff.diff(\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\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: 30 });\n\n  const adiffResults = getAdiffResults({\n    previousData,\n    currentData,\n    previousImageData,\n    currentImageData,\n  });\n\n  self.postMessage({ progress: 75 });\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\n  self.postMessage({ progress: 80 });\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n  self.postMessage({ progress: 85 });\n\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(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
+ /* 2 */
57
+ /***/ function(module, exports) {
58
+
59
+ eval("\"use strict\";\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = flattenImageData;\n/**\n * @param {Array} imageData a 2d array\n * @return {Uint8ClampedArray}\n */\nfunction flattenImageData(imageData) {\n var width = imageData[0].length;\n var result = new Uint8ClampedArray(imageData.length * width * 4);\n imageData.forEach(function (row, y) {\n row.forEach(function (pixel, x) {\n var index = y * width * 4 + x * 4;\n result[index] = pixel[0];\n result[index + 1] = pixel[1];\n result[index + 2] = pixel[2];\n result[index + 3] = pixel[3];\n });\n });\n return result;\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZmxhdHRlbkltYWdlRGF0YS5qcz8yZGRiIl0sIm5hbWVzIjpbImZsYXR0ZW5JbWFnZURhdGEiLCJpbWFnZURhdGEiLCJ3aWR0aCIsImxlbmd0aCIsInJlc3VsdCIsIlVpbnQ4Q2xhbXBlZEFycmF5IiwiZm9yRWFjaCIsInJvdyIsInkiLCJwaXhlbCIsIngiLCJpbmRleCJdLCJtYXBwaW5ncyI6Ijs7Ozs7a0JBSXdCQSxnQjtBQUp4Qjs7OztBQUllLFNBQVNBLGdCQUFULENBQTBCQyxTQUExQixFQUFxQztBQUNsRCxNQUFNQyxRQUFRRCxVQUFVLENBQVYsRUFBYUUsTUFBM0I7QUFDQSxNQUFNQyxTQUFTLElBQUlDLGlCQUFKLENBQXNCSixVQUFVRSxNQUFWLEdBQW1CRCxLQUFuQixHQUEyQixDQUFqRCxDQUFmO0FBQ0FELFlBQVVLLE9BQVYsQ0FBa0IsVUFBQ0MsR0FBRCxFQUFNQyxDQUFOLEVBQVk7QUFDNUJELFFBQUlELE9BQUosQ0FBWSxVQUFDRyxLQUFELEVBQVFDLENBQVIsRUFBYztBQUN4QixVQUFNQyxRQUFTSCxJQUFJTixLQUFKLEdBQVksQ0FBYixHQUFtQlEsSUFBSSxDQUFyQztBQUNBTixhQUFPTyxLQUFQLElBQWdCRixNQUFNLENBQU4sQ0FBaEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDRCxLQU5EO0FBT0QsR0FSRDtBQVNBLFNBQU9MLE1BQVA7QUFDRCIsImZpbGUiOiIyLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAcGFyYW0ge0FycmF5fSBpbWFnZURhdGEgYSAyZCBhcnJheVxuICogQHJldHVybiB7VWludDhDbGFtcGVkQXJyYXl9XG4gKi9cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIGZsYXR0ZW5JbWFnZURhdGEoaW1hZ2VEYXRhKSB7XG4gIGNvbnN0IHdpZHRoID0gaW1hZ2VEYXRhWzBdLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0ID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KGltYWdlRGF0YS5sZW5ndGggKiB3aWR0aCAqIDQpO1xuICBpbWFnZURhdGEuZm9yRWFjaCgocm93LCB5KSA9PiB7XG4gICAgcm93LmZvckVhY2goKHBpeGVsLCB4KSA9PiB7XG4gICAgICBjb25zdCBpbmRleCA9ICh5ICogd2lkdGggKiA0KSArICh4ICogNCk7XG4gICAgICByZXN1bHRbaW5kZXhdID0gcGl4ZWxbMF07XG4gICAgICByZXN1bHRbaW5kZXggKyAxXSA9IHBpeGVsWzFdO1xuICAgICAgcmVzdWx0W2luZGV4ICsgMl0gPSBwaXhlbFsyXTtcbiAgICAgIHJlc3VsdFtpbmRleCArIDNdID0gcGl4ZWxbM107XG4gICAgfSk7XG4gIH0pO1xuICByZXR1cm4gcmVzdWx0O1xufVxuXG5cblxuLyoqIFdFQlBBQ0sgRk9PVEVSICoqXG4gKiogLi9qcy9zcmMvZmxhdHRlbkltYWdlRGF0YS5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
60
+
61
+ /***/ }
62
+ /******/ ]);
@@ -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\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\n\nfunction imageTo2DArray(_ref) {\n var imageData = _ref.imageData;\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 var getPixelAt = function getPixelAt(x, y) {\n if (x > width || y > height) {\n return undefined;\n }\n\n var startIndex = y * rowSize + x * 4;\n return [imageData[startIndex], imageData[startIndex + 1], imageData[startIndex + 2], imageData[startIndex + 3]];\n };\n\n var data = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n data.push(pixelsInRow);\n }\n return data;\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(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n\n var maxWidth = Math.max(previousData[0].length, currentData[0].length);\n\n var redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n var adiffResults = _adiff2.default.diff(previousData.map(function (d) {\n return btoa(d);\n }), currentData.map(function (d) {\n return btoa(d);\n }));\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 // add a green line to signal an addition\n previousData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\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 // add a red line to signal a deletion\n currentData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n}\n\nself.addEventListener('message', function (_ref3) {\n var _ref3$data = _ref3.data;\n var previousData = _ref3$data.previousData;\n var currentData = _ref3$data.currentData;\n\n computeAndInjectDiffs({ previousData: previousData, currentData: currentData });\n self.postMessage({ previousData: previousData, currentData: currentData });\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./js/src/workers/ComputeAndInjectDiffsWorker.js?ddb5"],"names":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","imageData","height","rowSize","getPixelAt","x","y","undefined","startIndex","data","row","pixelsInRow","col","computeAndInjectDiffs","previousData","currentData","maxWidth","Math","max","length","redLine","greenLine","adiffResults","diff","map","btoa","d","forEach","instruction","atIndex","deletedItems","addedItems","splice","self","addEventListener","postMessage","close"],"mappings":";;AAAA;;;;;;AAEA;;;;;;;AAOA,SAASA,oBAAT,CAA8BC,IAA9B,EAAoCC,KAApC,EAA2C;AACzC,MAAMC,OAAO,EAAb;AACA,OAAK,IAAIC,IAAI,CAAb,EAAgBA,IAAIF,KAApB,EAA2BE,GAA3B,EAAgC;AAC9BD,SAAKE,IAAL,CAAUJ,IAAV;AACD;AACD,SAAOE,IAAP;AACD;;AAED,SAASG,cAAT,OAAsD;AAAA,MAA5BC,SAA4B,QAA5BA,SAA4B;AAAA,MAAjBL,KAAiB,QAAjBA,KAAiB;AAAA,MAAVM,MAAU,QAAVA,MAAU;;AACpD;AACA;AACA,MAAMC,UAAUP,QAAQ,CAAxB;AACA,MAAMQ,aAAa,SAAbA,UAAa,CAACC,CAAD,EAAIC,CAAJ,EAAU;AAC3B,QAAID,IAAIT,KAAJ,IAAaU,IAAIJ,MAArB,EAA6B;AAC3B,aAAOK,SAAP;AACD;;AAED,QAAMC,aAAcF,IAAIH,OAAL,GAAiBE,IAAI,CAAxC;AACA,WAAO,CACLJ,UAAUO,UAAV,CADK,EAELP,UAAUO,aAAa,CAAvB,CAFK,EAGLP,UAAUO,aAAa,CAAvB,CAHK,EAILP,UAAUO,aAAa,CAAvB,CAJK,CAAP;AAMD,GAZD;;AAcA,MAAMC,OAAO,EAAb;AACA,OAAK,IAAIC,MAAM,CAAf,EAAkBA,MAAMR,MAAxB,EAAgCQ,KAAhC,EAAuC;AACrC,QAAMC,cAAc,EAApB;AACA,SAAK,IAAIC,MAAM,CAAf,EAAkBA,MAAMhB,KAAxB,EAA+BgB,KAA/B,EAAsC;AACpCD,kBAAYZ,IAAZ,CAAiBK,WAAWQ,GAAX,EAAgBF,GAAhB,CAAjB;AACD;AACDD,SAAKV,IAAL,CAAUY,WAAV;AACD;AACD,SAAOF,IAAP;AACD;;AAED;;;;;;;;;;;AAWA,SAASI,qBAAT,QAA8D;AAAA,MAA7BC,YAA6B,SAA7BA,YAA6B;AAAA,MAAfC,WAAe,SAAfA,WAAe;;AAC5D,MAAMC,WAAWC,KAAKC,GAAL,CACfJ,aAAa,CAAb,EAAgBK,MADD,EACSJ,YAAY,CAAZ,EAAeI,MADxB,CAAjB;;AAGA,MAAMC,UAAU1B,qBAAqB,CAAC,GAAD,EAAM,CAAN,EAAS,CAAT,EAAY,GAAZ,CAArB,EAAuCsB,QAAvC,CAAhB;AACA,MAAMK,YAAY3B,qBAAqB,CAAC,CAAD,EAAI,GAAJ,EAAS,CAAT,EAAY,GAAZ,CAArB,EAAuCsB,QAAvC,CAAlB;;AAEA,MAAMM,eAAe,gBAAMC,IAAN,CACnBT,aAAaU,GAAb,CAAiB;AAAA,WAAKC,KAAKC,CAAL,CAAL;AAAA,GAAjB,CADmB,EACaX,YAAYS,GAAZ,CAAgB;AAAA,WAAKC,KAAKC,CAAL,CAAL;AAAA,GAAhB,CADb,CAArB;;AAGA;AACAJ,eAAaK,OAAb,CAAqB,UAACC,WAAD,EAAiB;AACpC,QAAMC,UAAUD,YAAY,CAAZ,CAAhB;AACA,QAAME,eAAeF,YAAY,CAAZ,CAArB;AACA,QAAMG,aAAaH,YAAYT,MAAZ,GAAqB,CAAxC;;AAEA,SAAK,IAAIb,IAAI,CAAb,EAAgBA,IAAIW,KAAKC,GAAL,CAASY,YAAT,EAAuBC,UAAvB,CAApB,EAAwDzB,GAAxD,EAA6D;AAC3D,UAAIA,IAAIwB,YAAR,EAAsB;AACpB;AACD,OAFD,MAEO;AACL;AACAhB,qBAAakB,MAAb,CAAoBH,UAAUvB,CAA9B,EAAiC,CAAjC,EAAoCe,SAApC;AACD;AACF;AACF,GAbD;;AAeA;AACA,OAAK,IAAIvB,IAAIwB,aAAaH,MAAb,GAAsB,CAAnC,EAAsCrB,KAAK,CAA3C,EAA8CA,GAA9C,EAAmD;AACjD,QAAM8B,cAAcN,aAAaxB,CAAb,CAApB;AACA,QAAM+B,UAAUD,YAAY,CAAZ,CAAhB;AACA,QAAME,eAAeF,YAAY,CAAZ,CAArB;AACA,QAAMG,aAAaH,YAAYT,MAAZ,GAAqB,CAAxC;;AAEA,SAAK,IAAIb,IAAI,CAAb,EAAgBA,IAAIW,KAAKC,GAAL,CAASY,YAAT,EAAuBC,UAAvB,CAApB,EAAwDzB,GAAxD,EAA6D;AAC3D,UAAIA,IAAIyB,UAAR,EAAoB;AAClB;AACD,OAFD,MAEO;AACL;AACAhB,oBAAYiB,MAAZ,CAAmBH,UAAUvB,CAA7B,EAAgC,CAAhC,EAAmCc,OAAnC;AACD;AACF;AACF;AACF;;AAEDa,KAAKC,gBAAL,CAAsB,SAAtB,EAAiC,iBAA6C;AAAA,yBAA1CzB,IAA0C;AAAA,MAAlCK,YAAkC,cAAlCA,YAAkC;AAAA,MAApBC,WAAoB,cAApBA,WAAoB;;AAC5EF,wBAAsB,EAAEC,0BAAF,EAAgBC,wBAAhB,EAAtB;AACAkB,OAAKE,WAAL,CAAiB,EAAErB,0BAAF,EAAgBC,wBAAhB,EAAjB;AACAkB,OAAKG,KAAL;AACD,CAJD","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\n\nfunction imageTo2DArray({ imageData, width, height }) {\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  const getPixelAt = (x, y) => {\n    if (x > width || y > height) {\n      return undefined;\n    }\n\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      imageData[startIndex],\n      imageData[startIndex + 1],\n      imageData[startIndex + 2],\n      imageData[startIndex + 3],\n    ];\n  };\n\n  const data = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    data.push(pixelsInRow);\n  }\n  return data;\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(\n    previousData[0].length, currentData[0].length);\n\n  const redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n  const adiffResults = adiff.diff(\n    previousData.map(d => btoa(d)), currentData.map(d => btoa(d)));\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        // add a green line to signal an addition\n        previousData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\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        // add a red line to signal a deletion\n        currentData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n}\n\nself.addEventListener('message', ({ data: { previousData, currentData } }) => {\n  computeAndInjectDiffs({ previousData, currentData });\n  self.postMessage({ previousData, currentData });\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
+ /******/ ]);
@@ -44,7 +44,7 @@
44
44
  /* 0 */
45
45
  /***/ function(module, exports, __webpack_require__) {
46
46
 
47
- eval("'use strict';\n\nvar _adiff = __webpack_require__(1);\n\nvar _adiff2 = _interopRequireDefault(_adiff);\n\nvar _flattenImageData = __webpack_require__(2);\n\nvar _flattenImageData2 = _interopRequireDefault(_flattenImageData);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nvar TRANSPARENT_PIXEL = [0, 0, 0, 0];\n\n/**\n * Construct a line of transparent pixels\n *\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(TRANSPARENT_PIXEL);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction getAdiffResults(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n\n if (previousData.width !== currentData.width) {\n // we know that all rows will be different here, so we can take a shortcut\n var diff = [0, // diff starts at index 0\n previousData.height];\n diff.length = currentData.height + 2; // number of additions\n return [diff];\n }\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 return _adiff2.default.diff(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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n self.postMessage({ progress: 98 });\n\n return {\n currentData: {\n data: (0, _flattenImageData2.default)(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: (0, _flattenImageData2.default)(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":["TRANSPARENT_PIXEL","constructColoredLine","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","getAdiffResults","previousData","currentData","previousImageData","currentImageData","diff","length","hashedPreviousData","map","JSON","stringify","self","postMessage","progress","hashedCurrentData","computeAndInjectDiffs","maxWidth","Math","max","redLine","greenLine","adiffResults","forEach","instruction","atIndex","deletedItems","addedItems","splice","addEventListener","result","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\nimport flattenImageData from '../flattenImageData';\n\nconst TRANSPARENT_PIXEL = [0, 0, 0, 0];\n\n/**\n * Construct a line of transparent pixels\n *\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(TRANSPARENT_PIXEL);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction getAdiffResults({\n  previousData,\n  currentData,\n  previousImageData,\n  currentImageData,\n}) {\n  if (previousData.width !== currentData.width) {\n    // we know that all rows will be different here, so we can take a shortcut\n    const diff = [\n      0, // diff starts at index 0\n      previousData.height, // number of deletions\n    ];\n    diff.length = currentData.height + 2; // number of additions\n    return [diff];\n  }\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  return adiff.diff(\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n  self.postMessage({ progress: 98 });\n\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(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":""}");
47
+ eval("'use strict';\n\nvar _adiff = __webpack_require__(1);\n\nvar _adiff2 = _interopRequireDefault(_adiff);\n\nvar _flattenImageData = __webpack_require__(2);\n\nvar _flattenImageData2 = _interopRequireDefault(_flattenImageData);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction hash(a) {\n var res = 0;\n for (var i = 0; i < a.length; i++) {\n res += a[i];\n }\n}\n\nfunction getAdiffResults(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n\n if (previousData.width !== currentData.width) {\n // we know that all rows will be different here, so we can take a shortcut\n var diff = [0, // diff starts at index 0\n previousData.height];\n diff.length = currentData.height + 2; // number of additions\n return [diff];\n }\n var start = performance.now();\n var p = previousImageData.map(hash);\n var c = currentImageData.map(hash);\n\n console.log('hashing', performance.now() - start);\n\n var r = _adiff2.default.diff(p, c);\n console.log('total', performance.now() - start);\n return r;\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n var adiffResults = getAdiffResults({\n previousData: previousData,\n currentData: currentData,\n previousImageData: previousImageData,\n currentImageData: currentImageData\n });\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n\n return {\n currentData: {\n data: (0, _flattenImageData2.default)(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: (0, _flattenImageData2.default)(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":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","hash","a","res","length","getAdiffResults","previousData","currentData","previousImageData","currentImageData","diff","start","performance","now","p","map","c","console","log","r","computeAndInjectDiffs","maxWidth","Math","max","redLine","greenLine","adiffResults","forEach","instruction","atIndex","deletedItems","addedItems","splice","self","addEventListener","result","postMessage","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\nimport flattenImageData from '../flattenImageData';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction hash(a) {\n  let res = 0;\n  for (let i = 0; i < a.length; i++) {\n    res += a[i];\n  }\n}\n\nfunction getAdiffResults({\n  previousData,\n  currentData,\n  previousImageData,\n  currentImageData,\n}) {\n  if (previousData.width !== currentData.width) {\n    // we know that all rows will be different here, so we can take a shortcut\n    const diff = [\n      0, // diff starts at index 0\n      previousData.height, // number of deletions\n    ];\n    diff.length = currentData.height + 2; // number of additions\n    return [diff];\n  }\n  const start = performance.now();\n  const p = previousImageData.map(hash);\n  const c = currentImageData.map(hash);\n\n  console.log('hashing', performance.now() - start);\n\n  const r = adiff.diff(p, c);\n  console.log('total', performance.now() - start);\n  return r;\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n  const previousImageData = imageTo2DArray(\n    previousData, maxWidth - previousData.width);\n\n  const currentImageData = imageTo2DArray(\n    currentData, maxWidth - currentData.width);\n\n  const adiffResults = getAdiffResults({\n    previousData,\n    currentData,\n    previousImageData,\n    currentImageData,\n  });\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(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
48
 
49
49
  /***/ },
50
50
  /* 1 */
@@ -0,0 +1,62 @@
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 _getDiffPixel = __webpack_require__(1);\n\nvar _getDiffPixel2 = _interopRequireDefault(_getDiffPixel);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nself.addEventListener('message', function (_ref) {\n var _ref$data = _ref.data;\n var previous = _ref$data.previous;\n var current = _ref$data.current;\n var offset = _ref$data.offset;\n var chunkSize = _ref$data.chunkSize;\n\n var data = new Uint8ClampedArray(previous.length);\n\n for (var i = offset; i < chunkSize; i += 4) {\n var pixel = (0, _getDiffPixel2.default)(previous.slice(i, i + 4), current.slice(i, i + 4));\n data[i + 0] = pixel[0]; // r\n data[i + 1] = pixel[1]; // g\n data[i + 2] = pixel[2]; // b\n data[i + 3] = pixel[3]; // a\n }\n self.postMessage(data);\n self.close();\n});//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvd29ya2Vycy9JbWFnZURpZmZXb3JrZXIuanM/ZjExYiJdLCJuYW1lcyI6WyJzZWxmIiwiYWRkRXZlbnRMaXN0ZW5lciIsImRhdGEiLCJwcmV2aW91cyIsImN1cnJlbnQiLCJvZmZzZXQiLCJjaHVua1NpemUiLCJVaW50OENsYW1wZWRBcnJheSIsImxlbmd0aCIsImkiLCJwaXhlbCIsInNsaWNlIiwicG9zdE1lc3NhZ2UiLCJjbG9zZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQTs7Ozs7O0FBRUFBLEtBQUtDLGdCQUFMLENBQXNCLFNBQXRCLEVBQWlDLGdCQUt6QjtBQUFBLHVCQUw0QkMsSUFLNUI7QUFBQSxNQUpOQyxRQUlNLGFBSk5BLFFBSU07QUFBQSxNQUhOQyxPQUdNLGFBSE5BLE9BR007QUFBQSxNQUZOQyxNQUVNLGFBRk5BLE1BRU07QUFBQSxNQUROQyxTQUNNLGFBRE5BLFNBQ007O0FBQ04sTUFBTUosT0FBTyxJQUFJSyxpQkFBSixDQUFzQkosU0FBU0ssTUFBL0IsQ0FBYjs7QUFFQSxPQUFLLElBQUlDLElBQUlKLE1BQWIsRUFBcUJJLElBQUlILFNBQXpCLEVBQW9DRyxLQUFLLENBQXpDLEVBQTRDO0FBQzFDLFFBQU1DLFFBQVEsNEJBQ1pQLFNBQVNRLEtBQVQsQ0FBZUYsQ0FBZixFQUFrQkEsSUFBSSxDQUF0QixDQURZLEVBRVpMLFFBQVFPLEtBQVIsQ0FBY0YsQ0FBZCxFQUFpQkEsSUFBSSxDQUFyQixDQUZZLENBQWQ7QUFJQVAsU0FBS08sSUFBSSxDQUFULElBQWNDLE1BQU0sQ0FBTixDQUFkLENBTDBDLENBS2xCO0FBQ3hCUixTQUFLTyxJQUFJLENBQVQsSUFBY0MsTUFBTSxDQUFOLENBQWQsQ0FOMEMsQ0FNbEI7QUFDeEJSLFNBQUtPLElBQUksQ0FBVCxJQUFjQyxNQUFNLENBQU4sQ0FBZCxDQVAwQyxDQU9sQjtBQUN4QlIsU0FBS08sSUFBSSxDQUFULElBQWNDLE1BQU0sQ0FBTixDQUFkLENBUjBDLENBUWxCO0FBQ3pCO0FBQ0RWLE9BQUtZLFdBQUwsQ0FBaUJWLElBQWpCO0FBQ0FGLE9BQUthLEtBQUw7QUFDRCxDQXBCRCIsImZpbGUiOiIwLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGdldERpZmZQaXhlbCBmcm9tICcuLi9nZXREaWZmUGl4ZWwnO1xuXG5zZWxmLmFkZEV2ZW50TGlzdGVuZXIoJ21lc3NhZ2UnLCAoeyBkYXRhOiB7XG4gIHByZXZpb3VzLFxuICBjdXJyZW50LFxuICBvZmZzZXQsXG4gIGNodW5rU2l6ZSxcbn0gfSkgPT4ge1xuICBjb25zdCBkYXRhID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KHByZXZpb3VzLmxlbmd0aCk7XG5cbiAgZm9yIChsZXQgaSA9IG9mZnNldDsgaSA8IGNodW5rU2l6ZTsgaSArPSA0KSB7XG4gICAgY29uc3QgcGl4ZWwgPSBnZXREaWZmUGl4ZWwoXG4gICAgICBwcmV2aW91cy5zbGljZShpLCBpICsgNCksXG4gICAgICBjdXJyZW50LnNsaWNlKGksIGkgKyA0KVxuICAgICk7XG4gICAgZGF0YVtpICsgMF0gPSBwaXhlbFswXTsgLy8gclxuICAgIGRhdGFbaSArIDFdID0gcGl4ZWxbMV07IC8vIGdcbiAgICBkYXRhW2kgKyAyXSA9IHBpeGVsWzJdOyAvLyBiXG4gICAgZGF0YVtpICsgM10gPSBwaXhlbFszXTsgLy8gYVxuICB9XG4gIHNlbGYucG9zdE1lc3NhZ2UoZGF0YSk7XG4gIHNlbGYuY2xvc2UoKTtcbn0pO1xuXG5cblxuLyoqIFdFQlBBQ0sgRk9PVEVSICoqXG4gKiogLi9qcy9zcmMvd29ya2Vycy9JbWFnZURpZmZXb3JrZXIuanNcbiAqKi8iXSwic291cmNlUm9vdCI6IiJ9");
48
+
49
+ /***/ },
50
+ /* 1 */
51
+ /***/ function(module, exports, __webpack_require__) {
52
+
53
+ eval("'use strict';\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = getDiffPixel;\n\nvar _euclideanDistance = __webpack_require__(2);\n\nvar _euclideanDistance2 = _interopRequireDefault(_euclideanDistance);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\nfunction getDiffPixel(previousPixel, currentPixel) {\n if (!previousPixel) {\n return currentPixel;\n }\n\n if (!currentPixel) {\n return previousPixel;\n }\n\n var diff = (0, _euclideanDistance2.default)(previousPixel, currentPixel);\n if (diff === 0) {\n return [currentPixel[0], currentPixel[1], currentPixel[2], 50];\n }\n\n if (diff < 0.2) {\n diff = 0.2;\n }\n return [255, 0, 0, 255 * diff]; // TODO don't use red here\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZ2V0RGlmZlBpeGVsLmpzPzRiNjgiXSwibmFtZXMiOlsiZ2V0RGlmZlBpeGVsIiwicHJldmlvdXNQaXhlbCIsImN1cnJlbnRQaXhlbCIsImRpZmYiXSwibWFwcGluZ3MiOiI7Ozs7O2tCQUV3QkEsWTs7QUFGeEI7Ozs7OztBQUVlLFNBQVNBLFlBQVQsQ0FBc0JDLGFBQXRCLEVBQXFDQyxZQUFyQyxFQUFtRDtBQUNoRSxNQUFJLENBQUNELGFBQUwsRUFBb0I7QUFDbEIsV0FBT0MsWUFBUDtBQUNEOztBQUVELE1BQUksQ0FBQ0EsWUFBTCxFQUFtQjtBQUNqQixXQUFPRCxhQUFQO0FBQ0Q7O0FBRUQsTUFBSUUsT0FBTyxpQ0FBa0JGLGFBQWxCLEVBQWlDQyxZQUFqQyxDQUFYO0FBQ0EsTUFBSUMsU0FBUyxDQUFiLEVBQWdCO0FBQ2QsV0FBTyxDQUNMRCxhQUFhLENBQWIsQ0FESyxFQUVMQSxhQUFhLENBQWIsQ0FGSyxFQUdMQSxhQUFhLENBQWIsQ0FISyxFQUlMLEVBSkssQ0FBUDtBQU1EOztBQUVELE1BQUlDLE9BQU8sR0FBWCxFQUFnQjtBQUNkQSxXQUFPLEdBQVA7QUFDRDtBQUNELFNBQU8sQ0FBQyxHQUFELEVBQU0sQ0FBTixFQUFTLENBQVQsRUFBWSxNQUFNQSxJQUFsQixDQUFQLENBdEJnRSxDQXNCaEM7QUFDakMiLCJmaWxlIjoiMS5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBldWNsaWRlYW5EaXN0YW5jZSBmcm9tICcuL2V1Y2xpZGVhbkRpc3RhbmNlJztcblxuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gZ2V0RGlmZlBpeGVsKHByZXZpb3VzUGl4ZWwsIGN1cnJlbnRQaXhlbCkge1xuICBpZiAoIXByZXZpb3VzUGl4ZWwpIHtcbiAgICByZXR1cm4gY3VycmVudFBpeGVsO1xuICB9XG5cbiAgaWYgKCFjdXJyZW50UGl4ZWwpIHtcbiAgICByZXR1cm4gcHJldmlvdXNQaXhlbDtcbiAgfVxuXG4gIGxldCBkaWZmID0gZXVjbGlkZWFuRGlzdGFuY2UocHJldmlvdXNQaXhlbCwgY3VycmVudFBpeGVsKTtcbiAgaWYgKGRpZmYgPT09IDApIHtcbiAgICByZXR1cm4gW1xuICAgICAgY3VycmVudFBpeGVsWzBdLFxuICAgICAgY3VycmVudFBpeGVsWzFdLFxuICAgICAgY3VycmVudFBpeGVsWzJdLFxuICAgICAgNTAsXG4gICAgXTtcbiAgfVxuXG4gIGlmIChkaWZmIDwgMC4yKSB7XG4gICAgZGlmZiA9IDAuMjtcbiAgfVxuICByZXR1cm4gWzI1NSwgMCwgMCwgMjU1ICogZGlmZl07IC8vIFRPRE8gZG9uJ3QgdXNlIHJlZCBoZXJlXG59XG5cblxuXG4vKiogV0VCUEFDSyBGT09URVIgKipcbiAqKiAuL2pzL3NyYy9nZXREaWZmUGl4ZWwuanNcbiAqKi8iXSwic291cmNlUm9vdCI6IiJ9");
54
+
55
+ /***/ },
56
+ /* 2 */
57
+ /***/ function(module, exports) {
58
+
59
+ eval("\"use strict\";\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = euclideanDistance;\n/**\n * Compute a score that represents the difference between 2 pixels\n *\n * This method simply takes the Euclidean distance between the RGBA channels\n * of 2 colors over the maximum possible Euclidean distance. This gives us a\n * percentage of how different the two colors are.\n *\n * Although it would be more perceptually accurate to calculate a proper\n * Delta E in Lab colorspace, we probably don't need perceptual accuracy for\n * this application, and it is nice to avoid the overhead of converting RGBA\n * to Lab.\n *\n * Returns a float number between 0 and 1 where 1 is completely different\n * and 0 is no difference\n */\nfunction euclideanDistance(rgba1, rgba2) {\n var distance = (rgba1[0] - rgba2[0]) * (rgba1[0] - rgba2[0]) + (rgba1[1] - rgba2[1]) * (rgba1[1] - rgba2[1]) + (rgba1[2] - rgba2[2]) * (rgba1[2] - rgba2[2]) + (rgba1[3] - rgba2[3]) * (rgba1[3] - rgba2[3]);\n\n return Math.sqrt(distance) / 4 / 255;\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZXVjbGlkZWFuRGlzdGFuY2UuanM/ZDgzYiJdLCJuYW1lcyI6WyJldWNsaWRlYW5EaXN0YW5jZSIsInJnYmExIiwicmdiYTIiLCJkaXN0YW5jZSIsIk1hdGgiLCJzcXJ0Il0sIm1hcHBpbmdzIjoiOzs7OztrQkFld0JBLGlCO0FBZnhCOzs7Ozs7Ozs7Ozs7Ozs7QUFlZSxTQUFTQSxpQkFBVCxDQUEyQkMsS0FBM0IsRUFBa0NDLEtBQWxDLEVBQXlDO0FBQ3RELE1BQU1DLFdBQVksQ0FBQ0YsTUFBTSxDQUFOLElBQVdDLE1BQU0sQ0FBTixDQUFaLEtBQXlCRCxNQUFNLENBQU4sSUFBV0MsTUFBTSxDQUFOLENBQXBDLENBQUQsR0FDQyxDQUFDRCxNQUFNLENBQU4sSUFBV0MsTUFBTSxDQUFOLENBQVosS0FBeUJELE1BQU0sQ0FBTixJQUFXQyxNQUFNLENBQU4sQ0FBcEMsQ0FERCxHQUVDLENBQUNELE1BQU0sQ0FBTixJQUFXQyxNQUFNLENBQU4sQ0FBWixLQUF5QkQsTUFBTSxDQUFOLElBQVdDLE1BQU0sQ0FBTixDQUFwQyxDQUZELEdBR0MsQ0FBQ0QsTUFBTSxDQUFOLElBQVdDLE1BQU0sQ0FBTixDQUFaLEtBQXlCRCxNQUFNLENBQU4sSUFBV0MsTUFBTSxDQUFOLENBQXBDLENBSGxCOztBQUtBLFNBQVFFLEtBQUtDLElBQUwsQ0FBVUYsUUFBVixJQUFzQixDQUF2QixHQUE0QixHQUFuQztBQUNEIiwiZmlsZSI6IjIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENvbXB1dGUgYSBzY29yZSB0aGF0IHJlcHJlc2VudHMgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiAyIHBpeGVsc1xuICpcbiAqIFRoaXMgbWV0aG9kIHNpbXBseSB0YWtlcyB0aGUgRXVjbGlkZWFuIGRpc3RhbmNlIGJldHdlZW4gdGhlIFJHQkEgY2hhbm5lbHNcbiAqIG9mIDIgY29sb3JzIG92ZXIgdGhlIG1heGltdW0gcG9zc2libGUgRXVjbGlkZWFuIGRpc3RhbmNlLiBUaGlzIGdpdmVzIHVzIGFcbiAqIHBlcmNlbnRhZ2Ugb2YgaG93IGRpZmZlcmVudCB0aGUgdHdvIGNvbG9ycyBhcmUuXG4gKlxuICogQWx0aG91Z2ggaXQgd291bGQgYmUgbW9yZSBwZXJjZXB0dWFsbHkgYWNjdXJhdGUgdG8gY2FsY3VsYXRlIGEgcHJvcGVyXG4gKiBEZWx0YSBFIGluIExhYiBjb2xvcnNwYWNlLCB3ZSBwcm9iYWJseSBkb24ndCBuZWVkIHBlcmNlcHR1YWwgYWNjdXJhY3kgZm9yXG4gKiB0aGlzIGFwcGxpY2F0aW9uLCBhbmQgaXQgaXMgbmljZSB0byBhdm9pZCB0aGUgb3ZlcmhlYWQgb2YgY29udmVydGluZyBSR0JBXG4gKiB0byBMYWIuXG4gKlxuICogUmV0dXJucyBhIGZsb2F0IG51bWJlciBiZXR3ZWVuIDAgYW5kIDEgd2hlcmUgMSBpcyBjb21wbGV0ZWx5IGRpZmZlcmVudFxuICogYW5kIDAgaXMgbm8gZGlmZmVyZW5jZVxuICovXG5leHBvcnQgZGVmYXVsdCBmdW5jdGlvbiBldWNsaWRlYW5EaXN0YW5jZShyZ2JhMSwgcmdiYTIpIHtcbiAgY29uc3QgZGlzdGFuY2UgPSAoKHJnYmExWzBdIC0gcmdiYTJbMF0pICogKHJnYmExWzBdIC0gcmdiYTJbMF0pKVxuICAgICAgICAgICAgICAgICArICgocmdiYTFbMV0gLSByZ2JhMlsxXSkgKiAocmdiYTFbMV0gLSByZ2JhMlsxXSkpXG4gICAgICAgICAgICAgICAgICsgKChyZ2JhMVsyXSAtIHJnYmEyWzJdKSAqIChyZ2JhMVsyXSAtIHJnYmEyWzJdKSlcbiAgICAgICAgICAgICAgICAgKyAoKHJnYmExWzNdIC0gcmdiYTJbM10pICogKHJnYmExWzNdIC0gcmdiYTJbM10pKTtcblxuICByZXR1cm4gKE1hdGguc3FydChkaXN0YW5jZSkgLyA0KSAvIDI1NTtcbn1cblxuXG5cbi8qKiBXRUJQQUNLIEZPT1RFUiAqKlxuICoqIC4vanMvc3JjL2V1Y2xpZGVhbkRpc3RhbmNlLmpzXG4gKiovIl0sInNvdXJjZVJvb3QiOiIifQ==");
60
+
61
+ /***/ }
62
+ /******/ ]);
@@ -0,0 +1,62 @@
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\nvar _flattenImageData = __webpack_require__(2);\n\nvar _flattenImageData2 = _interopRequireDefault(_flattenImageData);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction getAdiffResults(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n\n if (previousData.width !== currentData.width) {\n // we know that all rows will be different here, so we can take a shortcut\n var diff = [0, // diff starts at index 0\n previousData.height];\n diff.length = currentData.height + 2; // number of additions\n }\n return _adiff2.default.diff(previousImageData.map(function (d) {\n return btoa(d);\n }), currentImageData.map(function (d) {\n return btoa(d);\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n var adiffResults = getAdiffResults({\n previousData: previousData,\n currentData: currentData,\n previousImageData: previousImageData,\n currentImageData: currentImageData\n });\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n\n return {\n currentData: {\n data: (0, _flattenImageData2.default)(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: (0, _flattenImageData2.default)(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":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","getAdiffResults","previousData","currentData","previousImageData","currentImageData","diff","length","map","btoa","d","computeAndInjectDiffs","maxWidth","Math","max","redLine","greenLine","adiffResults","forEach","instruction","atIndex","deletedItems","addedItems","splice","self","addEventListener","result","postMessage","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\nimport flattenImageData from '../flattenImageData';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction getAdiffResults({\n  previousData,\n  currentData,\n  previousImageData,\n  currentImageData,\n}) {\n  if (previousData.width !== currentData.width) {\n    // we know that all rows will be different here, so we can take a shortcut\n    const diff = [\n      0, // diff starts at index 0\n      previousData.height, // number of deletions\n    ];\n    diff.length = currentData.height + 2; // number of additions\n  }\n  return adiff.diff(\n    previousImageData.map(d => btoa(d)),\n    currentImageData.map(d => btoa(d))\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n  const previousImageData = imageTo2DArray(\n    previousData, maxWidth - previousData.width);\n\n  const currentImageData = imageTo2DArray(\n    currentData, maxWidth - currentData.width);\n\n  const adiffResults = getAdiffResults({\n    previousData,\n    currentData,\n    previousImageData,\n    currentImageData,\n  });\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(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
+ /* 2 */
57
+ /***/ function(module, exports) {
58
+
59
+ eval("\"use strict\";\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = flattenImageData;\n/**\n * @param {Array} imageData a 2d array\n * @return {Uint8ClampedArray}\n */\nfunction flattenImageData(imageData) {\n var width = imageData[0].length;\n var result = new Uint8ClampedArray(imageData.length * width * 4);\n imageData.forEach(function (row, y) {\n row.forEach(function (pixel, x) {\n var index = y * width * 4 + x * 4;\n result[index] = pixel[0];\n result[index + 1] = pixel[1];\n result[index + 2] = pixel[2];\n result[index + 3] = pixel[3];\n });\n });\n return result;\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZmxhdHRlbkltYWdlRGF0YS5qcz8yZGRiIl0sIm5hbWVzIjpbImZsYXR0ZW5JbWFnZURhdGEiLCJpbWFnZURhdGEiLCJ3aWR0aCIsImxlbmd0aCIsInJlc3VsdCIsIlVpbnQ4Q2xhbXBlZEFycmF5IiwiZm9yRWFjaCIsInJvdyIsInkiLCJwaXhlbCIsIngiLCJpbmRleCJdLCJtYXBwaW5ncyI6Ijs7Ozs7a0JBSXdCQSxnQjtBQUp4Qjs7OztBQUllLFNBQVNBLGdCQUFULENBQTBCQyxTQUExQixFQUFxQztBQUNsRCxNQUFNQyxRQUFRRCxVQUFVLENBQVYsRUFBYUUsTUFBM0I7QUFDQSxNQUFNQyxTQUFTLElBQUlDLGlCQUFKLENBQXNCSixVQUFVRSxNQUFWLEdBQW1CRCxLQUFuQixHQUEyQixDQUFqRCxDQUFmO0FBQ0FELFlBQVVLLE9BQVYsQ0FBa0IsVUFBQ0MsR0FBRCxFQUFNQyxDQUFOLEVBQVk7QUFDNUJELFFBQUlELE9BQUosQ0FBWSxVQUFDRyxLQUFELEVBQVFDLENBQVIsRUFBYztBQUN4QixVQUFNQyxRQUFTSCxJQUFJTixLQUFKLEdBQVksQ0FBYixHQUFtQlEsSUFBSSxDQUFyQztBQUNBTixhQUFPTyxLQUFQLElBQWdCRixNQUFNLENBQU4sQ0FBaEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDRCxLQU5EO0FBT0QsR0FSRDtBQVNBLFNBQU9MLE1BQVA7QUFDRCIsImZpbGUiOiIyLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAcGFyYW0ge0FycmF5fSBpbWFnZURhdGEgYSAyZCBhcnJheVxuICogQHJldHVybiB7VWludDhDbGFtcGVkQXJyYXl9XG4gKi9cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIGZsYXR0ZW5JbWFnZURhdGEoaW1hZ2VEYXRhKSB7XG4gIGNvbnN0IHdpZHRoID0gaW1hZ2VEYXRhWzBdLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0ID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KGltYWdlRGF0YS5sZW5ndGggKiB3aWR0aCAqIDQpO1xuICBpbWFnZURhdGEuZm9yRWFjaCgocm93LCB5KSA9PiB7XG4gICAgcm93LmZvckVhY2goKHBpeGVsLCB4KSA9PiB7XG4gICAgICBjb25zdCBpbmRleCA9ICh5ICogd2lkdGggKiA0KSArICh4ICogNCk7XG4gICAgICByZXN1bHRbaW5kZXhdID0gcGl4ZWxbMF07XG4gICAgICByZXN1bHRbaW5kZXggKyAxXSA9IHBpeGVsWzFdO1xuICAgICAgcmVzdWx0W2luZGV4ICsgMl0gPSBwaXhlbFsyXTtcbiAgICAgIHJlc3VsdFtpbmRleCArIDNdID0gcGl4ZWxbM107XG4gICAgfSk7XG4gIH0pO1xuICByZXR1cm4gcmVzdWx0O1xufVxuXG5cblxuLyoqIFdFQlBBQ0sgRk9PVEVSICoqXG4gKiogLi9qcy9zcmMvZmxhdHRlbkltYWdlRGF0YS5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
60
+
61
+ /***/ }
62
+ /******/ ]);
@@ -0,0 +1,62 @@
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\nvar _flattenImageData = __webpack_require__(2);\n\nvar _flattenImageData2 = _interopRequireDefault(_flattenImageData);\n\nfunction _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n var line = [];\n for (var i = 0; i < width; i++) {\n line.push(rgba);\n }\n return line;\n}\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 var getPixelAt = function getPixelAt(x, y) {\n var startIndex = y * rowSize + x * 4;\n return [data[startIndex], data[startIndex + 1], data[startIndex + 2], data[startIndex + 3]];\n };\n\n var newData = [];\n for (var row = 0; row < height; row++) {\n var pixelsInRow = [];\n for (var col = 0; col < width; col++) {\n pixelsInRow.push(getPixelAt(col, row));\n }\n for (var pad = 0; pad < paddingRight; pad++) {\n pixelsInRow.push([0, 0, 0, 0]);\n }\n newData.push(pixelsInRow);\n }\n return newData;\n}\n\nfunction getAdiffResults(_ref2) {\n var previousData = _ref2.previousData;\n var currentData = _ref2.currentData;\n var previousImageData = _ref2.previousImageData;\n var currentImageData = _ref2.currentImageData;\n\n var adiffResults = void 0;\n if (previousData.width !== currentData.width) {\n // we know that all rows will be different here, so we can take a shortcut\n return [0, // diff starts at index 0\n previousData.height, // number of deletions\n { length: currentData.height }];\n }\n return _adiff2.default.diff(previousImageData.map(function (d) {\n return btoa(d);\n }), currentImageData.map(function (d) {\n return btoa(d);\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n var greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n var previousImageData = imageTo2DArray(previousData, maxWidth - previousData.width);\n\n var currentImageData = imageTo2DArray(currentData, maxWidth - currentData.width);\n\n var adiffResults = getAdiffResults({\n previousData: previousData,\n currentData: currentData,\n previousImageData: previousImageData,\n currentImageData: currentImageData\n });\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 // add a green line to signal an addition\n previousImageData.splice(atIndex + y, 0, greenLine);\n }\n }\n });\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 // add a red line to signal a deletion\n currentImageData.splice(atIndex + y, 0, redLine);\n }\n }\n }\n\n return {\n currentData: {\n data: (0, _flattenImageData2.default)(currentImageData),\n height: currentImageData.length,\n width: maxWidth\n },\n previousData: {\n data: (0, _flattenImageData2.default)(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":["constructColoredLine","rgba","width","line","i","push","imageTo2DArray","paddingRight","data","height","rowSize","getPixelAt","x","y","startIndex","newData","row","pixelsInRow","col","pad","getAdiffResults","previousData","currentData","previousImageData","currentImageData","adiffResults","length","diff","map","btoa","d","computeAndInjectDiffs","maxWidth","Math","max","redLine","greenLine","forEach","instruction","atIndex","deletedItems","addedItems","splice","self","addEventListener","result","postMessage","close"],"mappings":"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","file":"0.js","sourcesContent":["import adiff from 'adiff';\n\nimport flattenImageData from '../flattenImageData';\n\n/**\n * Construct a line of pixels of a certain rgba color\n *\n * @param {Array} rgba\n * @param {Number} width\n * @return {Array}\n */\nfunction constructColoredLine(rgba, width) {\n  const line = [];\n  for (let i = 0; i < width; i++) {\n    line.push(rgba);\n  }\n  return line;\n}\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  const getPixelAt = (x, y) => {\n    const startIndex = (y * rowSize) + (x * 4);\n    return [\n      data[startIndex],\n      data[startIndex + 1],\n      data[startIndex + 2],\n      data[startIndex + 3],\n    ];\n  };\n\n  const newData = [];\n  for (let row = 0; row < height; row++) {\n    const pixelsInRow = [];\n    for (let col = 0; col < width; col++) {\n      pixelsInRow.push(getPixelAt(col, row));\n    }\n    for (let pad = 0; pad < paddingRight; pad++) {\n      pixelsInRow.push([0, 0, 0, 0]);\n    }\n    newData.push(pixelsInRow);\n  }\n  return newData;\n}\n\nfunction getAdiffResults({\n  previousData,\n  currentData,\n  previousImageData,\n  currentImageData,\n}) {\n  let adiffResults;\n  if (previousData.width !== currentData.width) {\n    // we know that all rows will be different here, so we can take a shortcut\n    return [\n      0, // diff starts at index 0\n      previousData.height, // number of deletions\n      { length: currentData.height }, // number of additions\n    ];\n  }\n  return adiff.diff(\n    previousImageData.map(d => btoa(d)),\n    currentImageData.map(d => btoa(d))\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 redLine = constructColoredLine([255, 0, 0, 255], maxWidth);\n  const greenLine = constructColoredLine([0, 255, 0, 255], maxWidth);\n\n  const previousImageData = imageTo2DArray(\n    previousData, maxWidth - previousData.width);\n\n  const currentImageData = imageTo2DArray(\n    currentData, maxWidth - currentData.width);\n\n  const adiffResults = getAdiffResults({\n    previousData,\n    currentData,\n    previousImageData,\n    currentImageData,\n  });\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        // add a green line to signal an addition\n        previousImageData.splice(atIndex + y, 0, greenLine);\n      }\n    }\n  });\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        // add a red line to signal a deletion\n        currentImageData.splice(atIndex + y, 0, redLine);\n      }\n    }\n  }\n\n  return {\n    currentData: {\n      data: flattenImageData(currentImageData),\n      height: currentImageData.length,\n      width: maxWidth,\n    },\n    previousData: {\n      data: flattenImageData(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
+ /* 2 */
57
+ /***/ function(module, exports) {
58
+
59
+ eval("\"use strict\";\n\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = flattenImageData;\n/**\n * @param {Array} imageData a 2d array\n * @return {Uint8ClampedArray}\n */\nfunction flattenImageData(imageData) {\n var width = imageData[0].length;\n var result = new Uint8ClampedArray(imageData.length * width * 4);\n imageData.forEach(function (row, y) {\n row.forEach(function (pixel, x) {\n var index = y * width * 4 + x * 4;\n result[index] = pixel[0];\n result[index + 1] = pixel[1];\n result[index + 2] = pixel[2];\n result[index + 3] = pixel[3];\n });\n });\n return result;\n}//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9qcy9zcmMvZmxhdHRlbkltYWdlRGF0YS5qcz8yZGRiIl0sIm5hbWVzIjpbImZsYXR0ZW5JbWFnZURhdGEiLCJpbWFnZURhdGEiLCJ3aWR0aCIsImxlbmd0aCIsInJlc3VsdCIsIlVpbnQ4Q2xhbXBlZEFycmF5IiwiZm9yRWFjaCIsInJvdyIsInkiLCJwaXhlbCIsIngiLCJpbmRleCJdLCJtYXBwaW5ncyI6Ijs7Ozs7a0JBSXdCQSxnQjtBQUp4Qjs7OztBQUllLFNBQVNBLGdCQUFULENBQTBCQyxTQUExQixFQUFxQztBQUNsRCxNQUFNQyxRQUFRRCxVQUFVLENBQVYsRUFBYUUsTUFBM0I7QUFDQSxNQUFNQyxTQUFTLElBQUlDLGlCQUFKLENBQXNCSixVQUFVRSxNQUFWLEdBQW1CRCxLQUFuQixHQUEyQixDQUFqRCxDQUFmO0FBQ0FELFlBQVVLLE9BQVYsQ0FBa0IsVUFBQ0MsR0FBRCxFQUFNQyxDQUFOLEVBQVk7QUFDNUJELFFBQUlELE9BQUosQ0FBWSxVQUFDRyxLQUFELEVBQVFDLENBQVIsRUFBYztBQUN4QixVQUFNQyxRQUFTSCxJQUFJTixLQUFKLEdBQVksQ0FBYixHQUFtQlEsSUFBSSxDQUFyQztBQUNBTixhQUFPTyxLQUFQLElBQWdCRixNQUFNLENBQU4sQ0FBaEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDQUwsYUFBT08sUUFBUSxDQUFmLElBQW9CRixNQUFNLENBQU4sQ0FBcEI7QUFDRCxLQU5EO0FBT0QsR0FSRDtBQVNBLFNBQU9MLE1BQVA7QUFDRCIsImZpbGUiOiIyLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAcGFyYW0ge0FycmF5fSBpbWFnZURhdGEgYSAyZCBhcnJheVxuICogQHJldHVybiB7VWludDhDbGFtcGVkQXJyYXl9XG4gKi9cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uIGZsYXR0ZW5JbWFnZURhdGEoaW1hZ2VEYXRhKSB7XG4gIGNvbnN0IHdpZHRoID0gaW1hZ2VEYXRhWzBdLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0ID0gbmV3IFVpbnQ4Q2xhbXBlZEFycmF5KGltYWdlRGF0YS5sZW5ndGggKiB3aWR0aCAqIDQpO1xuICBpbWFnZURhdGEuZm9yRWFjaCgocm93LCB5KSA9PiB7XG4gICAgcm93LmZvckVhY2goKHBpeGVsLCB4KSA9PiB7XG4gICAgICBjb25zdCBpbmRleCA9ICh5ICogd2lkdGggKiA0KSArICh4ICogNCk7XG4gICAgICByZXN1bHRbaW5kZXhdID0gcGl4ZWxbMF07XG4gICAgICByZXN1bHRbaW5kZXggKyAxXSA9IHBpeGVsWzFdO1xuICAgICAgcmVzdWx0W2luZGV4ICsgMl0gPSBwaXhlbFsyXTtcbiAgICAgIHJlc3VsdFtpbmRleCArIDNdID0gcGl4ZWxbM107XG4gICAgfSk7XG4gIH0pO1xuICByZXR1cm4gcmVzdWx0O1xufVxuXG5cblxuLyoqIFdFQlBBQ0sgRk9PVEVSICoqXG4gKiogLi9qcy9zcmMvZmxhdHRlbkltYWdlRGF0YS5qc1xuICoqLyJdLCJzb3VyY2VSb290IjoiIn0=");
60
+
61
+ /***/ }
62
+ /******/ ]);