diff --git a/karma.conf.js b/karma.conf.js index 8a4add3..e7678cc 100644 --- a/karma.conf.js +++ b/karma.conf.js @@ -1,6 +1,12 @@ module.exports = function (config) { const configuration = { browsers: ['ChromeHeadless'], + // The tests are sometimes run before the tests were completely written + // to disc; this is a known problem unfortunately. This is a hack to at + // least keep the browsers connected so the tests are compiled and run + // again even if a developer isn't aware of this + autoWatchBatchDelay: 100, + browserNoActivityTimeout: 60 * 1000 * 10, // The directory where the output file lives basePath: 'public/test', // The file itself diff --git a/package-lock.json b/package-lock.json index dddf3de..546d220 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4,6 +4,14 @@ "lockfileVersion": 1, "requires": true, "dependencies": { + "@babel/runtime": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.3.4.tgz", + "integrity": "sha512-IvfvnMdSaLBateu0jfsYIpZTxAc2cKEXEMiezGGN75QcBcecDUKd3PgLAncT0oOgxKy8dd8hrJKj9MfzgfZd6g==", + "requires": { + "regenerator-runtime": "^0.12.0" + } + }, "@hugojosefson/color-hash": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/@hugojosefson/color-hash/-/color-hash-2.0.3.tgz", @@ -12,6 +20,20 @@ "string-hash": "^1.1.3" } }, + "@mrmlnc/readdir-enhanced": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", + "integrity": "sha512-bPHp6Ji8b41szTOcaP63VlnbbO5Ny6dwAATtY6JTjh5N2OLrb5Qk/Th5cRkRQhkWCt+EJsYrNB0MiL+Gpn6e3g==", + "requires": { + "call-me-maybe": "^1.0.1", + "glob-to-regexp": "^0.3.0" + } + }, + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==" + }, "@snyk/dep-graph": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/@snyk/dep-graph/-/dep-graph-1.1.2.tgz", @@ -30,6 +52,31 @@ "integrity": "sha512-mLwF+ccuvRZMS0SxUAxA3dAp8mB3m2FxIsBIUWFTYvzxl+E4XTZb8uFrUqXHbcxhZH1Z8taHohNTbzXZn3M8ag==", "dev": true }, + "@types/events": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.0.tgz", + "integrity": "sha512-EaObqwIvayI5a8dCzhFrjKzVwKLxjoG9T6Ppd5CEo07LRKfQ8Yokw54r5+Wq7FaBQ+yXRvQAYPrHwya1/UFt9g==" + }, + "@types/glob": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.1.tgz", + "integrity": "sha512-1Bh06cbWJUHMC97acuD6UMG29nMt0Aqz1vF3guLfG+kHHJhy3AyohZFFxYk2f7Q1SQIrNwvncxAE0N/9s70F2w==", + "requires": { + "@types/events": "*", + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "@types/minimatch": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz", + "integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==" + }, + "@types/node": { + "version": "11.10.5", + "resolved": "https://registry.npmjs.org/@types/node/-/node-11.10.5.tgz", + "integrity": "sha512-DuIRlQbX4K+d5I+GMnv+UfnGh+ist0RdlvOp+JZ7ePJ6KQONCFQv/gKYSU1ZzbVdFSUCKZOltjmpFAGGv5MdYA==" + }, "@yarnpkg/lockfile": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@yarnpkg/lockfile/-/lockfile-1.1.0.tgz", @@ -103,14 +150,12 @@ "ansi-regex": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=" }, "ansi-styles": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, "requires": { "color-convert": "^1.9.0" } @@ -125,7 +170,6 @@ "version": "2.0.0", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", - "dev": true, "requires": { "micromatch": "^3.1.4", "normalize-path": "^2.1.1" @@ -135,7 +179,6 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", - "dev": true, "requires": { "remove-trailing-separator": "^1.0.1" } @@ -168,7 +211,6 @@ "version": "1.0.10", "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "dev": true, "requires": { "sprintf-js": "~1.0.2" } @@ -176,20 +218,17 @@ "arr-diff": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", - "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", - "dev": true + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=" }, "arr-flatten": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", - "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", - "dev": true + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==" }, "arr-union": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", - "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", - "dev": true + "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=" }, "array-filter": { "version": "0.0.1", @@ -215,17 +254,10 @@ "integrity": "sha1-FziZ0//Rx9k4PkR5Ul2+J4yrXys=", "dev": true }, - "array-slice": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/array-slice/-/array-slice-0.2.3.tgz", - "integrity": "sha1-3Tz7gO15c6dRF82sabC5nshhhvU=", - "dev": true - }, "array-union": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", - "dev": true, "requires": { "array-uniq": "^1.0.1" } @@ -233,14 +265,12 @@ "array-uniq": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", - "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", - "dev": true + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=" }, "array-unique": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", - "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", - "dev": true + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=" }, "arraybuffer.slice": { "version": "0.0.7", @@ -302,8 +332,7 @@ "assign-symbols": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", - "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", - "dev": true + "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=" }, "ast-types": { "version": "0.12.2", @@ -320,8 +349,7 @@ "async-each": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.1.tgz", - "integrity": "sha1-GdOGodntxufByF04iu28xW0zYC0=", - "dev": true + "integrity": "sha1-GdOGodntxufByF04iu28xW0zYC0=" }, "async-foreach": { "version": "0.1.3", @@ -344,8 +372,20 @@ "atob": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", - "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", - "dev": true + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==" + }, + "autoprefixer": { + "version": "9.4.10", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-9.4.10.tgz", + "integrity": "sha512-XR8XZ09tUrrSzgSlys4+hy5r2/z4Jp7Ag3pHm31U4g/CTccYPOVe19AkaJ4ey/vRd1sfj+5TtuD6I0PXtutjvQ==", + "requires": { + "browserslist": "^4.4.2", + "caniuse-lite": "^1.0.30000940", + "normalize-range": "^0.1.2", + "num2fraction": "^1.2.2", + "postcss": "^7.0.14", + "postcss-value-parser": "^3.3.1" + } }, "aws-sign2": { "version": "0.7.0", @@ -368,14 +408,12 @@ "balanced-match": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", - "dev": true + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" }, "base": { "version": "0.11.2", "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", - "dev": true, "requires": { "cache-base": "^1.0.1", "class-utils": "^0.3.5", @@ -390,7 +428,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, "requires": { "is-descriptor": "^1.0.0" } @@ -399,7 +436,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -408,7 +444,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -417,7 +452,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, "requires": { "is-accessor-descriptor": "^1.0.0", "is-data-descriptor": "^1.0.0", @@ -465,8 +499,7 @@ "binary-extensions": { "version": "1.13.0", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.0.tgz", - "integrity": "sha512-EgmjVLMn22z7eGGv3kcnHwSnJXmFHjISTY9E/S5lIcTD3Oxw05QTcBLNkJFzcb3cNueUdF/IN4U+d78V0zO8Hw==", - "dev": true + "integrity": "sha512-EgmjVLMn22z7eGGv3kcnHwSnJXmFHjISTY9E/S5lIcTD3Oxw05QTcBLNkJFzcb3cNueUdF/IN4U+d78V0zO8Hw==" }, "blob": { "version": "0.0.5", @@ -551,7 +584,6 @@ "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, "requires": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" @@ -561,7 +593,6 @@ "version": "2.3.2", "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", - "dev": true, "requires": { "arr-flatten": "^1.1.0", "array-unique": "^0.3.2", @@ -579,7 +610,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } @@ -663,6 +693,16 @@ "pako": "~1.0.5" } }, + "browserslist": { + "version": "4.4.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.4.2.tgz", + "integrity": "sha512-ISS/AIAiHERJ3d45Fz0AVYKkgcy+F/eJHzKEvv1j0wwKGKD9T3BrwKr/5g45L+Y4XIK5PlTqefHciRFcfE1Jxg==", + "requires": { + "caniuse-lite": "^1.0.30000939", + "electron-to-chromium": "^1.3.113", + "node-releases": "^1.1.8" + } + }, "buffer": { "version": "4.9.1", "resolved": "https://registry.npmjs.org/buffer/-/buffer-4.9.1.tgz", @@ -735,7 +775,6 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", - "dev": true, "requires": { "collection-visit": "^1.0.0", "component-emitter": "^1.2.1", @@ -748,6 +787,11 @@ "unset-value": "^1.0.0" } }, + "call-me-maybe": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.1.tgz", + "integrity": "sha1-JtII6onje1y95gJQoV8DHBak1ms=" + }, "callsite": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/callsite/-/callsite-1.0.0.tgz", @@ -770,6 +814,11 @@ "map-obj": "^1.0.0" } }, + "caniuse-lite": { + "version": "1.0.30000942", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30000942.tgz", + "integrity": "sha512-wLf+IhZUy2rfz48tc40OH7jHjXjnvDFEYqBHluINs/6MgzoNLPf25zhE4NOVzqxLKndf+hau81sAW0RcGHIaBQ==" + }, "capture-stack-trace": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", @@ -786,7 +835,6 @@ "version": "2.4.2", "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "dev": true, "requires": { "ansi-styles": "^3.2.1", "escape-string-regexp": "^1.0.5", @@ -803,7 +851,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.0.tgz", "integrity": "sha512-5t6G2SH8eO6lCvYOoUpaRnF5Qfd//gd7qJAkwRUw9qlGVkiQ13uwQngqbWWaurOsaAm9+kUGbITADxt6H0XFNQ==", - "dev": true, "requires": { "anymatch": "^2.0.0", "async-each": "^1.0.1", @@ -822,8 +869,7 @@ "inherits": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" } } }, @@ -843,17 +889,10 @@ "safe-buffer": "^5.0.1" } }, - "circular-json": { - "version": "0.5.9", - "resolved": "https://registry.npmjs.org/circular-json/-/circular-json-0.5.9.tgz", - "integrity": "sha512-4ivwqHpIFJZBuhN3g/pEcdbnGUywkBblloGbkglyloVjjR3uT6tieI89MVOfbP2tHX5sgb01FuLgAOzebNlJNQ==", - "dev": true - }, "class-utils": { "version": "0.3.6", "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", - "dev": true, "requires": { "arr-union": "^3.1.0", "define-property": "^0.2.5", @@ -865,7 +904,6 @@ "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, "requires": { "is-descriptor": "^0.1.0" } @@ -973,14 +1011,12 @@ "code-point-at": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", - "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", - "dev": true + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=" }, "collection-visit": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", - "dev": true, "requires": { "map-visit": "^1.0.0", "object-visit": "^1.0.0" @@ -990,7 +1026,6 @@ "version": "1.9.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.1.tgz", "integrity": "sha512-mjGanIiwQJskCC18rPR6OmrZ6fm2Lc7PeGFYwCmy5J34wC6F1PzdGL6xeMfmgicfYcNLGuVFA3WzXtIDCQSZxQ==", - "dev": true, "requires": { "color-name": "^1.1.1" } @@ -998,8 +1033,7 @@ "color-name": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" }, "colors": { "version": "1.3.3", @@ -1007,15 +1041,6 @@ "integrity": "sha512-mmGt/1pZqYRjMxB1axhTo16/snVZ5krrKkcmMeVKxzECMMXoCgnvTPp10QgHfcbQZw8Dq2jMNG6je4JlWU0gWg==", "dev": true }, - "combine-lists": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/combine-lists/-/combine-lists-1.0.1.tgz", - "integrity": "sha1-RYwH4J4NkA/Ci3Cj/sLazR0st/Y=", - "dev": true, - "requires": { - "lodash": "^4.5.0" - } - }, "combined-stream": { "version": "1.0.7", "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.7.tgz", @@ -1040,8 +1065,7 @@ "component-emitter": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", - "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=", - "dev": true + "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=" }, "component-inherit": { "version": "0.0.3", @@ -1052,8 +1076,7 @@ "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "dev": true + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" }, "configstore": { "version": "3.1.2", @@ -1117,20 +1140,40 @@ "copy-descriptor": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", - "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", - "dev": true + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=" }, "core-js": { - "version": "2.6.4", - "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.4.tgz", - "integrity": "sha512-05qQ5hXShcqGkPZpXEFLIpxayZscVD2kuMBZewxiIPPEagukO4mqgPA9CWhUvFBJfy3ODdK2p9xyHh7FTU9/7A==", + "version": "2.6.5", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.5.tgz", + "integrity": "sha512-klh/kDpwX8hryYL14M9w/xei6vrv6sE8gTHDG7/T/+SEovB/G4ejwcfE/CBzO6Edsu+OETZMZ3wcX/EjUkrl5A==", "dev": true }, "core-util-is": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", - "dev": true + "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" + }, + "cosmiconfig": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-4.0.0.tgz", + "integrity": "sha512-6e5vDdrXZD+t5v0L8CrurPeybg4Fmf+FCSYxXKYVAqLUtyCSbuyqE059d0kDthTNRzKVjL7QMgNpEUlsoYH3iQ==", + "requires": { + "is-directory": "^0.3.1", + "js-yaml": "^3.9.0", + "parse-json": "^4.0.0", + "require-from-string": "^2.0.1" + }, + "dependencies": { + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + } + } }, "create-ecdh": { "version": "4.0.3", @@ -1255,9 +1298,9 @@ "dev": true }, "date-format": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/date-format/-/date-format-1.2.0.tgz", - "integrity": "sha1-YV6CjiM90aubua4JUODOzPpuytg=", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/date-format/-/date-format-2.0.0.tgz", + "integrity": "sha512-M6UqVvZVgFYqZL1SfHsRGIQSz3ZL+qgbsV5Lp1Vj61LZVYuEwcMXYay7DRDtYs2HQQBK5hQtQ0fD9aEJ89V0LA==", "dev": true }, "date-now": { @@ -1270,7 +1313,6 @@ "version": "2.6.9", "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, "requires": { "ms": "2.0.0" } @@ -1278,14 +1320,12 @@ "decamelize": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", - "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", - "dev": true + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" }, "decode-uri-component": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", - "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", - "dev": true + "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=" }, "deep-extend": { "version": "0.6.0", @@ -1312,7 +1352,6 @@ "version": "2.0.2", "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", - "dev": true, "requires": { "is-descriptor": "^1.0.2", "isobject": "^3.0.1" @@ -1322,7 +1361,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -1331,7 +1369,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -1340,7 +1377,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, "requires": { "is-accessor-descriptor": "^1.0.0", "is-data-descriptor": "^1.0.0", @@ -1378,6 +1414,11 @@ "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", "dev": true }, + "dependency-graph": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/dependency-graph/-/dependency-graph-0.8.0.tgz", + "integrity": "sha512-DCvzSq2UiMsuLnj/9AL484ummEgLtZIcRS7YvtO38QnpX3vqh9nJ8P+zhu8Ja+SmLrBHO2iDbva20jq38qvBkQ==" + }, "des.js": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/des.js/-/des.js-1.0.0.tgz", @@ -1405,6 +1446,29 @@ "randombytes": "^2.0.0" } }, + "dir-glob": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.2.2.tgz", + "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", + "requires": { + "path-type": "^3.0.0" + }, + "dependencies": { + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "requires": { + "pify": "^3.0.0" + } + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=" + } + } + }, "dockerfile-ast": { "version": "0.0.12", "resolved": "https://registry.npmjs.org/dockerfile-ast/-/dockerfile-ast-0.0.12.tgz", @@ -1463,6 +1527,11 @@ "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", "dev": true }, + "electron-to-chromium": { + "version": "1.3.113", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.113.tgz", + "integrity": "sha512-De+lPAxEcpxvqPTyZAXELNpRZXABRxf+uL/rSykstQhzj/B0l1150G/ExIIxKc16lI89Hgz81J0BHAcbTqK49g==" + }, "elliptic": { "version": "6.4.1", "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.4.1.tgz", @@ -1498,6 +1567,14 @@ "iconv-lite": "~0.4.13" } }, + "end-of-stream": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.1.tgz", + "integrity": "sha512-1MkrZNvWTKCaigbn+W15elq2BB/L22nqrSY5DKlo3X6+vclJm8Bb5djXJBmEX6fS3+zCh/F4VBK5Z2KxJt4s2Q==", + "requires": { + "once": "^1.4.0" + } + }, "engine.io": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-3.2.1.tgz", @@ -1525,7 +1602,7 @@ }, "engine.io-client": { "version": "3.2.1", - "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-3.2.1.tgz", + "resolved": "http://registry.npmjs.org/engine.io-client/-/engine.io-client-3.2.1.tgz", "integrity": "sha512-y5AbkytWeM4jQr7m/koQLc5AxpRKC1hEVUb/s1FUAWEJq5AzJJ4NLvzuKPuxtDi5Mq755WuDvZ6Iv2rXj4PTzw==", "dev": true, "requires": { @@ -1576,7 +1653,6 @@ "version": "1.3.2", "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", - "dev": true, "requires": { "is-arrayish": "^0.2.1" } @@ -1629,8 +1705,7 @@ "escape-string-regexp": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", - "dev": true + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" }, "escodegen": { "version": "1.11.0", @@ -1700,39 +1775,10 @@ "strip-eof": "^1.0.0" } }, - "expand-braces": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/expand-braces/-/expand-braces-0.1.2.tgz", - "integrity": "sha1-SIsdHSRRyz06axks/AMPRMWFX+o=", - "dev": true, - "requires": { - "array-slice": "^0.2.3", - "array-unique": "^0.2.1", - "braces": "^0.1.2" - }, - "dependencies": { - "array-unique": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.2.1.tgz", - "integrity": "sha1-odl8yvy8JiXMcPrc6zalDFiwGlM=", - "dev": true - }, - "braces": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/braces/-/braces-0.1.5.tgz", - "integrity": "sha1-wIVxEIUpHYt1/ddOqw+FlygHEeY=", - "dev": true, - "requires": { - "expand-range": "^0.1.0" - } - } - } - }, "expand-brackets": { "version": "2.1.4", "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", - "dev": true, "requires": { "debug": "^2.3.3", "define-property": "^0.2.5", @@ -1747,7 +1793,6 @@ "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, "requires": { "is-descriptor": "^0.1.0" } @@ -1756,37 +1801,12 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } } } }, - "expand-range": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/expand-range/-/expand-range-0.1.1.tgz", - "integrity": "sha1-TLjtoJk8pW+k9B/ELzy7TMrf8EQ=", - "dev": true, - "requires": { - "is-number": "^0.1.1", - "repeat-string": "^0.2.2" - }, - "dependencies": { - "is-number": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-0.1.1.tgz", - "integrity": "sha1-aaevEWlj1HIG7JvZtIoUIW8eOAY=", - "dev": true - }, - "repeat-string": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-0.2.2.tgz", - "integrity": "sha1-x6jTI2BoNiBZp+RlH8aITosftK4=", - "dev": true - } - } - }, "extend": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", @@ -1797,7 +1817,6 @@ "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", - "dev": true, "requires": { "assign-symbols": "^1.0.0", "is-extendable": "^1.0.1" @@ -1807,7 +1826,6 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, "requires": { "is-plain-object": "^2.0.4" } @@ -1829,7 +1847,6 @@ "version": "2.0.4", "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", - "dev": true, "requires": { "array-unique": "^0.3.2", "define-property": "^1.0.0", @@ -1845,7 +1862,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, "requires": { "is-descriptor": "^1.0.0" } @@ -1854,7 +1870,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } @@ -1863,7 +1878,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -1872,7 +1886,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -1881,7 +1894,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, "requires": { "is-accessor-descriptor": "^1.0.0", "is-data-descriptor": "^1.0.0", @@ -1902,6 +1914,19 @@ "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", "dev": true }, + "fast-glob": { + "version": "2.2.6", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.6.tgz", + "integrity": "sha512-0BvMaZc1k9F+MeWWMe8pL6YltFzZYcJsYU7D4JyDA6PAczaXvxqQQ/z+mDF7/4Mw01DeUc+i3CTKajnkANkV4w==", + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, "fast-json-stable-stringify": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", @@ -1981,7 +2006,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", - "dev": true, "requires": { "extend-shallow": "^2.0.1", "is-number": "^3.0.0", @@ -1993,7 +2017,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } @@ -2040,30 +2063,35 @@ "dev": true }, "follow-redirects": { - "version": "1.6.1", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.6.1.tgz", - "integrity": "sha512-t2JCjbzxQpWvbhts3l6SH1DKzSrx8a+SsaVf4h6bG4kOXUuPYS/kg2Lr4gQSb7eemaHqJkOThF1BGyjlUkO1GQ==", + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.7.0.tgz", + "integrity": "sha512-m/pZQy4Gj287eNy94nivy5wchN3Kp+Q5WgUPNy5lJSZ3sgkVKSYV/ZChMAQVIgx1SqfZ2zBZtPA2YlXIWxxJOQ==", "dev": true, "requires": { - "debug": "=3.1.0" + "debug": "^3.2.6" }, "dependencies": { "debug": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", - "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", "dev": true, "requires": { - "ms": "2.0.0" + "ms": "^2.1.1" } + }, + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true } } }, "for-in": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", - "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", - "dev": true + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=" }, "for-own": { "version": "1.0.0", @@ -2095,7 +2123,6 @@ "version": "0.2.1", "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", - "dev": true, "requires": { "map-cache": "^0.2.2" } @@ -2123,14 +2150,12 @@ "fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", - "dev": true + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" }, "fsevents": { "version": "1.2.7", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.7.tgz", "integrity": "sha512-Pxm6sI2MeBD7RdD12RYsqaP0nMiwx8eZBXCa6z2L+mRHm2DYrOYwihmhjpkdjUHwQhslWQjRpEgNq4XvBmaAuw==", - "dev": true, "optional": true, "requires": { "nan": "^2.9.2", @@ -2140,25 +2165,21 @@ "abbrev": { "version": "1.1.1", "bundled": true, - "dev": true, "optional": true }, "ansi-regex": { "version": "2.1.1", "bundled": true, - "dev": true, "optional": true }, "aproba": { "version": "1.2.0", "bundled": true, - "dev": true, "optional": true }, "are-we-there-yet": { "version": "1.1.5", "bundled": true, - "dev": true, "optional": true, "requires": { "delegates": "^1.0.0", @@ -2168,13 +2189,11 @@ "balanced-match": { "version": "1.0.0", "bundled": true, - "dev": true, "optional": true }, "brace-expansion": { "version": "1.1.11", "bundled": true, - "dev": true, "optional": true, "requires": { "balanced-match": "^1.0.0", @@ -2184,37 +2203,31 @@ "chownr": { "version": "1.1.1", "bundled": true, - "dev": true, "optional": true }, "code-point-at": { "version": "1.1.0", "bundled": true, - "dev": true, "optional": true }, "concat-map": { "version": "0.0.1", "bundled": true, - "dev": true, "optional": true }, "console-control-strings": { "version": "1.1.0", "bundled": true, - "dev": true, "optional": true }, "core-util-is": { "version": "1.0.2", "bundled": true, - "dev": true, "optional": true }, "debug": { "version": "2.6.9", "bundled": true, - "dev": true, "optional": true, "requires": { "ms": "2.0.0" @@ -2223,25 +2236,21 @@ "deep-extend": { "version": "0.6.0", "bundled": true, - "dev": true, "optional": true }, "delegates": { "version": "1.0.0", "bundled": true, - "dev": true, "optional": true }, "detect-libc": { "version": "1.0.3", "bundled": true, - "dev": true, "optional": true }, "fs-minipass": { "version": "1.2.5", "bundled": true, - "dev": true, "optional": true, "requires": { "minipass": "^2.2.1" @@ -2250,13 +2259,11 @@ "fs.realpath": { "version": "1.0.0", "bundled": true, - "dev": true, "optional": true }, "gauge": { "version": "2.7.4", "bundled": true, - "dev": true, "optional": true, "requires": { "aproba": "^1.0.3", @@ -2272,7 +2279,6 @@ "glob": { "version": "7.1.3", "bundled": true, - "dev": true, "optional": true, "requires": { "fs.realpath": "^1.0.0", @@ -2286,13 +2292,11 @@ "has-unicode": { "version": "2.0.1", "bundled": true, - "dev": true, "optional": true }, "iconv-lite": { "version": "0.4.24", "bundled": true, - "dev": true, "optional": true, "requires": { "safer-buffer": ">= 2.1.2 < 3" @@ -2301,7 +2305,6 @@ "ignore-walk": { "version": "3.0.1", "bundled": true, - "dev": true, "optional": true, "requires": { "minimatch": "^3.0.4" @@ -2310,7 +2313,6 @@ "inflight": { "version": "1.0.6", "bundled": true, - "dev": true, "optional": true, "requires": { "once": "^1.3.0", @@ -2320,19 +2322,16 @@ "inherits": { "version": "2.0.3", "bundled": true, - "dev": true, "optional": true }, "ini": { "version": "1.3.5", "bundled": true, - "dev": true, "optional": true }, "is-fullwidth-code-point": { "version": "1.0.0", "bundled": true, - "dev": true, "optional": true, "requires": { "number-is-nan": "^1.0.0" @@ -2341,13 +2340,11 @@ "isarray": { "version": "1.0.0", "bundled": true, - "dev": true, "optional": true }, "minimatch": { "version": "3.0.4", "bundled": true, - "dev": true, "optional": true, "requires": { "brace-expansion": "^1.1.7" @@ -2356,13 +2353,11 @@ "minimist": { "version": "0.0.8", "bundled": true, - "dev": true, "optional": true }, "minipass": { "version": "2.3.5", "bundled": true, - "dev": true, "optional": true, "requires": { "safe-buffer": "^5.1.2", @@ -2372,7 +2367,6 @@ "minizlib": { "version": "1.2.1", "bundled": true, - "dev": true, "optional": true, "requires": { "minipass": "^2.2.1" @@ -2381,7 +2375,6 @@ "mkdirp": { "version": "0.5.1", "bundled": true, - "dev": true, "optional": true, "requires": { "minimist": "0.0.8" @@ -2390,13 +2383,11 @@ "ms": { "version": "2.0.0", "bundled": true, - "dev": true, "optional": true }, "needle": { "version": "2.2.4", "bundled": true, - "dev": true, "optional": true, "requires": { "debug": "^2.1.2", @@ -2407,7 +2398,6 @@ "node-pre-gyp": { "version": "0.10.3", "bundled": true, - "dev": true, "optional": true, "requires": { "detect-libc": "^1.0.2", @@ -2425,7 +2415,6 @@ "nopt": { "version": "4.0.1", "bundled": true, - "dev": true, "optional": true, "requires": { "abbrev": "1", @@ -2435,13 +2424,11 @@ "npm-bundled": { "version": "1.0.5", "bundled": true, - "dev": true, "optional": true }, "npm-packlist": { "version": "1.2.0", "bundled": true, - "dev": true, "optional": true, "requires": { "ignore-walk": "^3.0.1", @@ -2451,7 +2438,6 @@ "npmlog": { "version": "4.1.2", "bundled": true, - "dev": true, "optional": true, "requires": { "are-we-there-yet": "~1.1.2", @@ -2463,19 +2449,16 @@ "number-is-nan": { "version": "1.0.1", "bundled": true, - "dev": true, "optional": true }, "object-assign": { "version": "4.1.1", "bundled": true, - "dev": true, "optional": true }, "once": { "version": "1.4.0", "bundled": true, - "dev": true, "optional": true, "requires": { "wrappy": "1" @@ -2484,19 +2467,16 @@ "os-homedir": { "version": "1.0.2", "bundled": true, - "dev": true, "optional": true }, "os-tmpdir": { "version": "1.0.2", "bundled": true, - "dev": true, "optional": true }, "osenv": { "version": "0.1.5", "bundled": true, - "dev": true, "optional": true, "requires": { "os-homedir": "^1.0.0", @@ -2506,19 +2486,16 @@ "path-is-absolute": { "version": "1.0.1", "bundled": true, - "dev": true, "optional": true }, "process-nextick-args": { "version": "2.0.0", "bundled": true, - "dev": true, "optional": true }, "rc": { "version": "1.2.8", "bundled": true, - "dev": true, "optional": true, "requires": { "deep-extend": "^0.6.0", @@ -2530,7 +2507,6 @@ "minimist": { "version": "1.2.0", "bundled": true, - "dev": true, "optional": true } } @@ -2538,7 +2514,6 @@ "readable-stream": { "version": "2.3.6", "bundled": true, - "dev": true, "optional": true, "requires": { "core-util-is": "~1.0.0", @@ -2553,7 +2528,6 @@ "rimraf": { "version": "2.6.3", "bundled": true, - "dev": true, "optional": true, "requires": { "glob": "^7.1.3" @@ -2562,43 +2536,36 @@ "safe-buffer": { "version": "5.1.2", "bundled": true, - "dev": true, "optional": true }, "safer-buffer": { "version": "2.1.2", "bundled": true, - "dev": true, "optional": true }, "sax": { "version": "1.2.4", "bundled": true, - "dev": true, "optional": true }, "semver": { "version": "5.6.0", "bundled": true, - "dev": true, "optional": true }, "set-blocking": { "version": "2.0.0", "bundled": true, - "dev": true, "optional": true }, "signal-exit": { "version": "3.0.2", "bundled": true, - "dev": true, "optional": true }, "string-width": { "version": "1.0.2", "bundled": true, - "dev": true, "optional": true, "requires": { "code-point-at": "^1.0.0", @@ -2609,7 +2576,6 @@ "string_decoder": { "version": "1.1.1", "bundled": true, - "dev": true, "optional": true, "requires": { "safe-buffer": "~5.1.0" @@ -2618,7 +2584,6 @@ "strip-ansi": { "version": "3.0.1", "bundled": true, - "dev": true, "optional": true, "requires": { "ansi-regex": "^2.0.0" @@ -2627,13 +2592,11 @@ "strip-json-comments": { "version": "2.0.1", "bundled": true, - "dev": true, "optional": true }, "tar": { "version": "4.4.8", "bundled": true, - "dev": true, "optional": true, "requires": { "chownr": "^1.1.1", @@ -2648,13 +2611,11 @@ "util-deprecate": { "version": "1.0.2", "bundled": true, - "dev": true, "optional": true }, "wide-align": { "version": "1.1.3", "bundled": true, - "dev": true, "optional": true, "requires": { "string-width": "^1.0.2 || 2" @@ -2663,13 +2624,11 @@ "wrappy": { "version": "1.0.2", "bundled": true, - "dev": true, "optional": true }, "yallist": { "version": "3.0.3", "bundled": true, - "dev": true, "optional": true } } @@ -2793,8 +2752,7 @@ "get-caller-file": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-1.0.3.tgz", - "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==", - "dev": true + "integrity": "sha512-3t6rVToeoZfYSGd8YoLFR2DJkiQrIiUrGcjvFX2mDw3bn6k2OtwHN0TNCLbBO+w8qTvimhDkv+LSscbJY1vE6w==" }, "get-stdin": { "version": "4.0.1", @@ -2825,8 +2783,7 @@ "get-value": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", - "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", - "dev": true + "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=" }, "getpass": { "version": "0.1.7", @@ -2862,9 +2819,9 @@ } }, "lodash": { - "version": "4.17.10", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.10.tgz", - "integrity": "sha512-UejweD1pDoXu+AD825lWwp4ZGtSwgnpZxb3JDViD7StjQz+Nb/6l093lx4OQ0foGWNRoc19mWy7BzL+UAK2iVg==", + "version": "4.17.11", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", + "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==", "dev": true } } @@ -2887,7 +2844,6 @@ "version": "3.1.0", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", - "dev": true, "requires": { "is-glob": "^3.1.0", "path-dirname": "^1.0.0" @@ -2897,13 +2853,17 @@ "version": "3.1.0", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", - "dev": true, "requires": { "is-extglob": "^2.1.0" } } } }, + "glob-to-regexp": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", + "integrity": "sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=" + }, "global-dirs": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-0.1.1.tgz", @@ -2959,8 +2919,7 @@ "graceful-fs": { "version": "4.1.11", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", - "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", - "dev": true + "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=" }, "graphlib": { "version": "2.1.7", @@ -3045,8 +3004,7 @@ "has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" }, "has-symbols": { "version": "1.0.0", @@ -3064,7 +3022,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", - "dev": true, "requires": { "get-value": "^2.0.6", "has-values": "^1.0.0", @@ -3075,7 +3032,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", - "dev": true, "requires": { "is-number": "^3.0.0", "kind-of": "^4.0.0" @@ -3085,7 +3041,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -3154,7 +3109,7 @@ }, "http-errors": { "version": "1.6.3", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "resolved": "http://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", "dev": true, "requires": { @@ -3272,12 +3227,33 @@ "integrity": "sha512-GguP+DRY+pJ3soyIiGPTvdiVXjZ+DbXOxGpXn3eMvNW4x4irjqXm4wHKscC+TfxSJ0yw/S1F24tqdMNsMZTiLA==", "dev": true }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==" + }, "immediate": { "version": "3.0.6", "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=", "dev": true }, + "import-cwd": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-cwd/-/import-cwd-2.1.0.tgz", + "integrity": "sha1-qmzzbnInYShcs3HsZRn1PiQ1sKk=", + "requires": { + "import-from": "^2.1.0" + } + }, + "import-from": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/import-from/-/import-from-2.1.0.tgz", + "integrity": "sha1-M1238qev/VOqpHHUuAId7ja387E=", + "requires": { + "resolve-from": "^3.0.0" + } + }, "import-lazy": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", @@ -3315,7 +3291,6 @@ "version": "1.0.6", "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "dev": true, "requires": { "once": "^1.3.0", "wrappy": "1" @@ -3324,8 +3299,7 @@ "inherits": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz", - "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=", - "dev": true + "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=" }, "ini": { "version": "1.3.5", @@ -3355,6 +3329,14 @@ "through": "^2.3.6" } }, + "invariant": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", + "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "requires": { + "loose-envify": "^1.0.0" + } + }, "invert-kv": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-1.0.0.tgz", @@ -3371,7 +3353,6 @@ "version": "0.1.6", "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", - "dev": true, "requires": { "kind-of": "^3.0.2" }, @@ -3380,7 +3361,6 @@ "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -3390,14 +3370,12 @@ "is-arrayish": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", - "dev": true + "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=" }, "is-binary-path": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", - "dev": true, "requires": { "binary-extensions": "^1.0.0" } @@ -3405,8 +3383,7 @@ "is-buffer": { "version": "1.1.6", "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", - "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", - "dev": true + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" }, "is-builtin-module": { "version": "1.0.0", @@ -3436,7 +3413,6 @@ "version": "0.1.4", "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", - "dev": true, "requires": { "kind-of": "^3.0.2" }, @@ -3445,7 +3421,6 @@ "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -3462,7 +3437,6 @@ "version": "0.1.6", "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", - "dev": true, "requires": { "is-accessor-descriptor": "^0.1.6", "is-data-descriptor": "^0.1.4", @@ -3472,22 +3446,24 @@ "kind-of": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", - "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", - "dev": true + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==" } } }, + "is-directory": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/is-directory/-/is-directory-0.3.1.tgz", + "integrity": "sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE=" + }, "is-extendable": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", - "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", - "dev": true + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=" }, "is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "dev": true + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=" }, "is-finite": { "version": "1.0.2", @@ -3501,14 +3477,12 @@ "is-fullwidth-code-point": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=" }, "is-glob": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.0.tgz", "integrity": "sha1-lSHHaEXMJhCoUgPd8ICpWML/q8A=", - "dev": true, "requires": { "is-extglob": "^2.1.1" } @@ -3533,7 +3507,6 @@ "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, "requires": { "kind-of": "^3.0.2" }, @@ -3542,7 +3515,6 @@ "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -3574,7 +3546,6 @@ "version": "2.0.4", "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", - "dev": true, "requires": { "isobject": "^3.0.1" } @@ -3635,8 +3606,7 @@ "is-windows": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", - "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", - "dev": true + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==" }, "is-wsl": { "version": "1.1.0", @@ -3647,8 +3617,7 @@ "isarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", - "dev": true + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" }, "isbinaryfile": { "version": "3.0.3", @@ -3662,14 +3631,12 @@ "isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "dev": true + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=" }, "isobject": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", - "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", - "dev": true + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=" }, "isomorphic-fetch": { "version": "2.2.1", @@ -3701,7 +3668,6 @@ "version": "3.12.1", "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.12.1.tgz", "integrity": "sha512-um46hB9wNOKlwkHgiuyEVAybXBjwFUV0Z/RaHJblRd9DXltue9FTYvzCr9ErQrK9Adz5MU4gHWVaNUfdmrC8qA==", - "dev": true, "requires": { "argparse": "^1.0.7", "esprima": "^4.0.0" @@ -3710,8 +3676,7 @@ "esprima": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==" } } }, @@ -3724,8 +3689,7 @@ "json-parse-better-errors": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", - "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", - "dev": true + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==" }, "json-schema": { "version": "0.2.3", @@ -3749,7 +3713,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", - "dev": true, "requires": { "graceful-fs": "^4.1.6" } @@ -3826,28 +3789,27 @@ } }, "karma": { - "version": "3.1.4", - "resolved": "https://registry.npmjs.org/karma/-/karma-3.1.4.tgz", - "integrity": "sha512-31Vo8Qr5glN+dZEVIpnPCxEGleqE0EY6CtC2X9TagRV3rRQ3SNrvfhddICkJgUK3AgqpeKSZau03QumTGhGoSw==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/karma/-/karma-4.0.1.tgz", + "integrity": "sha512-ind+4s03BqIXas7ZmraV3/kc5+mnqwCd+VDX1FndS6jxbt03kQKX2vXrWxNLuCjVYmhMwOZosAEKMM0a2q7w7A==", "dev": true, "requires": { "bluebird": "^3.3.0", "body-parser": "^1.16.1", + "braces": "^2.3.2", "chokidar": "^2.0.3", "colors": "^1.1.0", - "combine-lists": "^1.0.0", "connect": "^3.6.0", "core-js": "^2.2.0", "di": "^0.0.1", "dom-serialize": "^2.2.0", - "expand-braces": "^0.1.1", "flatted": "^2.0.0", "glob": "^7.1.1", "graceful-fs": "^4.1.2", "http-proxy": "^1.13.0", "isbinaryfile": "^3.0.0", - "lodash": "^4.17.5", - "log4js": "^3.0.0", + "lodash": "^4.17.11", + "log4js": "^4.0.0", "mime": "^2.3.1", "minimatch": "^3.0.2", "optimist": "^0.6.1", @@ -3890,8 +3852,7 @@ "kind-of": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.2.tgz", - "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==", - "dev": true + "integrity": "sha512-s5kLOcnH0XqDO+FvuaLX8DDjZ18CGFk7VygH40QoKPUQhW4e2rvM0rwUq0t8IQDOwYSeLK01U90OjzBTme2QqA==" }, "latest-version": { "version": "3.1.0", @@ -3949,11 +3910,26 @@ "strip-bom": "^2.0.0" } }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + }, + "dependencies": { + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=" + } + } + }, "lodash": { "version": "4.17.11", "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", - "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==", - "dev": true + "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==" }, "lodash.assign": { "version": "4.2.0", @@ -4003,17 +3979,25 @@ "integrity": "sha1-2HV7HagH3eJIFrDWqEvqGnYjCyM=", "dev": true }, + "log-symbols": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", + "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", + "requires": { + "chalk": "^2.0.1" + } + }, "log4js": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/log4js/-/log4js-3.0.6.tgz", - "integrity": "sha512-ezXZk6oPJCWL483zj64pNkMuY/NcRX5MPiB0zE6tjZM137aeusrOnW1ecxgF9cmwMWkBMhjteQxBPoZBh9FDxQ==", + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/log4js/-/log4js-4.0.2.tgz", + "integrity": "sha512-KE7HjiieVDPPdveA3bJZSuu0n8chMkFl8mIoisBFxwEJ9FmXe4YzNuiqSwYUiR1K8q8/5/8Yd6AClENY1RA9ww==", "dev": true, "requires": { - "circular-json": "^0.5.5", - "date-format": "^1.2.0", + "date-format": "^2.0.0", "debug": "^3.1.0", + "flatted": "^2.0.0", "rfdc": "^1.1.2", - "streamroller": "0.7.0" + "streamroller": "^1.0.1" }, "dependencies": { "debug": { @@ -4090,11 +4074,18 @@ } } }, + "map-age-cleaner": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz", + "integrity": "sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w==", + "requires": { + "p-defer": "^1.0.0" + } + }, "map-cache": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", - "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", - "dev": true + "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=" }, "map-obj": { "version": "1.0.1", @@ -4106,7 +4097,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", - "dev": true, "requires": { "object-visit": "^1.0.0" } @@ -4124,10 +4114,20 @@ }, "media-typer": { "version": "0.3.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "resolved": "http://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=", "dev": true }, + "mem": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/mem/-/mem-4.1.0.tgz", + "integrity": "sha512-I5u6Q1x7wxO0kdOpYBB28xueHADYps5uty/zg936CiG8NTe5sJL8EjrCuLneuDW3PlMdZBGDIn8BirEVdovZvg==", + "requires": { + "map-age-cleaner": "^0.1.1", + "mimic-fn": "^1.0.0", + "p-is-promise": "^2.0.0" + } + }, "memorystream": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/memorystream/-/memorystream-0.3.1.tgz", @@ -4160,11 +4160,15 @@ } } }, + "merge2": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.2.3.tgz", + "integrity": "sha512-gdUU1Fwj5ep4kplwcmftruWofEFt6lfpkkr3h860CXbAB9c3hGb55EOL2ali0Td5oebvW0E1+3Sr+Ur7XfKpRA==" + }, "micromatch": { "version": "3.1.10", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", - "dev": true, "requires": { "arr-diff": "^4.0.0", "array-unique": "^0.3.2", @@ -4215,8 +4219,7 @@ "mimic-fn": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", - "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", - "dev": true + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==" }, "minimalistic-assert": { "version": "1.0.1", @@ -4234,7 +4237,6 @@ "version": "3.0.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "dev": true, "requires": { "brace-expansion": "^1.1.7" } @@ -4249,7 +4251,6 @@ "version": "1.3.1", "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.1.tgz", "integrity": "sha512-8ZItLHeEgaqEvd5lYBXfm4EZSFCX29Jb9K+lAHhDKzReKBQKj3R+7NOF6tjqYi9t4oI8VUfaWITJQm86wnXGNQ==", - "dev": true, "requires": { "for-in": "^1.0.2", "is-extendable": "^1.0.1" @@ -4259,7 +4260,6 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, "requires": { "is-plain-object": "^2.0.4" } @@ -4296,8 +4296,7 @@ "ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" }, "mute-stream": { "version": "0.0.7", @@ -4308,14 +4307,12 @@ "nan": { "version": "2.12.1", "resolved": "https://registry.npmjs.org/nan/-/nan-2.12.1.tgz", - "integrity": "sha512-JY7V6lRkStKcKTvHO5NVSQRv+RV+FIL5pvDoLiAtSL9pKlC5x9PKQcZDsq7m4FO4d57mkhC6Z+QhAh3Jdk5JFw==", - "dev": true + "integrity": "sha512-JY7V6lRkStKcKTvHO5NVSQRv+RV+FIL5pvDoLiAtSL9pKlC5x9PKQcZDsq7m4FO4d57mkhC6Z+QhAh3Jdk5JFw==" }, "nanomatch": { "version": "1.2.13", "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", - "dev": true, "requires": { "arr-diff": "^4.0.0", "array-unique": "^0.3.2", @@ -4368,8 +4365,7 @@ "nice-try": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", - "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", - "dev": true + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==" }, "node-fetch": { "version": "1.7.3", @@ -4460,6 +4456,14 @@ "which": "^1.3.0" } }, + "node-releases": { + "version": "1.1.9", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.9.tgz", + "integrity": "sha512-oic3GT4OtbWWKfRolz5Syw0Xus0KRFxeorLNj0s93ofX6PWyuzKjsiGxsCtWktBwwmTF6DdRRf2KreGqeOk5KA==", + "requires": { + "semver": "^5.3.0" + } + }, "node-sass": { "version": "4.11.0", "resolved": "https://registry.npmjs.org/node-sass/-/node-sass-4.11.0.tgz", @@ -4563,8 +4567,12 @@ "normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" + }, + "normalize-range": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", + "integrity": "sha1-LRDAa9/TEuqXd2laTShDlFa3WUI=" }, "normalize-url": { "version": "1.9.1", @@ -4697,7 +4705,6 @@ "version": "2.0.2", "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", - "dev": true, "requires": { "path-key": "^2.0.0" } @@ -4720,11 +4727,15 @@ "integrity": "sha1-l33/1xdgErnsMNKjnbXPcqBDnt0=", "dev": true }, + "num2fraction": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/num2fraction/-/num2fraction-1.2.2.tgz", + "integrity": "sha1-b2gragJ6Tp3fpFZM0lidHU5mnt4=" + }, "number-is-nan": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", - "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", - "dev": true + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" }, "oauth-sign": { "version": "0.9.0", @@ -4747,7 +4758,6 @@ "version": "0.1.0", "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", - "dev": true, "requires": { "copy-descriptor": "^0.1.0", "define-property": "^0.2.5", @@ -4758,7 +4768,6 @@ "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, "requires": { "is-descriptor": "^0.1.0" } @@ -4767,7 +4776,6 @@ "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -4784,7 +4792,6 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", - "dev": true, "requires": { "isobject": "^3.0.0" } @@ -4793,7 +4800,6 @@ "version": "1.3.0", "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", - "dev": true, "requires": { "isobject": "^3.0.1" } @@ -4811,7 +4817,6 @@ "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "dev": true, "requires": { "wrappy": "1" } @@ -4918,11 +4923,41 @@ "os-tmpdir": "^1.0.0" } }, + "p-defer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-defer/-/p-defer-1.0.0.tgz", + "integrity": "sha1-n26xgvbJqozXQwBKfU+WsZaw+ww=" + }, "p-finally": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", - "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", - "dev": true + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=" + }, + "p-is-promise": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-2.0.0.tgz", + "integrity": "sha512-pzQPhYMCAgLAKPWD2jC3Se9fEfrD9npNos0y150EeqZll7akhEgGhTW/slB6lHku8AvYGiJ+YJ5hfHKePPgFWg==" + }, + "p-limit": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.2.0.tgz", + "integrity": "sha512-pZbTJpoUsCzV48Mc9Nh51VbwO0X9cuPFE8gYwx9BTCt9SF8/b7Zljd2fVgOxhIF/HDTKgpVzs+GPhyKfjLLFRQ==", + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "requires": { + "p-limit": "^2.0.0" + } + }, + "p-try": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.0.0.tgz", + "integrity": "sha512-hMp0onDKIajHfIkdRk3P4CdCmErkYAxxDtP3Wx/4nZ3aGlau2VKh3mZpcuFkH27WQkL/3WBCPOktzA9ZOAnMQQ==" }, "pac-proxy-agent": { "version": "2.0.2", @@ -4989,9 +5024,9 @@ "dev": true }, "parse-asn1": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.3.tgz", - "integrity": "sha512-VrPoetlz7B/FqjBLD2f5wBVZvsZVLnRUrxVLfRYhGXCODa/NWE4p3Wp+6+aV3ZPL3KM7/OZmxDIwwijD7yuucg==", + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.4.tgz", + "integrity": "sha512-Qs5duJcuvNExRfFZ99HDD3z4mAi3r9Wl/FOjEOijlxwCZs7E7mW2vjTpgQ4J8LpTF8x5v+1Vn5UQFejmWT11aw==", "dev": true, "requires": { "asn1.js": "^4.0.0", @@ -5038,8 +5073,7 @@ "pascalcase": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", - "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", - "dev": true + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=" }, "path-browserify": { "version": "0.0.0", @@ -5050,8 +5084,7 @@ "path-dirname": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", - "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", - "dev": true + "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=" }, "path-exists": { "version": "2.1.0", @@ -5065,8 +5098,7 @@ "path-is-absolute": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", - "dev": true + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" }, "path-is-inside": { "version": "1.0.2", @@ -5077,8 +5109,7 @@ "path-key": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", - "dev": true + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=" }, "path-type": { "version": "1.1.0", @@ -5119,8 +5150,7 @@ "pify": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=" }, "pinkie": { "version": "2.0.4", @@ -5140,8 +5170,234 @@ "posix-character-classes": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", - "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", - "dev": true + "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=" + }, + "postcss": { + "version": "7.0.14", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.14.tgz", + "integrity": "sha512-NsbD6XUUMZvBxtQAJuWDJeeC4QFsmWsfozWxCJPWf3M55K9iu2iMDaKqyoOdTJ1R4usBXuxlVFAIo8rZPQD4Bg==", + "requires": { + "chalk": "^2.4.2", + "source-map": "^0.6.1", + "supports-color": "^6.1.0" + }, + "dependencies": { + "supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "postcss-cli": { + "version": "6.1.2", + "resolved": "https://registry.npmjs.org/postcss-cli/-/postcss-cli-6.1.2.tgz", + "integrity": "sha512-jIWfIkqt8cTThSpH8DBaNxHlBf99OKSem2RseRpfVPqWayxHKQB0IWdS/IF5XSGeFU5QslSDTdVHnw6qggXGkA==", + "requires": { + "chalk": "^2.1.0", + "chokidar": "^2.0.0", + "dependency-graph": "^0.8.0", + "fs-extra": "^7.0.0", + "get-stdin": "^6.0.0", + "globby": "^9.0.0", + "postcss": "^7.0.0", + "postcss-load-config": "^2.0.0", + "postcss-reporter": "^6.0.0", + "pretty-hrtime": "^1.0.3", + "read-cache": "^1.0.0", + "yargs": "^12.0.1" + }, + "dependencies": { + "camelcase": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.2.0.tgz", + "integrity": "sha512-IXFsBS2pC+X0j0N/GE7Dm7j3bsEBp+oTpb7F50dwEVX7rf3IgwO9XatnegTsDtniKCUtEJH4fSU6Asw7uoVLfQ==" + }, + "cliui": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-4.1.0.tgz", + "integrity": "sha512-4FG+RSG9DL7uEwRUZXZn3SS34DiDPfzP0VOiEwtUWlE+AR2EIg+hSyvrIgUUfhdgR/UkAeW2QHgeP+hWrXs7jQ==", + "requires": { + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0", + "wrap-ansi": "^2.0.0" + } + }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + } + }, + "execa": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", + "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "requires": { + "cross-spawn": "^6.0.0", + "get-stream": "^4.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "requires": { + "locate-path": "^3.0.0" + } + }, + "fs-extra": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", + "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "get-stdin": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-6.0.0.tgz", + "integrity": "sha512-jp4tHawyV7+fkkSKyvjuLZswblUtz+SQKzSWnBbii16BuZksJlU1wuBYXY75r+duh/llF1ur6oNwi+2ZzjKZ7g==" + }, + "get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "requires": { + "pump": "^3.0.0" + } + }, + "glob": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==", + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "globby": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-9.1.0.tgz", + "integrity": "sha512-VtYjhHr7ncls724Of5W6Kaahz0ag7dB4G62/2HsN+xEKG6SrPzM1AJMerGxQTwJGnN9reeyxdvXbuZYpfssCvg==", + "requires": { + "@types/glob": "^7.1.1", + "array-union": "^1.0.2", + "dir-glob": "^2.2.1", + "fast-glob": "^2.2.6", + "glob": "^7.1.3", + "ignore": "^4.0.3", + "pify": "^4.0.1", + "slash": "^2.0.0" + } + }, + "invert-kv": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/invert-kv/-/invert-kv-2.0.0.tgz", + "integrity": "sha512-wPVv/y/QQ/Uiirj/vh3oP+1Ww+AWehmi1g5fFWGPF6IpCBCDVrhgHRMvrLfdYcwDh3QJbGXDW4JAuzxElLSqKA==" + }, + "lcid": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lcid/-/lcid-2.0.0.tgz", + "integrity": "sha512-avPEb8P8EGnwXKClwsNUgryVjllcRqtMYa49NTsbQagYuT1DcXnl1915oxWjoyGrXR6zH/Y0Zc96xWsPcoDKeA==", + "requires": { + "invert-kv": "^2.0.0" + } + }, + "os-locale": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/os-locale/-/os-locale-3.1.0.tgz", + "integrity": "sha512-Z8l3R4wYWM40/52Z+S265okfFj8Kt2cC2MKY+xNi3kFs+XGI7WXu/I309QQQYbRW4ijiZ+yxs9pqEhJh0DqW3Q==", + "requires": { + "execa": "^1.0.0", + "lcid": "^2.0.0", + "mem": "^4.0.0" + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==" + }, + "which-module": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", + "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=" + }, + "yargs": { + "version": "12.0.5", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-12.0.5.tgz", + "integrity": "sha512-Lhz8TLaYnxq/2ObqHDql8dX8CJi97oHxrjUcYtzKbbykPtVW9WB+poxI+NM2UIzsMgNCZTIf0AQwsjK5yMAqZw==", + "requires": { + "cliui": "^4.0.0", + "decamelize": "^1.2.0", + "find-up": "^3.0.0", + "get-caller-file": "^1.0.1", + "os-locale": "^3.0.0", + "require-directory": "^2.1.1", + "require-main-filename": "^1.0.1", + "set-blocking": "^2.0.0", + "string-width": "^2.0.0", + "which-module": "^2.0.0", + "y18n": "^3.2.1 || ^4.0.0", + "yargs-parser": "^11.1.1" + } + }, + "yargs-parser": { + "version": "11.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-11.1.1.tgz", + "integrity": "sha512-C6kB/WJDiaxONLJQnF8ccx9SEeoTTLek8RVbaOIsrAUS8VrBEXfmeSnCZxygc+XC2sNMBIwOOnfcxiynjHsVSQ==", + "requires": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + } + } + } + }, + "postcss-load-config": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-2.0.0.tgz", + "integrity": "sha512-V5JBLzw406BB8UIfsAWSK2KSwIJ5yoEIVFb4gVkXci0QdKgA24jLmHZ/ghe/GgX0lJ0/D1uUK1ejhzEY94MChQ==", + "requires": { + "cosmiconfig": "^4.0.0", + "import-cwd": "^2.0.0" + } + }, + "postcss-reporter": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/postcss-reporter/-/postcss-reporter-6.0.1.tgz", + "integrity": "sha512-LpmQjfRWyabc+fRygxZjpRxfhRf9u/fdlKf4VHG4TSPbV2XNsuISzYW1KL+1aQzx53CAppa1bKG4APIB/DOXXw==", + "requires": { + "chalk": "^2.4.1", + "lodash": "^4.17.11", + "log-symbols": "^2.2.0", + "postcss": "^7.0.7" + } + }, + "postcss-value-parser": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", + "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==" }, "prelude-ls": { "version": "1.1.2", @@ -5155,6 +5411,11 @@ "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", "dev": true }, + "pretty-hrtime": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/pretty-hrtime/-/pretty-hrtime-1.0.3.tgz", + "integrity": "sha1-t+PqQkNaTJsnWdmeDyAesZWALuE=" + }, "process": { "version": "0.11.10", "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", @@ -5164,8 +5425,7 @@ "process-nextick-args": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz", - "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==", - "dev": true + "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==" }, "promise": { "version": "7.3.1", @@ -5179,7 +5439,6 @@ "version": "15.6.1", "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.6.1.tgz", "integrity": "sha512-4ec7bY1Y66LymSUOH/zARVYObB23AT2h8cf6e/O6ZALB/N0sqZFEx7rq6EYPX2MkOdKORuooI/H5k9TlR4q7kQ==", - "dev": true, "requires": { "fbjs": "^0.8.16", "loose-envify": "^1.3.1", @@ -5251,6 +5510,15 @@ "safe-buffer": "^5.1.2" } }, + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, "punycode": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", @@ -5292,9 +5560,9 @@ "dev": true }, "randombytes": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.0.6.tgz", - "integrity": "sha512-CIQ5OFxf4Jou6uOKe9t1AOgqpeU5fd70A8NPdHSGeYXqXsPe6peOwI0cUl88RWZ6sP1vPMV3avd/R6cZ5/sP1A==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", "dev": true, "requires": { "safe-buffer": "^5.1.0" @@ -5360,45 +5628,67 @@ } }, "react": { - "version": "16.8.1", - "resolved": "https://registry.npmjs.org/react/-/react-16.8.1.tgz", - "integrity": "sha512-wLw5CFGPdo7p/AgteFz7GblI2JPOos0+biSoxf1FPsGxWQZdN/pj6oToJs1crn61DL3Ln7mN86uZ4j74p31ELQ==", + "version": "16.8.4", + "resolved": "https://registry.npmjs.org/react/-/react-16.8.4.tgz", + "integrity": "sha512-0GQ6gFXfUH7aZcjGVymlPOASTuSjlQL4ZtVC5YKH+3JL6bBLCVO21DknzmaPlI90LN253ojj02nsapy+j7wIjg==", "requires": { "loose-envify": "^1.1.0", "object-assign": "^4.1.1", "prop-types": "^15.6.2", - "scheduler": "^0.13.1" + "scheduler": "^0.13.4" }, "dependencies": { "prop-types": { - "version": "15.6.2", - "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.6.2.tgz", - "integrity": "sha512-3pboPvLiWD7dkI3qf3KbUe6hKFKa52w+AE0VCqECtf+QHAKgOL37tTaNCnuX1nAAQ4ZhyP+kYVKf8rLmJ/feDQ==", + "version": "15.7.2", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.7.2.tgz", + "integrity": "sha512-8QQikdH7//R2vurIJSutZ1smHYTcLpRWEOlHnzcWHmBYrOGUysKwSsrC89BCiFj3CbrfJ/nXFdJepOVrY1GCHQ==", "requires": { - "loose-envify": "^1.3.1", - "object-assign": "^4.1.1" + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.8.1" + }, + "dependencies": { + "loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "requires": { + "js-tokens": "^3.0.0 || ^4.0.0" + } + } } } } }, "react-dom": { - "version": "16.8.1", - "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-16.8.1.tgz", - "integrity": "sha512-N74IZUrPt6UiDjXaO7UbDDFXeUXnVhZzeRLy/6iqqN1ipfjrhR60Bp5NuBK+rv3GMdqdIuwIl22u1SYwf330bg==", + "version": "16.8.4", + "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-16.8.4.tgz", + "integrity": "sha512-Ob2wK7XG2tUDt7ps7LtLzGYYB6DXMCLj0G5fO6WeEICtT4/HdpOi7W/xLzZnR6RCG1tYza60nMdqtxzA8FaPJQ==", "requires": { "loose-envify": "^1.1.0", "object-assign": "^4.1.1", "prop-types": "^15.6.2", - "scheduler": "^0.13.1" + "scheduler": "^0.13.4" }, "dependencies": { "prop-types": { - "version": "15.6.2", - "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.6.2.tgz", - "integrity": "sha512-3pboPvLiWD7dkI3qf3KbUe6hKFKa52w+AE0VCqECtf+QHAKgOL37tTaNCnuX1nAAQ4ZhyP+kYVKf8rLmJ/feDQ==", + "version": "15.7.2", + "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.7.2.tgz", + "integrity": "sha512-8QQikdH7//R2vurIJSutZ1smHYTcLpRWEOlHnzcWHmBYrOGUysKwSsrC89BCiFj3CbrfJ/nXFdJepOVrY1GCHQ==", "requires": { - "loose-envify": "^1.3.1", - "object-assign": "^4.1.1" + "loose-envify": "^1.4.0", + "object-assign": "^4.1.1", + "react-is": "^16.8.1" + }, + "dependencies": { + "loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "requires": { + "js-tokens": "^3.0.0 || ^4.0.0" + } + } } } } @@ -5419,6 +5709,29 @@ "prop-types": "^15.6.0" } }, + "react-is": { + "version": "16.8.4", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.8.4.tgz", + "integrity": "sha512-PVadd+WaUDOAciICm/J1waJaSvgq+4rHE/K70j0PFqKhkTBsPv/82UGQJNXAngz1fOQLLxI6z1sEDmJDQhCTAA==" + }, + "react-sortable-hoc": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/react-sortable-hoc/-/react-sortable-hoc-1.6.1.tgz", + "integrity": "sha512-ZIt3wjGeHQETMyv/dE9Wfk9a6T9IP9kGO++00vIiQkol+w7Gh6sUarBNWkGINgwYqwMWV0ALPAQdq4Ixwur8jw==", + "requires": { + "@babel/runtime": "^7.2.0", + "invariant": "^2.2.4", + "prop-types": "^15.5.7" + } + }, + "read-cache": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", + "integrity": "sha1-5mTvMRYRZsl1HNvo28+GtftY93Q=", + "requires": { + "pify": "^2.3.0" + } + }, "read-pkg": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-1.1.0.tgz", @@ -5444,7 +5757,6 @@ "version": "2.3.6", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", - "dev": true, "requires": { "core-util-is": "~1.0.0", "inherits": "~2.0.3", @@ -5458,8 +5770,7 @@ "inherits": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" } } }, @@ -5467,7 +5778,6 @@ "version": "2.2.1", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", - "dev": true, "requires": { "graceful-fs": "^4.1.11", "micromatch": "^3.1.10", @@ -5499,11 +5809,15 @@ "strip-indent": "^1.0.1" } }, + "regenerator-runtime": { + "version": "0.12.1", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.12.1.tgz", + "integrity": "sha512-odxIc1/vDlo4iZcfXqRYFj0vpXFNoGdKMAUieAlFYO6m/nl5e9KR/beGf41z4a1FI+aQgtjhuaSlDxQ0hmkrHg==" + }, "regex-not": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", - "dev": true, "requires": { "extend-shallow": "^3.0.2", "safe-regex": "^1.1.0" @@ -5531,20 +5845,17 @@ "remove-trailing-separator": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", - "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", - "dev": true + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=" }, "repeat-element": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", - "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", - "dev": true + "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==" }, "repeat-string": { "version": "1.6.1", "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", - "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", - "dev": true + "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=" }, "repeating": { "version": "2.0.1", @@ -5586,14 +5897,17 @@ "require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", - "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", - "dev": true + "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=" + }, + "require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==" }, "require-main-filename": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-1.0.1.tgz", - "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=", - "dev": true + "integrity": "sha1-l/cXtp1IeE9fUmpsWqj/3aBVpNE=" }, "requires-port": { "version": "1.0.0", @@ -5601,11 +5915,15 @@ "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=", "dev": true }, + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=" + }, "resolve-url": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", - "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", - "dev": true + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=" }, "restore-cursor": { "version": "2.0.0", @@ -5620,8 +5938,7 @@ "ret": { "version": "0.1.15", "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", - "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", - "dev": true + "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==" }, "rfdc": { "version": "1.1.2", @@ -5675,14 +5992,12 @@ "safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", - "dev": true + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" }, "safe-regex": { "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", + "resolved": "http://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", - "dev": true, "requires": { "ret": "~0.1.10" } @@ -5784,9 +6099,9 @@ "dev": true }, "scheduler": { - "version": "0.13.1", - "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.13.1.tgz", - "integrity": "sha512-VJKOkiKIN2/6NOoexuypwSrybx13MY7NSy9RNt8wPvZDMRT1CW6qlpF5jXRToXNHz3uWzbm2elNpZfXfGPqP9A==", + "version": "0.13.4", + "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.13.4.tgz", + "integrity": "sha512-cvSOlRPxOHs5dAhP9yiS/6IDmVAVxmk33f0CtTJRkmUWcb1Us+t7b1wqdzoC0REw2muC9V5f1L/w5R5uKGaepA==", "requires": { "loose-envify": "^1.1.0", "object-assign": "^4.1.1" @@ -5822,8 +6137,7 @@ "semver": { "version": "5.5.0", "resolved": "https://registry.npmjs.org/semver/-/semver-5.5.0.tgz", - "integrity": "sha512-4SJ3dm0WAwWy/NVeioZh5AntkdJoWKxHxcmyP622fOkgHa4z3R0TdBJICINyaSDE6uNwVc8gZr+ZinwZAH4xIA==", - "dev": true + "integrity": "sha512-4SJ3dm0WAwWy/NVeioZh5AntkdJoWKxHxcmyP622fOkgHa4z3R0TdBJICINyaSDE6uNwVc8gZr+ZinwZAH4xIA==" }, "semver-diff": { "version": "2.1.0", @@ -5837,14 +6151,12 @@ "set-blocking": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", - "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", - "dev": true + "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" }, "set-value": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.0.tgz", "integrity": "sha512-hw0yxk9GT/Hr5yJEYnHNKYXkIA8mVJgd9ditYZCe16ZczcaELYYcfvaXesNACk2O8O0nTiPQcQhGUQj8JLzeeg==", - "dev": true, "requires": { "extend-shallow": "^2.0.1", "is-extendable": "^0.1.1", @@ -5856,7 +6168,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } @@ -5885,9 +6196,9 @@ } }, "shadow-cljs": { - "version": "2.7.30", - "resolved": "https://registry.npmjs.org/shadow-cljs/-/shadow-cljs-2.7.30.tgz", - "integrity": "sha512-ZRYNZfz+ssBFafA7oMiNlkVxXcpSH1L48roq0jVGNx6raA2nGrQlxNWogNkyEgcDtqgNJuVR/WaoNHTo6k5qpg==", + "version": "2.8.14", + "resolved": "https://registry.npmjs.org/shadow-cljs/-/shadow-cljs-2.8.14.tgz", + "integrity": "sha512-StU3ir8zPj9hP98cFSufTGl0u6F3AO0EYE11NDYXqT8PzIwYEsD+7+ozGkydonJsTxLvWsNcO4hzaoWOt1wJ1w==", "dev": true, "requires": { "mkdirp": "^0.5.1", @@ -5958,7 +6269,6 @@ "version": "1.2.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", - "dev": true, "requires": { "shebang-regex": "^1.0.0" } @@ -5966,8 +6276,7 @@ "shebang-regex": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", - "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", - "dev": true + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=" }, "shell-quote": { "version": "1.6.1", @@ -5990,8 +6299,12 @@ "signal-exit": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", - "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", - "dev": true + "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=" + }, + "slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==" }, "smart-buffer": { "version": "1.1.15", @@ -6003,7 +6316,6 @@ "version": "0.8.2", "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", - "dev": true, "requires": { "base": "^0.11.1", "debug": "^2.2.0", @@ -6019,7 +6331,6 @@ "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, "requires": { "is-descriptor": "^0.1.0" } @@ -6028,7 +6339,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } @@ -6036,8 +6346,7 @@ "source-map": { "version": "0.5.7", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" } } }, @@ -6045,7 +6354,6 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", - "dev": true, "requires": { "define-property": "^1.0.0", "isobject": "^3.0.0", @@ -6056,7 +6364,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, "requires": { "is-descriptor": "^1.0.0" } @@ -6065,7 +6372,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -6074,7 +6380,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, "requires": { "kind-of": "^6.0.0" } @@ -6083,7 +6388,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, "requires": { "is-accessor-descriptor": "^1.0.0", "is-data-descriptor": "^1.0.0", @@ -6096,7 +6400,6 @@ "version": "3.0.1", "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", - "dev": true, "requires": { "kind-of": "^3.2.0" }, @@ -6105,7 +6408,6 @@ "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -6595,7 +6897,7 @@ }, "socket.io-parser": { "version": "3.2.0", - "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.2.0.tgz", + "resolved": "http://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.2.0.tgz", "integrity": "sha512-FYiBx7rc/KORMJlgsXysflWx/RIvtqZbyGLlHZvjfmPTPeuD/I8MaW7cfFrj5tRltICJdgwflhfZ3NVVbVLFQA==", "dev": true, "requires": { @@ -6653,14 +6955,12 @@ "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" }, "source-map-resolve": { "version": "0.5.2", "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.2.tgz", "integrity": "sha512-MjqsvNwyz1s0k81Goz/9vRBe9SZdB09Bdw+/zYyO+3CuPk6fouTaxscHkgtE8jKvf01kVfl8riHzERQ/kefaSA==", - "dev": true, "requires": { "atob": "^2.1.1", "decode-uri-component": "^0.2.0", @@ -6682,8 +6982,7 @@ "source-map-url": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", - "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", - "dev": true + "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=" }, "spdx-correct": { "version": "3.0.2", @@ -6721,7 +7020,6 @@ "version": "3.1.0", "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", - "dev": true, "requires": { "extend-shallow": "^3.0.0" } @@ -6729,8 +7027,7 @@ "sprintf-js": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", - "dev": true + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" }, "sshpk": { "version": "1.16.1", @@ -6753,7 +7050,6 @@ "version": "0.1.2", "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", - "dev": true, "requires": { "define-property": "^0.2.5", "object-copy": "^0.1.0" @@ -6763,7 +7059,6 @@ "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, "requires": { "is-descriptor": "^0.1.0" } @@ -6809,17 +7104,27 @@ } }, "streamroller": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/streamroller/-/streamroller-0.7.0.tgz", - "integrity": "sha512-WREzfy0r0zUqp3lGO096wRuUp7ho1X6uo/7DJfTlEi0Iv/4gT7YHqXDjKC2ioVGBZtE8QzsQD9nx1nIuoZ57jQ==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/streamroller/-/streamroller-1.0.3.tgz", + "integrity": "sha512-P7z9NwP51EltdZ81otaGAN3ob+/F88USJE546joNq7bqRNTe6jc74fTBDyynxP4qpIfKlt/CesEYicuMzI0yJg==", "dev": true, "requires": { - "date-format": "^1.2.0", + "async": "^2.6.1", + "date-format": "^2.0.0", "debug": "^3.1.0", - "mkdirp": "^0.5.1", - "readable-stream": "^2.3.0" + "fs-extra": "^7.0.0", + "lodash": "^4.17.10" }, "dependencies": { + "async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.2.tgz", + "integrity": "sha512-H1qVYh1MYhEEFLsP97cVKqCGo7KfCyTt6uEWqsTBr9SO84oK9Uwbyd/yCW+6rKJLHksBNUVWZDAjfS+Ccx0Bbg==", + "dev": true, + "requires": { + "lodash": "^4.17.11" + } + }, "debug": { "version": "3.2.6", "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", @@ -6829,6 +7134,17 @@ "ms": "^2.1.1" } }, + "fs-extra": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", + "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, "ms": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", @@ -6852,7 +7168,6 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", - "dev": true, "requires": { "is-fullwidth-code-point": "^2.0.0", "strip-ansi": "^4.0.0" @@ -6873,7 +7188,6 @@ "version": "1.1.1", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", - "dev": true, "requires": { "safe-buffer": "~5.1.0" } @@ -6882,7 +7196,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", - "dev": true, "requires": { "ansi-regex": "^3.0.0" } @@ -6899,8 +7212,7 @@ "strip-eof": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", - "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", - "dev": true + "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=" }, "strip-indent": { "version": "1.0.1", @@ -6936,7 +7248,6 @@ "version": "5.5.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "dev": true, "requires": { "has-flag": "^3.0.0" } @@ -7038,7 +7349,6 @@ "version": "0.3.0", "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", - "dev": true, "requires": { "kind-of": "^3.0.2" }, @@ -7047,7 +7357,6 @@ "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, "requires": { "is-buffer": "^1.1.5" } @@ -7058,7 +7367,6 @@ "version": "3.0.2", "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", - "dev": true, "requires": { "define-property": "^2.0.2", "extend-shallow": "^3.0.2", @@ -7070,7 +7378,6 @@ "version": "2.1.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", - "dev": true, "requires": { "is-number": "^3.0.0", "repeat-string": "^1.6.1" @@ -7194,7 +7501,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.0.tgz", "integrity": "sha1-XHHDTLW61dzr4+oM0IIHulqhrqQ=", - "dev": true, "requires": { "arr-union": "^3.1.0", "get-value": "^2.0.6", @@ -7206,7 +7512,6 @@ "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, "requires": { "is-extendable": "^0.1.0" } @@ -7215,7 +7520,6 @@ "version": "0.4.3", "resolved": "https://registry.npmjs.org/set-value/-/set-value-0.4.3.tgz", "integrity": "sha1-fbCPnT0i3H945Trzw79GZuzfzPE=", - "dev": true, "requires": { "extend-shallow": "^2.0.1", "is-extendable": "^0.1.1", @@ -7237,8 +7541,7 @@ "universalify": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.1.tgz", - "integrity": "sha1-+nG63UQ3r0wUiEHjs7Fl+enlkLc=", - "dev": true + "integrity": "sha1-+nG63UQ3r0wUiEHjs7Fl+enlkLc=" }, "unpipe": { "version": "1.0.0", @@ -7250,7 +7553,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", - "dev": true, "requires": { "has-value": "^0.3.1", "isobject": "^3.0.0" @@ -7260,7 +7562,6 @@ "version": "0.3.1", "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", - "dev": true, "requires": { "get-value": "^2.0.3", "has-values": "^0.1.4", @@ -7271,7 +7572,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", - "dev": true, "requires": { "isarray": "1.0.0" } @@ -7281,8 +7581,7 @@ "has-values": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", - "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", - "dev": true + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=" } } }, @@ -7295,8 +7594,7 @@ "upath": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/upath/-/upath-1.1.0.tgz", - "integrity": "sha512-bzpH/oBhoS/QI/YtbkqCg6VEiPYjSZtrHQM6/QnJS6OL9pKUFLqb3aFh4Scvwm45+7iAgiMkLhSbaZxUqmrprw==", - "dev": true + "integrity": "sha512-bzpH/oBhoS/QI/YtbkqCg6VEiPYjSZtrHQM6/QnJS6OL9pKUFLqb3aFh4Scvwm45+7iAgiMkLhSbaZxUqmrprw==" }, "update-notifier": { "version": "2.5.0", @@ -7328,8 +7626,7 @@ "urix": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", - "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", - "dev": true + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=" }, "url": { "version": "0.11.0", @@ -7361,8 +7658,7 @@ "use": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", - "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", - "dev": true + "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==" }, "useragent": { "version": "2.3.0", @@ -7394,8 +7690,7 @@ "util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", - "dev": true + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" }, "utils-merge": { "version": "1.0.1", @@ -7460,7 +7755,6 @@ "version": "1.3.0", "resolved": "https://registry.npmjs.org/which/-/which-1.3.0.tgz", "integrity": "sha512-xcJpopdamTuY5duC/KnTTNBraPK54YwpenP4lzxU8H91GudWpFv38u0CKjclE1Wi2EH2EDz5LRcHcKbCIzqGyg==", - "dev": true, "requires": { "isexe": "^2.0.0" } @@ -7514,7 +7808,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", - "dev": true, "requires": { "string-width": "^1.0.1", "strip-ansi": "^3.0.1" @@ -7523,14 +7816,12 @@ "ansi-regex": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", - "dev": true + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" }, "is-fullwidth-code-point": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", - "dev": true, "requires": { "number-is-nan": "^1.0.0" } @@ -7539,7 +7830,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", - "dev": true, "requires": { "code-point-at": "^1.0.0", "is-fullwidth-code-point": "^1.0.0", @@ -7550,7 +7840,6 @@ "version": "3.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", - "dev": true, "requires": { "ansi-regex": "^2.0.0" } @@ -7560,8 +7849,7 @@ "wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "dev": true + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" }, "write-file-atomic": { "version": "2.4.2", @@ -7628,8 +7916,7 @@ "y18n": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/y18n/-/y18n-3.2.1.tgz", - "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE=", - "dev": true + "integrity": "sha1-bRX7qITAhnnA136I53WegR4H+kE=" }, "yallist": { "version": "2.1.2", diff --git a/package.json b/package.json index 2d69048..415766b 100644 --- a/package.json +++ b/package.json @@ -5,13 +5,13 @@ "main": "index.js", "scripts": { "build:cljs": "shadow-cljs release app", - "build:sass": "node-sass --output-style compressed src/sass/app.sass public/app/style.css", + "build:sass": "node-sass --output-style compressed src/sass/app.sass | postcss -o public/app/style.css", "build": "rm -r public/*; run-p copy:* build:*", "copy:assets": "cp -R src/assets/* public/", "copy:icons": "cp -R node_modules/open-iconic/font/fonts public", "deploy": "npm run build && gh-pages -d public -m \"Deploying $(git rev-parse --short HEAD)\"", "dev:cljs": "shadow-cljs watch app test", - "dev:sass": "npm run build:sass; node-sass -w src/sass/app.sass public/app/style.css", + "dev:sass": "npm run build:sass; node-sass -w src/sass/app.sass | postcss -o public/app/style.css", "dev:test": "karma start --reporters notify,progress --auto-watch", "dev": "rm -r public/*; npm-run-all copy:* test:compile -p dev:*", "test": "run-s test:compile test:run", @@ -26,15 +26,18 @@ }, "dependencies": { "@hugojosefson/color-hash": "^2.0.3", + "autoprefixer": "^9.4.10", "bulma": "^0.7.3", "create-react-class": "^15.6.3", "open-iconic": "^1.1.1", - "react": "^16.8.1", - "react-dom": "^16.8.1" + "postcss-cli": "^6.1.2", + "react": "^16.8.4", + "react-dom": "^16.8.4", + "react-sortable-hoc": "^1.6.1" }, "devDependencies": { "gh-pages": "^1.2.0", - "karma": "^3.1.4", + "karma": "^4.0.1", "karma-chrome-launcher": "^2.2.0", "karma-cljs-test": "^0.1.0", "karma-notify-reporter": "^1.1.0", @@ -43,6 +46,6 @@ "react-flip-move": "^3.0.3", "react-highlight.js": "^1.0.7", "sass": "^1.17.0", - "shadow-cljs": "^2.7.30" + "shadow-cljs": "^2.8.14" } } diff --git a/postcss.config.js b/postcss.config.js new file mode 100644 index 0000000..88752c6 --- /dev/null +++ b/postcss.config.js @@ -0,0 +1,5 @@ +module.exports = { + plugins: [ + require('autoprefixer') + ] +} diff --git a/shadow-cljs.edn b/shadow-cljs.edn index 6d36d43..448bac8 100644 --- a/shadow-cljs.edn +++ b/shadow-cljs.edn @@ -11,11 +11,11 @@ [funcool/bide "1.6.0"] [fipp "0.6.14"] ;; debugging - [day8.re-frame/re-frame-10x "0.3.3-react16"] - [day8.re-frame/tracing "0.5.1"] - [philoskim/debux "0.4.11"] + [day8.re-frame/re-frame-10x "0.3.7-react16"] + #_[day8.re-frame/tracing "0.5.1"] + [philoskim/debux "0.5.6"] ;; for CIDER - [cider/cider-nrepl "0.18.0"]] + [cider/cider-nrepl "0.21.1"]] :nrepl {:port 9000} diff --git a/src/cljs/airsonic_ui/audio/core.cljs b/src/cljs/airsonic_ui/audio/core.cljs index bd52044..132bc2e 100644 --- a/src/cljs/airsonic_ui/audio/core.cljs +++ b/src/cljs/airsonic_ui/audio/core.cljs @@ -6,8 +6,6 @@ [airsonic-ui.audio.playlist :as playlist] [goog.functions :refer [throttle]])) -;; TODO: Manage buffering - (defonce audio (atom nil)) (defn normalize-time-ranges [time-ranges] @@ -28,7 +26,6 @@ ; explanation of these events: https://developer.mozilla.org/en-US/Apps/Fundamentals/Audio_and_video_delivery/Cross-browser_audio_basics - (defn attach-listeners! [el] (let [emit-audio-update (throttle #(rf/dispatch [:audio/update (->status el)]) 16)] (doseq [event ["loadstart" "progress" "play" "timeupdate" "pause" "volumechange"]] @@ -56,7 +53,8 @@ (fn [_] (when-let [audio @audio] (.pause audio) - (set! (.-currentTime audio) 0)))) + (set! (.-currentTime audio) 0) + (set! (.-src audio) "")))) (rf/reg-fx :audio/toggle-play-pause @@ -102,25 +100,26 @@ (rf/reg-sub :audio/summary summary) -(defn playlist - "Lists the complete playlist" +(defn current-playlist + "Lists the complete current-queue" [summary _] - (:playlist summary)) + (:current-playlist summary)) (rf/reg-sub - :audio/playlist + :audio/current-playlist :<- [:audio/summary] - playlist) + current-playlist) (defn current-song "Gives us information about the currently played song as presented by the airsonic api" [playlist _] - (playlist/peek playlist)) + (when-not (empty? playlist) + (playlist/current-song playlist))) (rf/reg-sub :audio/current-song - :<- [:audio/playlist] + :<- [:audio/current-playlist] current-song) (defn playback-status diff --git a/src/cljs/airsonic_ui/audio/playlist.cljs b/src/cljs/airsonic_ui/audio/playlist.cljs index d36d2c8..47032ee 100644 --- a/src/cljs/airsonic_ui/audio/playlist.cljs +++ b/src/cljs/airsonic_ui/audio/playlist.cljs @@ -1,139 +1,196 @@ (ns airsonic-ui.audio.playlist "Implements playlist queues that support different kinds of repetition and - song ordering." - (:refer-clojure :exclude [peek]) - (:require [airsonic-ui.helpers :refer [find-where]])) + song ordering.") -(defrecord Playlist [queue playback-mode repeat-mode] +;; Turns out we can nicely implement this by thinly wrapping a sequence of items +;; We re-use the core ClojureScript protocols internally but provide a nice and +;; explicit API to consume + +(defprotocol IPlaylist + (current-song [this]) + (next-song [this]) + (previous-song [this]) + + (set-current-song [this song-idx] + "Advances the queue to the song given by song-idx") + (set-playback-mode [this playback-mode] + "Changes the playback mode of a playlist and re-shuffles it if necessary") + (set-repeat-mode [this repeat-mode] + "Allows you to change how the next and previous song are selected") + + (enqueue-last + [this song source] + [this song] + "Registers a song to be played last, optionally remembering the source route") + (enqueue-next + [this song source] + [this song] + "Registers a song to be played next, optionally remembering the source route") + + (move-song [this from-idx to-idx] + "Allows you to move a song in a playlist") + (remove-song [this song-idx] + "Removes a song from the playlist")) + +;; helpers to manage creating playlists + +(defn- mark-original-order + "This function is used if we switch from linear to shuffled; it allows us to + restore the order of the queue when it was created." + [items] + (->> (sort-by (comp :playlist/linear-order meta) items) + (map-indexed (fn [idx item] + (vary-meta item assoc :playlist/linear-order idx))))) + +(defn- linear-queue + [items] + (->> (mark-original-order items) + (map-indexed vector) + (into (sorted-map)))) + +(defn- shuffled-queue + [items] + (let [shuffled-indices (shuffle (range (count items)))] + (->> (mark-original-order items) + (map vector shuffled-indices) + (into (sorted-map))))) + +;; the exported interface: + +(defrecord Playlist [items current-idx playback-mode repeat-mode] cljs.core/ICounted - (-count [this] - (count (:queue this)))) + (-count [_] + (count items)) + + cljs.core/ISequential + cljs.core/ISeqable + (-seq [_] items) + + IPlaylist + (current-song [_] + (get items current-idx)) + + (next-song [this] + (update this :current-idx + (fn [current-idx] + (cond + (= repeat-mode :repeat-single) current-idx + + (or (= repeat-mode :repeat-all) + (< current-idx (dec (count this)))) + (mod (inc current-idx) (count this)))))) + + (previous-song [this] + (update this :current-idx + (fn [current-idx] + (cond + (= repeat-mode :repeat-single) current-idx + + (or (= repeat-mode :repeat-all) + (> current-idx 0)) + (mod (dec current-idx) (count this)) + + :else nil)))) + + (set-current-song [playlist song-idx] + (assoc playlist :current-idx song-idx)) + + (set-playback-mode [playlist playback-mode] + (let [current-song (current-song playlist) + queue-fn (case playback-mode + :shuffled shuffled-queue + :linear linear-queue) + next-playlist (-> (assoc playlist :playback-mode playback-mode) + (update :items (comp queue-fn vals))) + next-idx (first (keep (fn [[idx song]] + (when (= song current-song) + idx)) + (:items next-playlist)))] + ;; we have to find out the index of the currently playing song after the + ;; playlist was created because it might change when shuffling / unshuffling + (set-current-song next-playlist next-idx))) + + (set-repeat-mode [playlist repeat-mode] + (assoc playlist :repeat-mode repeat-mode)) + + (enqueue-last [this song source] + (let [order (inc (key (last items)))] + ;; Arguably this is a bit weird; but if you want to play something last in + ;; a shuffled playlist, you want to play it last I guess. + (assoc-in this [:items order] + (vary-meta song assoc + :playlist/linear-order order + :playlist/source source)))) + (enqueue-last [this song] (enqueue-last this song nil)) + + (enqueue-next [this song source] + ;; we slice the songs up until the currently playing one and increase the + ;; order for all the songs after + (let [songs (vec (vals items)) + reordered (-> (subvec songs 0 (inc current-idx)) + (conj (vary-meta song assoc + :playlist/linear-order (inc current-idx) + :playlist/source source)) + (concat (subvec songs (inc current-idx))))] + (assoc this :items (->> (map-indexed vector reordered) + (into (sorted-map)))))) + (enqueue-next [this song] (enqueue-next this song nil)) + + (move-song [this from-idx to-idx] + ;; we have to decide whether we move all items in-between + ;; one up or one down; this depends on whether we move our + ;; item to the front or to the back + (let [shift-fn (cond + (< from-idx to-idx) inc + (> from-idx to-idx) dec) + start (min from-idx to-idx) + end (inc (max from-idx to-idx)) + steps (range start end) + result (update this :items + (fn [items] + (-> (reduce (fn [result idx] + (assoc result idx (get items (shift-fn idx)))) + items steps) + (assoc to-idx (get items from-idx)))))] + (cond + (= from-idx current-idx) (assoc result :current-idx to-idx) + (<= to-idx current-idx from-idx) (update result :current-idx inc) + (>= to-idx current-idx from-idx) (update result :current-idx dec) + :else result))) + + (remove-song [this song-idx] + (cond-> (update this :items #(let [n-items (count %)] + (-> (reduce (fn [items idx] + (assoc items idx (get items (inc idx)))) + % (range song-idx n-items)) + (dissoc (dec n-items))))) + (= song-idx current-idx) (assoc :current-idx -1)))) + +;; constructor wrapper + +(defn set-item-source + "Can be used to attach a source route to an item" + [item source] + (vary-meta item assoc :playlist/source source)) + +(defn item-source + "Retrieve the source of an item in the playlist" + [item] + (:playlist/source (meta item))) (defmulti ->playlist "Creates a new playlist that behaves according to the given playback- and repeat-mode parameters." - (fn [queue & {:keys [playback-mode #_repeat-mode]}] - playback-mode)) - -(defn- mark-first-song [queue] - (let [[first-idx _] (find-where #(= 0 (:playlist/order %)) queue)] - (assoc-in queue [first-idx :playlist/currently-playing?] true))) + (fn [_ & {:keys [playback-mode]}] playback-mode)) (defmethod ->playlist :linear - [queue & {:keys [playback-mode repeat-mode]}] - (let [queue (-> (mapv (fn [order song] (assoc song :playlist/order order)) (range) queue) - (mark-first-song))] - (->Playlist queue playback-mode repeat-mode))) - -(defn- -shuffle-songs [queue] - (->> (shuffle (range (count queue))) - (mapv (fn [song order] (assoc song :playlist/order order)) queue))) + [items & {:keys [playback-mode repeat-mode source]}] + (->Playlist (->> (map #(set-item-source % source) items) + (linear-queue)) + 0 playback-mode repeat-mode)) (defmethod ->playlist :shuffled - [queue & {:keys [playback-mode repeat-mode]}] - (let [queue (conj (mapv #(update % :playlist/order inc) (-shuffle-songs (rest queue))) - (assoc (first queue) :playlist/order 0 :playlist/currently-playing? true))] - (->Playlist queue playback-mode repeat-mode))) - -(defn set-current-song - "Marks a song in the queue as currently playing, given its ID" - [playlist next-idx] - (let [[current-idx _] (find-where :playlist/currently-playing? (:queue playlist))] - (-> (if current-idx - (update-in playlist [:queue current-idx] dissoc :playlist/currently-playing?) - playlist) - (assoc-in [:queue next-idx :playlist/currently-playing?] true)))) - -(defn set-playback-mode - "Changes the playback mode of a playlist and re-shuffles it if necessary" - [playlist playback-mode] - (if (= playback-mode :shuffled) - ;; for shuffled playlists we reorder the songs make sure that the currently - ;; playing song has order 0 - (let [playlist (->playlist (:queue playlist) :playback-mode playback-mode :repeat-mode (:repeat-mode playlist)) - [current-idx current-song] (find-where :playlist/currently-playing? (:queue playlist)) - [swap-idx _] (find-where #(= 0 (:playlist/order %)) (:queue playlist))] - (-> (assoc-in playlist [:queue current-idx :playlist/order] 0) - (assoc-in [:queue swap-idx :playlist/order] (:playlist/order current-song)))) - ;; for linear songs we just make sure that the current does not change - (let [[current-idx _] (find-where :playlist/currently-playing? (:queue playlist))] - (-> (->playlist (:queue playlist) :playback-mode playback-mode :repeat-mode (:repeat-mode playlist)) - (set-current-song current-idx))))) - -(defn set-repeat-mode - "Allows to change the way the next and previous song of a playlist is selected" - [playlist repeat-mode] - (assoc playlist :repeat-mode repeat-mode)) - -(defn peek - "Returns the song in a playlist that is currently playing" - [playlist] - (->> (:queue playlist) - (filter :playlist/currently-playing?) - (first))) - -(defmulti next-song "Advances the currently playing song" :repeat-mode) - -(defmethod next-song :repeat-none - [playlist] - ;; this is pretty easy; get the next song and stop playing at the at - (let [[current-idx current-song] (find-where :playlist/currently-playing? (:queue playlist)) - [next-idx _] (find-where #(= (:playlist/order %) (inc (:playlist/order current-song))) (:queue playlist))] - (update playlist :queue - (fn [queue] - (cond-> queue - current-idx (update current-idx dissoc :playlist/currently-playing?) - next-idx (assoc-in [next-idx :playlist/currently-playing?] true)))))) - -(defmethod next-song :repeat-single [playlist] playlist) - -(defmethod next-song :repeat-all - [playlist] - (let [[current-idx current-song] (find-where :playlist/currently-playing? (:queue playlist)) - [next-idx _] (find-where #(= (:playlist/order %) (inc (:playlist/order current-song))) (:queue playlist))] - (-> (update-in playlist [:queue current-idx] dissoc :playlist/currently-playing?) - (update :queue - (fn [queue] - ;; we need special treatment here if we're playing the last song and - ;; have a shuffled playlist because we need to re-shuffle - (if next-idx - (assoc-in queue [next-idx :playlist/currently-playing?] true) - (case (:playback-mode playlist) - :linear (assoc-in queue [0 :playlist/currently-playing?] true) - :shuffled (let [queue' (-shuffle-songs queue) - [next-idx _] (find-where #(= (:playlist/order %) 0) queue')] - (assoc-in queue' [next-idx :playlist/currently-playing?] true))))))))) - -(defmulti previous-song "Goes back along the playback queue" :repeat-mode) - -(defmethod previous-song :repeat-single [playlist] playlist) - -(defmethod previous-song :repeat-none [playlist] - (let [[current-idx current-song] (find-where :playlist/currently-playing? (:queue playlist)) - [next-idx _] (find-where #(= (:playlist/order %) (dec (:playlist/order current-song))) (:queue playlist))] - (set-current-song playlist (or next-idx current-idx)))) - -(defmethod previous-song :repeat-all [playlist] - (let [[_ current-song] (find-where :playlist/currently-playing? (:queue playlist)) - [next-idx _] (find-where #(= (:playlist/order %) - (rem (dec (:playlist/order current-song)) (count playlist))) - (:queue playlist))] - (if next-idx - (set-current-song playlist next-idx) - (if (= :shuffled (:playback-mode playlist)) - (let [highest-order (dec (count playlist)) - playlist (update playlist :queue -shuffle-songs) - [last-idx _] (find-where #(= (:playlist/order %) highest-order) (:queue playlist))] - (set-current-song playlist last-idx)) - (set-current-song playlist (mod (dec (:playlist/order current-song)) (count playlist))))))) - -(defn enqueue-last [playlist song] - (let [highest-order (last (sort (map :playlist/order (:queue playlist))))] - (update playlist :queue conj (assoc song :playlist/order (inc highest-order))))) - -(defn enqueue-next [playlist song] - (let [[_ current-song] (find-where :playlist/currently-playing? (:queue playlist))] - (update playlist :queue - (fn [queue] - (-> (mapv #(if (> (:playlist/order %) (:playlist/order current-song)) (update % :playlist/order inc) %) queue) - (conj (assoc song :playlist/order (inc (:playlist/order current-song))))))))) + [items & {:keys [playback-mode repeat-mode source]}] + (->Playlist (->> (map #(set-item-source % source) items) + (shuffled-queue)) + 0 playback-mode repeat-mode)) diff --git a/src/cljs/airsonic_ui/components/audio_player/events.cljs b/src/cljs/airsonic_ui/components/audio_player/events.cljs index de96947..c476049 100644 --- a/src/cljs/airsonic_ui/components/audio_player/events.cljs +++ b/src/cljs/airsonic_ui/components/audio_player/events.cljs @@ -3,56 +3,85 @@ [airsonic-ui.audio.playlist :as playlist] [airsonic-ui.api.helpers :as api])) +; sets up the db, starts to play a song and adds the rest to a playlist +(defn play-all-songs [{:keys [db] + :routes/keys [current-route]} [_ songs start-idx]] + (let [playlist (-> (playlist/->playlist songs :playback-mode :linear :repeat-mode :repeat-all :source current-route) + (playlist/set-current-song start-idx))] + {:audio/play (api/stream-url (:credentials db) (playlist/current-song playlist)) + :db (assoc-in db [:audio :current-playlist] playlist)})) + (rf/reg-event-fx - ; sets up the db, starts to play a song and adds the rest to a playlist :audio-player/play-all - (fn [{:keys [db]} [_ songs start-idx]] - (let [playlist (-> (playlist/->playlist songs :playback-mode :linear :repeat-mode :repeat-all) - (playlist/set-current-song start-idx))] - {:audio/play (api/stream-url (:credentials db) (playlist/peek playlist)) - :db (assoc-in db [:audio :playlist] playlist)}))) + [(rf/inject-cofx :routes/current-route)] + play-all-songs) (rf/reg-event-db :audio-player/set-playback-mode (fn [db [_ playback-mode]] - (update-in db [:audio :playlist] #(playlist/set-playback-mode % playback-mode)))) + (update-in db [:audio :current-playlist] #(playlist/set-playback-mode % playback-mode)))) (rf/reg-event-db :audio-player/set-repeat-mode (fn [db [_ repeat-mode]] - (update-in db [:audio :playlist] #(playlist/set-repeat-mode % repeat-mode)))) + (update-in db [:audio :current-playlist] #(playlist/set-repeat-mode % repeat-mode)))) (rf/reg-event-fx :audio-player/next-song (fn [{:keys [db]} _] - (let [db (update-in db [:audio :playlist] playlist/next-song) - next (playlist/peek (get-in db [:audio :playlist]))] + (let [db (update-in db [:audio :current-playlist] playlist/next-song) + next (playlist/current-song (get-in db [:audio :current-playlist]))] {:db db :audio/play (api/stream-url (:credentials db) next)}))) (rf/reg-event-fx :audio-player/previous-song (fn [{:keys [db]} _] - (let [db (update-in db [:audio :playlist] playlist/previous-song) - prev (playlist/peek (get-in db [:audio :playlist]))] + (let [db (update-in db [:audio :current-playlist] playlist/previous-song) + song (playlist/current-song (get-in db [:audio :current-playlist]))] {:db db - :audio/play (api/stream-url (:credentials db) prev)}))) + :audio/play (api/stream-url (:credentials db) song)}))) -(rf/reg-event-db +(defn set-current-song [{:keys [db]} [_ idx]] + (let [db (update-in db [:audio :current-playlist] playlist/set-current-song idx) + song (playlist/current-song (get-in db [:audio :current-playlist]))] + {:db db + :audio/play (api/stream-url (:credentials db) song)})) + +(rf/reg-event-fx :audio-player/set-current-song set-current-song) + +(rf/reg-event-fx :audio-player/enqueue-next - (fn [db [_ song]] - (update-in db [:audio :playlist] #(playlist/enqueue-next % song)))) + [(rf/inject-cofx :routes/current-route)] + (fn [{:keys [db] + :routes/keys [current-route]} [_ song]] + {:db (update-in db [:audio :current-playlist] #(playlist/enqueue-next % song current-route))})) + +(rf/reg-event-fx + :audio-player/enqueue-last + [(rf/inject-cofx :routes/current-route)] + (fn [{:keys [db] + :routes/keys [current-route]} [_ song]] + {:db (update-in db [:audio :current-playlist] #(playlist/enqueue-last % song current-route))})) (rf/reg-event-db - :audio-player/enqueue-last - (fn [db [_ song]] - (update-in db [:audio :playlist] #(playlist/enqueue-last % song)))) + :audio-player/move-song + (fn [db [_ from-idx to-idx]] + (update-in db [:audio :current-playlist] #(playlist/move-song % from-idx to-idx)))) (rf/reg-event-fx :audio-player/toggle-play-pause (fn [_ _] {:audio/toggle-play-pause nil})) +(defn remove-song [{:keys [db]} [_ song-idx]] + (let [song-removed (update-in db [:audio :current-playlist] #(playlist/remove-song % song-idx))] + (cond-> {:db song-removed} + (nil? (playlist/current-song (get-in song-removed [:audio :current-playlist]))) + (assoc :audio/stop nil)))) + +(rf/reg-event-fx :audio-player/remove-song remove-song) + (defn audio-update "Reacts to audio events fired by the HTML5 audio player and plays the next track if necessary." @@ -65,7 +94,7 @@ (rf/reg-event-fx :audio-player/seek (fn [{:keys [db]} [_ percentage]] - (let [duration (:duration (playlist/peek (get-in db [:audio :playlist])))] + (let [duration (:duration (playlist/current-song (get-in db [:audio :current-playlist])))] {:audio/seek [percentage duration]}))) (rf/reg-event-fx diff --git a/src/cljs/airsonic_ui/components/audio_player/views.cljs b/src/cljs/airsonic_ui/components/audio_player/views.cljs index e1dfacf..8a35def 100644 --- a/src/cljs/airsonic_ui/components/audio_player/views.cljs +++ b/src/cljs/airsonic_ui/components/audio_player/views.cljs @@ -4,7 +4,7 @@ [airsonic-ui.routes :as routes] [airsonic-ui.helpers :as h] [airsonic-ui.views.cover :refer [cover]] - [airsonic-ui.views.icon :refer [icon]])) + [bulma.icon :refer [icon]])) ;; currently playing / coming next / audio controls... @@ -121,9 +121,8 @@ {:on-click toggle-volume-slider} [icon volume-icon]]])) -(defn playback-mode-controls [playlist] - (let [{:keys [repeat-mode playback-mode]} playlist - button :p.control>button.button.is-light +(defn playback-mode-controls [{:keys [repeat-mode playback-mode]}] + (let [button :p.control>button.button.is-light shuffle-button (h/add-classes button (when (= playback-mode :shuffled) :is-primary)) repeat-button (h/add-classes button (case repeat-mode :repeat-single :is-info @@ -142,7 +141,7 @@ (defn audio-player [] (let [current-song @(subscribe [:audio/current-song]) - playlist @(subscribe [:audio/playlist]) + current-playlist @(subscribe [:audio/current-playlist]) playback-status @(subscribe [:audio/playback-status]) is-playing? @(subscribe [:audio/is-playing?])] [:nav.audio-player @@ -153,6 +152,6 @@ [progress-indicators current-song playback-status] [playback-controls is-playing?] [volume-controls playback-status] - [playback-mode-controls playlist]] + [playback-mode-controls current-playlist]] ;; not playing anything [:p.navbar-item.idle-notification "No audio playing"])])) diff --git a/src/cljs/airsonic_ui/components/collection/views.cljs b/src/cljs/airsonic_ui/components/collection/views.cljs index ce98c0e..3070731 100644 --- a/src/cljs/airsonic_ui/components/collection/views.cljs +++ b/src/cljs/airsonic_ui/components/collection/views.cljs @@ -1,31 +1,36 @@ (ns airsonic-ui.components.collection.views "A collection is a list of audio files that belong together (e.g. an album or a podcast's overview)" - (:require [airsonic-ui.helpers :refer [format-duration]] - [airsonic-ui.routes :as routes :refer [url-for]] - [airsonic-ui.views.cover :refer [cover card]] - [airsonic-ui.views.icon :refer [icon]] - [airsonic-ui.views.song :as song])) + (:require [re-frame.core :refer [subscribe]] + [bulma.icon :refer [icon]] + [bulma.dropdown.views :refer [dropdown]] + [airsonic-ui.helpers :as h] + [airsonic-ui.routes :as routes] + [airsonic-ui.views.cover :refer [cover card]])) (defn collection-info [{:keys [songCount duration year]}] (vec (cond-> [:ul.is-smaller.collection-info [:li [icon :audio-spectrum] (str songCount (if (= 1 songCount) " track" " tracks"))] - [:li [icon :clock] (format-duration duration)]] + [:li [icon :clock] (h/format-duration duration)]] year (conj [:li [icon :calendar] (str "Released in " year)])))) -(defn album-card [album] - (let [{:keys [artist artistId name id]} album] - [card album - :url-fn #(url-for ::routes/album.detail {:id id}) - :content [:div - ;; link to album - [:div.title.is-5 - [:a {:href (url-for ::routes/album.detail {:id id}) - :title name} name]] - ;; link to artist page - [:div.subtitle.is-6 [:a {:href (url-for ::routes/artist.detail {:id artistId}) - :title artist} artist]]]])) +;; TODO: Maybe this view belongs somewhere else? +;; Something like a collection-grid component? + +(defn album-card + "A single element in a grid of albums. Shows the cover, artist and album name." + [{:keys [artist artistId name id] :as album}] + [card album + :url-fn #(routes/url-for ::routes/album.detail {:id id}) + :content [:div + ;; link to album + [:div.title.is-5 + [:a {:href (routes/url-for ::routes/album.detail {:id id}) + :title name} name]] + ;; link to artist page + [:div.subtitle.is-6 [:a {:href (routes/url-for ::routes/artist.detail {:id artistId}) + :title artist} artist]]]]) (defn listing [albums] ;; always show 5 in a row @@ -34,8 +39,54 @@ ^{:key idx} [:div.column.is-one-fifth-desktop.is-one-quarter-tablet.is-half-mobile [album-card album]])]) +;; TODO: Avoid duplication +(defn artist-link [{id :artistId, artist :artist}] + (if id + [:a {:href (routes/url-for ::routes/artist.detail {:id id})} artist] + artist)) + +(defn song-link [{:keys [songs song idx]}] + [:a + {:href "#" :on-click (h/muted-dispatch [:audio-player/play-all songs idx] :sync? true)} + (:title song)]) + +(defn song-actions [song] + [dropdown {:items [{:label "Play next" :event [:audio-player/enqueue-next song]} + {:label "Play last" :event [:audio-player/enqueue-last song]}]}]) + +(defn default-thead [] + [:thead>tr + [:td.is-narrow] + [:td.song-artist "Artist"] + [:td.song-title "Title"] + [:td.song-duration "Duration"] + [:td.is-narrow]]) + +(defn default-tbody [{:keys [songs current-song]}] + [:tbody + (for [[idx song] (map-indexed vector songs)] + ^{:key idx} + [(if (= (:id song) (:id current-song)) :tr.is-playing :tr) + [:td.song-tracknr.is-narrow (:track song)] + [:td.song-artist [artist-link song]] + [:td.song-title [song-link {:songs songs + :song song + :idx idx}]] + [:td.song-duration (h/format-duration (:duration song) :brief? true)] + [:td.song-actions.is-narrow [song-actions song]]])]) + +(defn song-table [{:keys [songs thead tbody] + :or {thead default-thead, tbody default-tbody}}] + ;; we subscribe here instead of one level higher up to make this a more + ;; reusable component; this way we can for example get a list of all songs + ;; in a search result and easily highlight the currently playing track + (let [current-song @(subscribe [:audio/current-song])] + [:table.song-listing-table.table.is-fullwidth + [thead] + [tbody {:songs songs, :current-song current-song}]])) + (defn detail - "Lists all songs in an album" + "Shows a detail view of a single album, listing all " [{:keys [album]}] [:div [:section.hero.is-small>div.hero-body @@ -46,4 +97,5 @@ [:h2.title (:name album)] [:h3.subtitle (:artist album)] [collection-info album]]]]] - [:section.section>div.container [song/listing (:song album)]]]) + [:section.section>div.container + [song-table {:songs (:song album)}]]]) diff --git a/src/cljs/airsonic_ui/components/current_queue/subs.cljs b/src/cljs/airsonic_ui/components/current_queue/subs.cljs new file mode 100644 index 0000000..d153aac --- /dev/null +++ b/src/cljs/airsonic_ui/components/current_queue/subs.cljs @@ -0,0 +1,15 @@ +(ns airsonic-ui.components.current-queue.subs + (:require [re-frame.core :as rf])) + +(defn queue-info [playlist] + {:count (count playlist) + :duration + (reduce (fn [acc [_ item]] + (+ acc (:duration item))) 0 (:items playlist))}) + +(println "registering the sub") + +(rf/reg-sub + :current-queue/info + :<- [:audio/current-playlist] + queue-info) diff --git a/src/cljs/airsonic_ui/components/current_queue/views.cljs b/src/cljs/airsonic_ui/components/current_queue/views.cljs index 1124f33..9b52ed8 100644 --- a/src/cljs/airsonic_ui/components/current_queue/views.cljs +++ b/src/cljs/airsonic_ui/components/current_queue/views.cljs @@ -1,12 +1,107 @@ (ns airsonic-ui.components.current-queue.views - (:require [re-frame.core :refer [subscribe]] - [airsonic-ui.views.song :as song] - [airsonic-ui.routes :as r])) + (:require [re-frame.core :refer [subscribe dispatch-sync]] + [reagent.core :as r] + ["react-sortable-hoc" :refer [SortableHandle]] + [bulma.icon :refer [icon]] + [bulma.dropdown.views :refer [dropdown]] + [airsonic-ui.helpers :as helpers] + [airsonic-ui.audio.playlist :as playlist] + [airsonic-ui.components.collection.views :as collection] + [airsonic-ui.components.sortable.views :as sortable] + [airsonic-ui.routes :as routes] + + ;; ↓ registers subscription handlers ↓ + [airsonic-ui.components.current-queue.subs])) + +(def SortHandle + (SortableHandle. + ;; Alternative to r/reactify-component, which doens't convert props and hiccup, + ;; is to just provide fn as component and use as-element or create-element + ;; to return React elements from the component. + (fn [] + (r/as-element [:span.is-size-7.has-text-grey-lighter + [icon :elevator]])))) + +(defn song-actions [{:keys [song idx]}] + [dropdown {:items [{:label "Remove from queue" + :event [:audio-player/remove-song idx]} + {:label "Go to source" + :event [:routes/do-navigation (playlist/item-source song)]}]}]) + +(defn artist-link [{id :artistId, artist :artist}] + (if id + [:a {:href (routes/url-for ::routes/artist.detail {:id id})} artist] + artist)) + +(defn song-link [song idx] + [:a + {:href "#" + :on-click (helpers/muted-dispatch [:audio-player/set-current-song idx])} + (:title song)]) + +(defn song-table-head [] + [:thead>tr + [:td.is-narrow] + [:td.song-artist "Artist"] + [:td.song-title "Title"] + [:td.song-duration "Duration"] + [:td.song-actions.is-narrow]]) + +(defn song-table-sortable-tbdoy [{:keys [songs current-song-idx]}] + ;; we need this closure to pass in custom arguments (current-song-idx) + (fn [] + [sortable/sortable-component + {:items songs + :container [:tbody] + :helper-class "sortable-is-moving" + + :render-item + (fn [{[idx song] :value}] + [(if (= idx current-song-idx) :tr.is-playing :tr) + [:td.sortable-handle.is-narrow [:> SortHandle]] + [:td.song-artist [artist-link song]] + [:td.song-title [song-link song idx]] + [:td.song-duration (helpers/format-duration (:duration song) :brief? true)] + [:td.song-actions.is-narrow [song-actions {:song song + :idx idx}]]]) + + :on-sort-end + (fn [{:keys [old-idx new-idx]}] + ;; if we don't dispatch-sync, the UI sometimes places the row back and + ;; resorts it a litle later + (dispatch-sync [:audio-player/move-song old-idx new-idx]))}])) + +(defn song-table [{:keys [songs current-song-idx]}] + [collection/song-table + {:songs songs + :thead song-table-head + :tbody (song-table-sortable-tbdoy {:songs songs + :current-song-idx current-song-idx})}]) + +(defn queue-info [{:keys [playlist-info]}] + [:ul.is-smaller.collection-info + [:li [icon :audio-spectrum] (str (:count playlist-info) + (if (pos? (:count playlist-info)) + " tracks" + " track"))] + [:li [icon :clock] (helpers/format-duration (:duration playlist-info))]]) + +(defn playlist [props] + [:div + [queue-info props] + [song-table {:songs (get-in props [:current-playlist :items]) + :current-song-idx (get-in props [:current-playlist :current-idx])}]]) + +(defn empty-playlist [] + [:p "You are currently not playing anything. Use the search or go to your " + [:a {:href (routes/url-for ::routes/library)} "Library"] " to start playing some music."]) (defn current-queue [] - [:section.section>div.container - [:h1.title "Current Queue"] - (if-let [playlist @(subscribe [:audio/playlist])] - [song/listing (:queue playlist)] - [:p "You are currently not playing anything. Use the search or go to your " - [:a {:href (r/url-for ::r/library)} "Library"] " to start playing some music."])]) + (let [current-playlist @(subscribe [:audio/current-playlist]) + playlist-info @(subscribe [:current-queue/info])] + [:section.section>div.container + [:h1.title "Current Queue"] + (if (empty? current-playlist) + [empty-playlist] + [playlist {:current-playlist current-playlist + :playlist-info playlist-info}])])) diff --git a/src/cljs/airsonic_ui/components/podcast/views.cljs b/src/cljs/airsonic_ui/components/podcast/views.cljs index 07ec8d2..bffbe95 100644 --- a/src/cljs/airsonic_ui/components/podcast/views.cljs +++ b/src/cljs/airsonic_ui/components/podcast/views.cljs @@ -4,7 +4,7 @@ [airsonic-ui.routes :as routes :refer [url-for]] [airsonic-ui.components.podcast.subs :as subs] [airsonic-ui.views.cover :refer [cover card]] - [airsonic-ui.views.icon :refer [icon]] + [bulma.icon :refer [icon]] [airsonic-ui.components.debug.views :refer [debug]])) ;; TODO: Implement detail pages for podcasts diff --git a/src/cljs/airsonic_ui/components/search/views.cljs b/src/cljs/airsonic_ui/components/search/views.cljs index bd3fdeb..4fb1d70 100644 --- a/src/cljs/airsonic_ui/components/search/views.cljs +++ b/src/cljs/airsonic_ui/components/search/views.cljs @@ -2,12 +2,15 @@ (:require [re-frame.core :refer [dispatch subscribe]] [goog.functions :refer [debounce]] [airsonic-ui.routes :as routes :refer [url-for]] - [airsonic-ui.views.song :as song] + [airsonic-ui.helpers :as h] + [airsonic-ui.components.collection.views :as collection] [airsonic-ui.views.cover :refer [card]])) +(def search + (debounce #(dispatch [:search/do-search (.. % -target -value)]) 100)) + (defn form [] - (let [search-term @(subscribe [:search/current-term]) - throttled-search (debounce #(dispatch [:search/do-search (.. % -target -value)]) 100)] + (let [search-term @(subscribe [:search/current-term])] (fn [] [:form {:on-submit #(.preventDefault %)} [:div.feld>p.control @@ -15,7 +18,7 @@ ;; the event might be gone when we the dispatched ;; function is fired, we need to persist it (.persist e) - (throttled-search e)) + (search e)) :default-value search-term :placeholder "Search"}]]]))) @@ -41,8 +44,34 @@ (defn album-results [{:keys [album]}] [result-cards (map (juxt album-url identity) album)]) -(defn song-results [{:keys [song]}] - [song/listing song]) +(defn song-table-thead [] + [:thead + [:td.song-artist "Artist"] + [:td.song-album "Album"] + [:td.song-title "Title"] + [:td.song-duration "Duration"] + [:td.song-actions.is-narrow]]) + +(defn album-link [{id :albumId :as song}] + [:a {:href (routes/url-for ::routes/album.detail {:id id})} (:album song)]) + +(defn song-table-tbody [{:keys [songs current-song]}] + [:tbody + (for [[idx song] (map-indexed vector songs)] + ^{:key idx} + [(if (= (:id song) (:id current-song)) :tr.is-playing :tr) + [:td.song-artist [collection/artist-link song]] + [:td.song-album [album-link song]] + [:td.song-title [collection/song-link {:songs songs + :song song + :idx idx}]] + [:td.song-duration (h/format-duration (:duration song) :brief? true)] + [:td.song-actions.is-narrow [collection/song-actions song]]])]) + +(defn song-results [{songs :song}] + [collection/song-table {:songs songs + :thead song-table-thead + :tbody song-table-tbody}]) (defn results [{:keys [search]}] (let [term @(subscribe [:search/current-term])] diff --git a/src/cljs/airsonic_ui/components/sortable/views.cljs b/src/cljs/airsonic_ui/components/sortable/views.cljs new file mode 100644 index 0000000..0c19d4c --- /dev/null +++ b/src/cljs/airsonic_ui/components/sortable/views.cljs @@ -0,0 +1,98 @@ +(ns airsonic-ui.components.sortable.views + (:require [reagent.core :as r] + [clojure.string :as str] + ["react-sortable-hoc" :refer [SortableHandle SortableElement + SortableContainer]])) +;; this code is taken and adapted from https://github.com/reagent-project/reagent/blob/72c95257c13e5de1531e16d1a06da7686041d3f4/examples/react-sortable-hoc/src/example/core.cljs + +(defn make-wrapper [{:keys [container render-item]}] + (let [SortableItem (SortableElement. + (r/reactify-component render-item))] + (SortableContainer. + (r/reactify-component + (fn [{:keys [items]}] + (into container + (for [[idx value] (map-indexed vector items)] + (r/create-element + SortableItem + #js {:key (str "item-" idx) + :index idx + :value value})))))))) + +(defn style-map + "Returns a map representing all currently set css styles; this makes sense + so we can save a non-updating version of it." + [node] + (let [style (js/window.getComputedStyle node)] + (into {} (keep (fn [idx] + (let [property (.item style idx)] + [property (.getPropertyValue style property)])) + (range (.-length style)))))) + +(defn node-seq + "Returns a seq of all of a node's children" + [node] + (loop [waiting [node] + nodes []] + (if-let [node (first waiting)] + (if-let [children (array-seq (.-children node))] + (recur (concat (rest waiting) children) (conj nodes node)) + (recur (rest waiting) (conj nodes node))) + (rest nodes)))) + +(defn style-snapshot + "Recursively grabs the of all of a node's children" + [node] + (into [] (map style-map (node-seq node)))) + +(defn style-from-map! + "Restores the styling saved in a stylemap" + [style-map node] + (let [style (str/join ";" (map (fn [[k v]] (str k ": " v)) style-map))] + (.setAttribute node "style" style))) + +(defn restore-snapshot + "Recursively restores the styling of all of a nodes children" + [style-snapshot node] + (let [nodes (vec (node-seq node))] + (dotimes [i (count nodes)] + (style-from-map! (nth style-snapshot i) (nth nodes i))))) + +(defonce saved-snapshot (atom nil)) + +(defn sortable-component + "This function allows us to generate sortable components in a reusable way. + It takes a prop-map with several keys: + + - :container A hiccup-vector that will be used as the container + - :items A seq containing the values we want to render and sort + - :render-item Decides how we render each child; will be passed {:value value} + - :on-sort-end Will be called with a map containing :old-idx & :new-idx + - :helper-class Will be appended to the element that's sorted when it's + appended to the body" + [{:keys [container items render-item on-sort-end helper-class]}] + (let [Wrapper (make-wrapper {:container container + :render-item render-item})] + (r/create-element + Wrapper + #js {:items items + :helperClass helper-class + :axis "y" + :lockAxis "y" + + ;; save the style of all of the rows children + :updateBeforeSortStart + (fn [event] + (reset! saved-snapshot (style-snapshot (.-node event)))) + :onSortStart + (fn [_] + ;; the node we get passed as parameter is the original node unfortunately + (restore-snapshot @saved-snapshot (js/document.querySelector "body > :last-child"))) + + ;; update the state to reflect the new order + :onSortEnd + (fn [event] + (on-sort-end {:old-idx (.-oldIndex event) + :new-idx (.-newIndex event)})) + + :useDragHandle true}))) diff --git a/src/cljs/airsonic_ui/helpers.cljs b/src/cljs/airsonic_ui/helpers.cljs index 6ed4015..58f14f6 100644 --- a/src/cljs/airsonic_ui/helpers.cljs +++ b/src/cljs/airsonic_ui/helpers.cljs @@ -4,14 +4,6 @@ [clojure.string :as str]) (:import [goog.string format])) -(defn find-where - "Returns the the first item in `coll` with its index for which `(p song)` - is truthy" - [p coll] - (->> (map-indexed vector coll) - (reduce (fn [_ [idx song]] - (when (p song) (reduced [idx song]))) nil))) - (defn muted-dispatch "Dispatches a re-frame event while canceling default DOM behavior; to be called for example in `:on-click`." diff --git a/src/cljs/airsonic_ui/views.cljs b/src/cljs/airsonic_ui/views.cljs index dbba2ae..0fa72c3 100644 --- a/src/cljs/airsonic_ui/views.cljs +++ b/src/cljs/airsonic_ui/views.cljs @@ -11,7 +11,7 @@ [airsonic-ui.views.notifications :refer [notification-list]] [airsonic-ui.views.breadcrumbs :refer [breadcrumbs]] [airsonic-ui.views.login :refer [login-form]] - [airsonic-ui.views.icon :refer [icon]] + [bulma.icon :refer [icon]] [airsonic-ui.components.about.views :refer [about]] [airsonic-ui.components.artist.views :as artist] diff --git a/src/cljs/airsonic_ui/views/song.cljs b/src/cljs/airsonic_ui/views/song.cljs deleted file mode 100644 index 2133a54..0000000 --- a/src/cljs/airsonic_ui/views/song.cljs +++ /dev/null @@ -1,34 +0,0 @@ -(ns airsonic-ui.views.song - (:require [re-frame.core :refer [subscribe]] - [airsonic-ui.helpers :refer [muted-dispatch format-duration]] - [airsonic-ui.routes :as routes :refer [url-for]] - [airsonic-ui.views.icon :refer [icon]])) - -(defn item [songs song idx] - (let [artist-id (:artistId song) - duration (:duration song)] - [:div - (if artist-id - [:a {:href (url-for ::routes/artist.detail {:id artist-id})} (:artist song)] - (:artist song)) - " - " - [:a - {:href "#" :on-click (muted-dispatch [:audio-player/play-all songs idx] :sync? true)} - (:title song)] - [:span.duration (format-duration duration)]])) - -(defn listing [songs] - (let [current-song @(subscribe [:audio/current-song])] - [:table.table.is-striped.is-hoverable.is-fullwidth.song-list>tbody - (for [[idx song] (map-indexed vector songs)] - (let [tag (if (= (:id song) (:id current-song)) :tr.song.is-playing :tr.song)] - ^{:key idx} [tag - [:td.grow [item songs song idx]] - [:td>a {:title "Play next" - :href "#" - :on-click (muted-dispatch [:audio-player/enqueue-next song])} - [icon :plus]] - [:td>a {:title "Play last" - :href "#" - :on-click (muted-dispatch [:audio-player/enqueue-last song])} - [icon :caret-right]]]))])) diff --git a/src/cljs/bulma/dropdown/events.cljs b/src/cljs/bulma/dropdown/events.cljs new file mode 100644 index 0000000..7b5f0aa --- /dev/null +++ b/src/cljs/bulma/dropdown/events.cljs @@ -0,0 +1,20 @@ +(ns bulma.dropdown.events + (:require [re-frame.core :as rf])) + +(defn show-dropdown [db [_ dropdown-id]] + (assoc-in db [:bulma :visible-dropdown] dropdown-id)) + +(rf/reg-event-db ::show show-dropdown) + +(defn hide-dropdown [db _] + (update db :bulma dissoc :visible-dropdown)) + +(rf/reg-event-db ::hide hide-dropdown) + +(defn toggle-dropdown [db [_ dropdown-id]] + (let [visible-dropdown (get-in db [:bulma :visible-dropdown])] + (if (= visible-dropdown dropdown-id) + (hide-dropdown db [::hide]) + (show-dropdown db [::show dropdown-id])))) + +(rf/reg-event-db ::toggle toggle-dropdown) diff --git a/src/cljs/bulma/dropdown/subs.cljs b/src/cljs/bulma/dropdown/subs.cljs new file mode 100644 index 0000000..cdeab23 --- /dev/null +++ b/src/cljs/bulma/dropdown/subs.cljs @@ -0,0 +1,22 @@ +(ns bulma.dropdown.subs + (:require [re-frame.core :as rf])) + +;; NOTE: This is almost the same as bulma.modal.subs +;; Maybe we can provide some abstraction that covers both, but maybe we shouldn't + +(defn visible-dropdown + "Gives us the ID of the currently visible dropdown" + [db _] + (get-in db [:bulma :visible-dropdown])) + +(rf/reg-sub ::visible-dropdown visible-dropdown) + +(defn visible? + "Predicate to check the visibility of a single modal" + [visible-dropdown [_ dropdown-id]] + (= visible-dropdown dropdown-id)) + +(rf/reg-sub + ::visible? + :<- [::visible-dropdown] + visible?) diff --git a/src/cljs/bulma/dropdown/views.cljs b/src/cljs/bulma/dropdown/views.cljs new file mode 100644 index 0000000..a697350 --- /dev/null +++ b/src/cljs/bulma/dropdown/views.cljs @@ -0,0 +1,43 @@ +(ns bulma.dropdown.views + (:require [re-frame.core :refer [dispatch subscribe]] + [reagent.core :as r] + [bulma.icon :refer [icon]] + [bulma.dropdown.events :as ev] + [bulma.dropdown.subs :as sub])) + +(defn choose-action [event-vector] + (fn [e] + (.preventDefault e) + (dispatch [::ev/hide]) + (dispatch event-vector))) + +(defn generate-id [] + (str "bulma-dropdown-" (random-uuid))) + +(defn click-overlay + [] + [:div {:style {:position "fixed" + :z-index 19 ;; <- 20 is the z-index of .dropdown-menu + :top 0 + :left 0 + :bottom 0 + :right 0} + :on-click #(dispatch [::ev/hide])}]) + +(defn dropdown [{:keys [items]}] + (let [dropdown-id (generate-id)] + (fn [] + (let [visible? @(subscribe [::sub/visible? dropdown-id])] + [(if visible? :div.dropdown.is-right.is-active :div.dropdown.is-right) + (when visible? [click-overlay]) + [:div.dropdown-trigger + [:span.is-small.button {:aria-haspopup "true" + :aria-controls dropdown-id + :on-click #(dispatch [::ev/toggle dropdown-id])} + [icon :ellipses]]] + [:div.dropdown-menu {:id dropdown-id, :role "menu"} + [:div.dropdown-content + (for [[idx {:keys [label event]}] (map-indexed vector items)] + ^{:key (str dropdown-id "-" idx)} + [:a.dropdown-item {:href "#" + :on-click (choose-action event)} label])]]])))) diff --git a/src/cljs/airsonic_ui/views/icon.cljs b/src/cljs/bulma/icon.cljs similarity index 50% rename from src/cljs/airsonic_ui/views/icon.cljs rename to src/cljs/bulma/icon.cljs index b7c9ee4..caec714 100644 --- a/src/cljs/airsonic_ui/views/icon.cljs +++ b/src/cljs/bulma/icon.cljs @@ -1,4 +1,4 @@ -(ns airsonic-ui.views.icon) +(ns bulma.icon) -(defn icon [glyph & extra] +(defn icon [glyph] [:span.icon [:span.oi {:data-glyph (name glyph)}]]) diff --git a/src/sass/app.sass b/src/sass/app.sass index 60b8fd7..2e8336a 100644 --- a/src/sass/app.sass +++ b/src/sass/app.sass @@ -218,18 +218,6 @@ .grow width: 100% -// duh -.song-list - .song - .duration - padding-left: .5rem - color: $grey-light - font-weight: normal - - &.is-playing - background-color: $light !important - font-weight: bold - // useful in general to pull elements closer together; bulma es very generous // with whitespace .section @@ -302,17 +290,6 @@ margin-right: 1rem margin-bottom: 0 - .collection-info - list-style: none - - li - display: inline-block - margin-left: 0.75rem - - &:first-child - margin-left: 0 - - .song-list counter-reset: track @@ -326,3 +303,54 @@ font-weight: normal display: inline padding-right: 0.375rem + +.collection-info + list-style: none + + li + display: inline-block + margin-left: 0.75rem + + &:first-child + margin-left: 0 + +.song-listing-table + tr.is-playing + background-color: $table-row-active-background-color + color: $table-row-active-color + + a, strong, td.song-duration, td.sort-handle span + color: currentColor + + span.button, div.dropdown + color: $table-color + + td + &.is-narrow + white-space: nowrap + + &.song-duration + text-align: right + + &.sortable-handle + -webkit-touch-callout: none + user-select: none + + tbody .song-duration + color: $grey-light + + tr:hover + .button + +// drag'n'drop +.sortable-handle + span + cursor: grabbing + user-select: none + +tr.sortable-is-moving.is-playing + background-color: $table-row-active-background-color + color: $table-row-active-color + + a, strong, td.song-duration, td.sort-handle span + color: currentColor diff --git a/test/cljs/airsonic_ui/audio/core_test.cljs b/test/cljs/airsonic_ui/audio/core_test.cljs index 4866db4..99f72fd 100644 --- a/test/cljs/airsonic_ui/audio/core_test.cljs +++ b/test/cljs/airsonic_ui/audio/core_test.cljs @@ -1,17 +1,20 @@ (ns airsonic-ui.audio.core-test (:require [airsonic-ui.audio.core :as audio] - [airsonic-ui.audio.playlist-test :as p] - [airsonic-ui.fixtures :as fixtures] + #_[airsonic-ui.audio.playlist-test :as p] + #_[airsonic-ui.fixtures :as fixtures] [cljs.test :refer [deftest testing is]])) (enable-console-print!) (deftest current-song-subscription - (letfn [(current-song [db] + ;; NOTE: Should the subscription be moved to the playlist.cljs? + #_(testing "Should provide information about the song" + (letfn [(current-song [db] (-> (audio/summary db [:audio/summary]) (audio/current-song [:audio/current-song])))] - (testing "Should provide information about the song" - (= fixtures/song (current-song p/fixture))))) + (= fixtures/song (current-song p/fixture)))) + (testing "Should work fine when no song is playing" + (is (nil? (audio/current-song nil [:audio/current-song]))))) (deftest playback-status-subscription (letfn [(is-playing? [playback-status] diff --git a/test/cljs/airsonic_ui/audio/playlist_test.cljs b/test/cljs/airsonic_ui/audio/playlist_test.cljs index e48ae14..c640d63 100644 --- a/test/cljs/airsonic_ui/audio/playlist_test.cljs +++ b/test/cljs/airsonic_ui/audio/playlist_test.cljs @@ -1,29 +1,12 @@ (ns airsonic-ui.audio.playlist-test (:require [cljs.test :refer [deftest testing is]] [airsonic-ui.audio.playlist :as playlist] - [airsonic-ui.helpers :refer [find-where]] [airsonic-ui.fixtures :as fixtures] - [airsonic-ui.test-helpers :as helpers] - [debux.cs.core :refer-macros [dbg]])) + [airsonic-ui.test-helpers :refer [song song-queue]] + #_[debux.cs.core :refer-macros [dbg]])) (enable-console-print!) -(defn- song [] - (hash-map :id (rand-int 9999) - :coverArt (rand-int 9999) - :year (+ 1900 (rand-int 118)) - :artist (helpers/rand-str) - :artistId (rand-int 100000) - :title (helpers/rand-str) - :album (helpers/rand-str))) - -(defn- song-queue - "Generates a seq of n different songs" - [n] - (let [r-int (atom 0)] - (with-redefs [rand-int #(mod (swap! r-int inc) %1)] - (repeatedly n song)))) - (def fixture {:audio {:current-song fixtures/song :playlist (song-queue 20) @@ -33,14 +16,22 @@ (deftest playlist-creation (testing "Playlist creation" - (testing "should give us the correct current song" + (testing "should give us the correct current song for linear playback-mode" (let [queue (song-queue 10)] - (doseq [playback-mode [:linear :shuffled] - repeat-mode [:repeat-none :repeat-single :repeat-all]] + (doseq [repeat-mode [:repeat-none :repeat-single :repeat-all]] (is (same-song? (first queue) - (-> (playlist/->playlist queue :playback-mode playback-mode :repeat-mode repeat-mode) - (playlist/peek))) - (str playback-mode ", " repeat-mode))))) + (-> (playlist/->playlist queue :playback-mode :linear :repeat-mode repeat-mode) + (playlist/current-song))) + (str "repeat-mode: " repeat-mode))))) + + (testing "any current song for shuffled playback mode" + (let [queue (song-queue 10)] + (doseq [repeat-mode [:repeat-none :repeat-single :repeat-all]] + (is (some? ((set queue) + (-> (playlist/->playlist queue :playback-mode :linear :repeat-mode repeat-mode) + (playlist/current-song)))) + (str "repeat-mode: " repeat-mode))))) + (testing "should give us a playlist with the correct number of tracks" (let [queue (song-queue 100)] (doseq [playback-mode [:linear :shuffled] @@ -55,21 +46,31 @@ (let [queue (song-queue 10) linear (playlist/->playlist queue :playback-mode :linear :repeat-mode :repeat-none) shuffled (playlist/set-playback-mode linear :shuffled)] + (testing "should indicate the new playback mode" + (is (= :linear (:playback-mode linear))) + (is (= :shuffled (:playback-mode shuffled)))) (testing "should re-order the tracks" - (is (not= (map :playlist/order (:queue shuffled)) (map :playlist/order (:queue linear))))) + (is (not= (:items shuffled) (:items linear)))) (testing "should not change the currently playing track" - (is (same-song? (playlist/peek linear) (playlist/peek shuffled)))) + (is (same-song? (playlist/current-song linear) (playlist/current-song shuffled)))) (testing "should not change the repeat mode" (is (= (:repeat-mode shuffled) (:repeat-mode linear)))))) (testing "from shuffled to linear" (let [queue (song-queue 10) shuffled (playlist/->playlist queue :playback-mode :shuffled :repeat-mode :repeat-none) linear (playlist/set-playback-mode shuffled :linear)] + (testing "should indicate the new playback mode" + (is (= :linear (:playback-mode linear))) + (is (= :shuffled (:playback-mode shuffled)))) (testing "should set the correct order for tracks" - (is (every? #(apply same-song? %) (interleave queue (:queue linear)))) - (is (< (:playlist/order (first (:queue linear))) (:playlist/order (last (:queue linear)))))) + (let [linear-order (comp :playlist/linear-order meta)] + (is (every? #(apply same-song? %) (interleave queue (vals (:items linear))))) + ;; every song should have a smaller order than its successor + (is (->> (map linear-order (vals (:items linear))) + (partition 2 1) + (every? (fn [[a b]] (< a b))))))) (testing "should not change the currently playing track" - (is (same-song? (playlist/peek linear) (playlist/peek shuffled)))) + (is (same-song? (playlist/current-song linear) (playlist/current-song shuffled)))) (testing "should not change the repeat mode" (is (= (:repeat-mode shuffled) (:repeat-mode linear)))))))) @@ -91,17 +92,18 @@ (let [queue (song-queue 5) playlist (playlist/->playlist queue :playback-mode :linear :repeat-mode repeat-mode)] (is (same-song? (nth queue 1) (-> (playlist/next-song playlist) - (playlist/peek))) + (playlist/current-song))) (str repeat-mode ", skipped once")) (is (same-song? (nth queue 2) (-> (playlist/next-song playlist) (playlist/next-song) - (playlist/peek))) + (playlist/current-song))) (str repeat-mode ", skipped twice"))))) + ;; TODO: Write this test (testing "Should go back to the first song when repeat-mode is all and we played the last song") (testing "Should always give the same track when repeat-mode is single" (let [queue (song-queue 3) playlist (playlist/->playlist queue :playback-mode :linear :repeat-mode :repeat-single) - played-back (map playlist/peek (iterate playlist/next-song playlist))] + played-back (map playlist/current-song (iterate playlist/next-song playlist))] (is (same-song? (first queue) (nth played-back 0))) (is (same-song? (first queue) (nth played-back 1))) (is (same-song? (first queue) (nth played-back 2))) @@ -110,7 +112,7 @@ (is (nil? (-> (song-queue 1) (playlist/->playlist :playback-mode :linear :repeat-mode :repeat-none) (playlist/next-song) - (playlist/peek)))))) + (playlist/current-song)))))) (deftest shuffled-next-song (testing "Should play every track once when called for the entire queue" @@ -118,35 +120,34 @@ (let [length 10 playlist (playlist/->playlist (song-queue length) :playback-mode :shuffled :repeat-mode repeat-mode) played-tracks (->> (iterate playlist/next-song playlist) - (map playlist/peek) + (map playlist/current-song) (take length))] (is (= (count played-tracks) (count (set played-tracks))) (str repeat-mode))))) - (testing "Should re-shuffle the playlist when wrapping around and repeat-mode is all" + (testing "Should keep the song order when wrapping around and repeat-mode is all" (let [playlist (playlist/->playlist (song-queue 100) :playback-mode :shuffled :repeat-mode :repeat-all) - [last-idx _] (find-where #(= (:playlist/order %) 99) (:queue playlist))] - (is (not= (map :playlist/order (:queue playlist)) - (map :playlist/order (:queue (-> (playlist/set-current-song playlist last-idx) - (playlist/next-song)))))))) + next-playlist (-> (playlist/set-current-song playlist 99) + (playlist/next-song))] + (= (playlist/current-song playlist) + (playlist/current-song next-playlist)))) + (testing "Should always give the same track when repeat-mode is single" - (let [queue (song-queue 3) - playlist (playlist/->playlist queue :playback-mode :shuffled :repeat-mode :repeat-single) - played-back (map playlist/peek (iterate playlist/next-song playlist))] - (is (same-song? (first queue) (nth played-back 0))) - (is (same-song? (first queue) (nth played-back 1))) - (is (same-song? (first queue) (nth played-back 2))) - (is (same-song? (first queue) (nth played-back 3)) "wrapping around"))) + (let [playlist (playlist/->playlist (song-queue 10) :playback-mode :shuffled :repeat-mode :repeat-single) + played-back (map playlist/current-song (iterate playlist/next-song playlist))] + (dotimes [i 3] + (is (same-song? (nth played-back i) (nth played-back (inc i))))))) + (testing "Should stop playing at the end of the queue when repeat-mode is none" (is (nil? (-> (song-queue 1) (playlist/->playlist :playback-mode :linear :repeat-mode :repeat-none) (playlist/next-song) - (playlist/peek)))))) + (playlist/current-song)))))) (deftest linear-previous-song (testing "Should always give the same track when repeat-mode is single" (let [queue (song-queue 3) playlist (playlist/->playlist queue :playback-mode :linear :repeat-mode :repeat-single) - played-back (map playlist/peek (iterate playlist/next-song playlist))] + played-back (map playlist/current-song (iterate playlist/next-song playlist))] (is (same-song? (first queue) (nth played-back 0))) (is (same-song? (first queue) (nth played-back 1))) (is (same-song? (first queue) (nth played-back 2))) @@ -158,61 +159,61 @@ (is (same-song? (nth queue 1) (-> (playlist/next-song playlist) (playlist/next-song) (playlist/previous-song) - (playlist/peek))))))) - (testing "Should repeatedly give the first song when repeat-mode is none" + (playlist/current-song))))))) + ;; TODO: Should it? + #_(testing "Should repeatedly give the first song when repeat-mode is none" (let [queue (song-queue 3) playlist (playlist/->playlist queue :playback-mode :linear :repeat-mode :repeat-none)] (is (same-song? (first queue) (-> (playlist/previous-song playlist) - (playlist/peek)))))) + (playlist/current-song)))))) (testing "Should wrap around to last song when repeat-mode is all" (let [queue (song-queue 3) playlist (playlist/->playlist queue :playback-mode :linear :repeat-mode :repeat-all)] (is (same-song? (last queue) (-> (playlist/previous-song playlist) - (playlist/peek))))))) + (playlist/current-song))))))) (deftest shuffled-previous-song (with-redefs [shuffle reverse] (testing "Should always give the same track when repeat-mode is single" (let [queue (song-queue 3) playlist (playlist/->playlist queue :playback-mode :shuffled :repeat-mode :repeat-single) - played-back (map playlist/peek (iterate playlist/next-song playlist))] - (is (same-song? (first queue) (nth played-back 0))) - (is (same-song? (first queue) (nth played-back 1))) - (is (same-song? (first queue) (nth played-back 2))) - (is (same-song? (first queue) (nth played-back 3)) "wrapping around"))) + played-back (map playlist/current-song (iterate playlist/next-song playlist))] + (dotimes [i 3] + (is (same-song? (nth played-back i) (nth played-back (inc i))))))) (testing "Should keep the playing order when repeat-mode is not single" (doseq [repeat-mode '(:repeat-none :repeat-all)] (let [queue (song-queue 3) playlist (playlist/->playlist queue :playback-mode :shuffled :repeat-mode repeat-mode)] - (is (same-song? (playlist/peek playlist) + (is (same-song? (playlist/current-song playlist) (-> playlist (playlist/next-song) (playlist/previous-song) - (playlist/peek))) + (playlist/current-song))) (str "for repeat mode " repeat-mode)) (is (same-song? (-> (playlist/next-song playlist) - (playlist/peek)) + (playlist/current-song)) (-> (playlist/next-song playlist) (playlist/next-song) (playlist/previous-song) - (playlist/peek))) + (playlist/current-song))) (str "for repeat mode " repeat-mode))))) - (testing "Should re-shuffle when repeat-mode is all and we go back to before the first track" - (let [playlist (with-redefs [shuffle identity] - (playlist/->playlist (song-queue 10) :playback-mode :shuffled :repeat-mode :repeat-all)) - playlist' (with-redefs [shuffle reverse] - (playlist/previous-song playlist))] - (is (not= (map :playlist/order (:queue playlist)) (map :playlist/order (:queue playlist')))))))) + (testing "Should keep the song order when repeat-mode is all and we go back to before the first track" + (let [playlist (playlist/->playlist (song-queue 10) :playback-mode :shuffled :repeat-mode :repeat-all) + next-playlist (-> (playlist/previous-song playlist) + (playlist/set-current-song 0))] + (is (= (playlist/current-song playlist) + (playlist/current-song next-playlist))))))) (deftest set-current-song (testing "Should correctly set the new song" - (let [queue (song-queue 3) - playlist (playlist/->playlist queue :playback-mode :shuffled :repeat-mode :repeat-single) - current-track (first queue) - next-track (-> (playlist/set-current-song playlist 1) - (playlist/peek))] - (is (not (nil? next-track))) - (is (not (same-song? current-track next-track)))))) + (doseq [repeat-mode [:repeat-all :repeat-none]] + (let [queue (song-queue 3) + playlist (playlist/->playlist queue :playback-mode :shuffled :repeat-mode repeat-mode) + next-track (-> (playlist/set-current-song playlist 1) + (playlist/current-song))] + (is (not (nil? next-track))) + (is (not (same-song? (playlist/current-song playlist) + next-track))))))) (deftest enqueue-last (testing "Should make sure the song is played last" @@ -223,12 +224,12 @@ (playlist/->playlist queue :playback-mode playback-mode :repeat-mode repeat-mode)) played-back (->> (iterate playlist/next-song playlist) (take (dec length)) - (map #(:id (playlist/peek %))) + (map #(:id (playlist/current-song %))) (set)) to-enqueue (song) playlist' (playlist/enqueue-last playlist to-enqueue)] (is (nil? (played-back (-> (->> (iterate playlist/next-song playlist') - (map playlist/peek)) + (map playlist/current-song)) (nth length) (:id)))) (str "for " playback-mode ", " repeat-mode))))) @@ -240,7 +241,7 @@ played-back-songs (fn played-back-songs [playlist] (->> (iterate playlist/next-song playlist) (take length) - (map playlist/peek) + (map playlist/current-song) (map :playlist/order))) played-back (played-back-songs playlist) played-back' (played-back-songs (playlist/enqueue-last playlist (song)))] @@ -249,11 +250,92 @@ (deftest enqueue-next (testing "Should play the song after the currently playing song" - (doseq [playback-mode '(:linear :shuffled) - repeat-mode '(:repeat-none :repeat-all)] + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all]] (let [length 5, queue (song-queue length) playlist (playlist/->playlist queue :playback-mode playback-mode :repeat-mode repeat-mode) next-song (song)] (is (same-song? next-song (-> (playlist/enqueue-next playlist next-song) (playlist/next-song) - (playlist/peek)))))))) + (playlist/current-song)))))))) + +(deftest move-track + (testing "Should correctly set the new order" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs) + playlist (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode)] + (is (same-song? (-> (playlist/next-song playlist) + (playlist/next-song) + (playlist/current-song)) + (-> (playlist/move-song playlist 2 1) + (playlist/next-song) + (playlist/current-song))))))) + (testing "Should update the currently playing track's index" + (testing "when inserting a track before" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs) + playlist (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode)] + (is (= 4 (-> (playlist/set-current-song playlist 3) + (playlist/move-song 5 3) + :current-idx)))))) + (testing "when moving a track behind it" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs) + playlist (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode)] + (is (= 2 (-> (playlist/set-current-song playlist 3) + (playlist/move-song 2 5) + :current-idx)))))) + (testing "when moving it" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs) + playlist (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode)] + (is (= 7 (-> (playlist/set-current-song playlist 3) + (playlist/move-song 3 7) + :current-idx)))))) + (testing "when the current track is outside of the modified range" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs) + playlist (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode)] + (is (= 3 (-> (playlist/set-current-song playlist 3) + (playlist/move-song 4 7) + :current-idx)))))))) + +(deftest remove-song + (with-redefs [shuffle identity] + (testing "Should remove a single song from the playlist" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs) + playlist (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode) + first-removed (playlist/remove-song playlist 0) + middle-removed (playlist/remove-song playlist 5) + last-removed (playlist/remove-song playlist 9) + song-not-in-list? (fn [song playlist] + (every? #(not (same-song? % song)) + (vals (:items playlist))))] + (is (= 9 (count first-removed) (count middle-removed) (count last-removed))) + (is (song-not-in-list? (first queue) first-removed)) + (is (same-song? (second queue) (get (:items first-removed) 0))) + (is (song-not-in-list? (nth queue 5) middle-removed)) + (is (same-song? (nth queue 6) (get (:items middle-removed) 5))) + (is (song-not-in-list? (last queue) last-removed))))) + (testing "Should pause if the currently playing song is removed" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-all :repeat-single]] + (let [n-songs 10 + queue (song-queue n-songs)] + (is (nil? (-> (playlist/->playlist queue :repeat-mode repeat-mode :playback-mode playback-mode) + (playlist/set-current-song 5) + (playlist/remove-song 5) + (playlist/current-song))))))))) diff --git a/test/cljs/airsonic_ui/components/audio_player/events_test.cljs b/test/cljs/airsonic_ui/components/audio_player/events_test.cljs index 4225bc3..a4e0e20 100644 --- a/test/cljs/airsonic_ui/components/audio_player/events_test.cljs +++ b/test/cljs/airsonic_ui/components/audio_player/events_test.cljs @@ -1,10 +1,39 @@ (ns airsonic-ui.components.audio-player.events-test (:require [cljs.test :refer-macros [deftest testing is]] - [airsonic-ui.test-helpers :refer [dispatches?]] + [airsonic-ui.audio.core :as audio] + [airsonic-ui.audio.playlist :as playlist] + [airsonic-ui.fixtures :as fixtures] + [airsonic-ui.test-helpers :refer [dispatches? song-queue]] [airsonic-ui.components.audio-player.events :as events])) - (deftest song-has-ended (testing "Should play the next song when current song has ended" (is (not (dispatches? (events/audio-update {} [:audio/update {:ended? false}]) :audio-player/next-song))) (is (dispatches? (events/audio-update {} [:audio/update {:ended? true}]) :audio-player/next-song)))) + +(deftest changing-current-song + (testing "Should correctly set the current song index" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-single :repeat-all]] + (let [n-songs 100 + next-idx (rand-int n-songs) + fixture {:db {:credentials fixtures/credentials + :audio {:current-playlist (playlist/->playlist (song-queue n-songs) :playback-mode playback-mode :repeat-mode repeat-mode)}}} + effects (events/set-current-song fixture [:audio/set-current-song next-idx])] + (is (= next-idx + (-> (:db effects) + (audio/summary [:audio/summary]) + (audio/current-playlist [:audio/current-playlist]) + (:current-idx))) + (str "for playback-mode " playback-mode " and repeat-mode " repeat-mode)) + (is (contains? effects :audio/play)))))) + +(deftest removing-currently-playing-song + (testing "Should stop all audio when removing the currently playing song" + (doseq [playback-mode [:linear :shuffled] + repeat-mode [:repeat-none :repeat-single :repeat-all]] + (let [n-songs 100 + fixture {:db {:credentials fixtures/credentials + :audio {:current-playlist (playlist/->playlist (song-queue n-songs) :playback-mode playback-mode :repeat-mode repeat-mode)}}}] + (is (contains? (events/remove-song fixture [:audio/remove-song 0]) :audio/stop)) + (is (not (contains? (events/remove-song fixture [:audio/remove-song 99]) :audio/stop))))))) diff --git a/test/cljs/airsonic_ui/helpers_test.cljs b/test/cljs/airsonic_ui/helpers_test.cljs index 42b2212..6b6023f 100644 --- a/test/cljs/airsonic_ui/helpers_test.cljs +++ b/test/cljs/airsonic_ui/helpers_test.cljs @@ -2,17 +2,6 @@ (:require [cljs.test :refer [deftest testing is]] [airsonic-ui.helpers :as helpers])) -(deftest find-where - (testing "Finds the correct item and index" - (is (= [0 1] (helpers/find-where (partial = 1) (range 1 10)))) - (is (= [2 {:foo true, :bar false}] (helpers/find-where :foo '({} - {:foo false - :bar true} - {:foo true - :bar false}))))) - (testing "Returns nil when nothing is found" - (is (nil? (helpers/find-where (partial = 2) (range 2)))))) - (deftest add-classes (testing "Should add classes to a simple hiccup keyword" (is (= :div.foo (helpers/add-classes :div :foo))) diff --git a/test/cljs/airsonic_ui/test_helpers.cljs b/test/cljs/airsonic_ui/test_helpers.cljs index 1762f47..a03b67e 100644 --- a/test/cljs/airsonic_ui/test_helpers.cljs +++ b/test/cljs/airsonic_ui/test_helpers.cljs @@ -17,3 +17,19 @@ (from arr #(-> (str 0 (.toString % 16)) (.substr -2))) (join ""))))) + +(defn song [] + (hash-map :id (rand-int 9999) + :coverArt (rand-int 9999) + :year (+ 1900 (rand-int 118)) + :artist (rand-str) + :artistId (rand-int 100000) + :title (rand-str) + :album (rand-str))) + +(defn song-queue + "Generates a seq of n different songs" + [n] + (let [r-int (atom 0)] + (with-redefs [rand-int #(mod (swap! r-int inc) %1)] + (repeatedly n song)))) diff --git a/test/cljs/bulma/dropdown_test.cljs b/test/cljs/bulma/dropdown_test.cljs new file mode 100644 index 0000000..005daa1 --- /dev/null +++ b/test/cljs/bulma/dropdown_test.cljs @@ -0,0 +1,40 @@ +(ns bulma.dropdown-test + (:require [cljs.test :refer-macros [deftest testing is]] + [bulma.dropdown.subs :as sub] + [bulma.dropdown.events :as ev])) + +;; NOTE: Here as well; this code is very much like the modal code +;; Not sure whether to explicitly duplicate it or provide some smarter +;; abstraction that's harder to understand at first sight + +(enable-console-print!) + +(deftest bulma-dropdowns + (testing "Should create a collection of dropdowns if there is none" + (let [new-db (ev/show-dropdown {} [::ev/show :some-dropdown-id])] + (is (= :some-dropdown-id (sub/visible-dropdown new-db [::sub/visible-dropdown]))))) + (testing "Should hide other dropdowns when displaying a new one" + (let [dropdown-ids [:some-id-1 :some-id-2 :some-id-3] + new-db (reduce (fn [db dropdown-id] + (ev/show-dropdown db [::ev/show dropdown-id])) + {} dropdown-ids)] + (is (= :some-id-3 (sub/visible-dropdown new-db [::sub/visible-dropdown]))))) + (testing "Should remove a dropdown from the collection when we hide it" + (let [dropdown-ids [:some-id-1 :some-id-2 :some-id-3] + new-db (-> (reduce (fn [db dropdown-id] + (ev/show-dropdown db [::ev/show dropdown-id])) + {} dropdown-ids) + (ev/hide-dropdown [::ev/hide]))] + (is (not (some? (sub/visible-dropdown new-db [::sub/visible-dropdown])))))) + (testing "Should tell us about the visibility of a dropdown with a predicate" + (is (true? (-> (ev/show-dropdown {} [::ev/show :getting-repetitive]) + (sub/visible-dropdown [::sub/visible-dropdown]) + (sub/visible? [::sub/visible? :getting-repetitive]))))) + (testing "Dropdown toggling" + (is (true? (-> (ev/toggle-dropdown {} [::ev/toggle :some-generic-dropdown]) + (sub/visible-dropdown [::sub/visible-dropdown]) + (sub/visible? [::sub/visible? :some-generic-dropdown])))) + (is (not (true? (-> (ev/toggle-dropdown {} [::ev/toggle :some-generic-dropdown]) + (ev/toggle-dropdown [::ev/toggle :some-generic-dropdown]) + (sub/visible-dropdown [::sub/visible-dropdown]) + (sub/visible? [::sub/visible? :some-generic-dropdown])))))))