From e2e606cb8a09485f88a6e3cd05dbcb994bd54ab5 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:19:38 -0700 Subject: [PATCH 01/12] updated for idea --- .gitignore | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.gitignore b/.gitignore index c2658d7..c5b7a00 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,4 @@ node_modules/ +.idea/ +yarn.lock + From 2845a851460fd1ee0949ea4cd0daf00f6290c57c Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:20:05 -0700 Subject: [PATCH 02/12] drop package-lock.json --- .gitignore | 1 + package-lock.json | 4084 --------------------------------------------- 2 files changed, 1 insertion(+), 4084 deletions(-) delete mode 100644 package-lock.json diff --git a/.gitignore b/.gitignore index c5b7a00..065b366 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,5 @@ node_modules/ .idea/ yarn.lock +package-lock.json diff --git a/package-lock.json b/package-lock.json deleted file mode 100644 index a138cb2..0000000 --- a/package-lock.json +++ /dev/null @@ -1,4084 +0,0 @@ -{ - "name": "rote", - "version": "0.3.0", - "lockfileVersion": 1, - "requires": true, - "dependencies": { - "@webassemblyjs/ast": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.8.5.tgz", - "integrity": "sha512-aJMfngIZ65+t71C3y2nBBg5FFG0Okt9m0XEgWZ7Ywgn1oMAT8cNwx00Uv1cQyHtidq0Xn94R4TAywO+LCQ+ZAQ==", - "dev": true, - "requires": { - "@webassemblyjs/helper-module-context": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/wast-parser": "1.8.5" - } - }, - "@webassemblyjs/floating-point-hex-parser": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.8.5.tgz", - "integrity": "sha512-9p+79WHru1oqBh9ewP9zW95E3XAo+90oth7S5Re3eQnECGq59ly1Ri5tsIipKGpiStHsUYmY3zMLqtk3gTcOtQ==", - "dev": true - }, - "@webassemblyjs/helper-api-error": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.8.5.tgz", - "integrity": "sha512-Za/tnzsvnqdaSPOUXHyKJ2XI7PDX64kWtURyGiJJZKVEdFOsdKUCPTNEVFZq3zJ2R0G5wc2PZ5gvdTRFgm81zA==", - "dev": true - }, - "@webassemblyjs/helper-buffer": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.8.5.tgz", - "integrity": "sha512-Ri2R8nOS0U6G49Q86goFIPNgjyl6+oE1abW1pS84BuhP1Qcr5JqMwRFT3Ah3ADDDYGEgGs1iyb1DGX+kAi/c/Q==", - "dev": true - }, - "@webassemblyjs/helper-code-frame": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.8.5.tgz", - "integrity": "sha512-VQAadSubZIhNpH46IR3yWO4kZZjMxN1opDrzePLdVKAZ+DFjkGD/rf4v1jap744uPVU6yjL/smZbRIIJTOUnKQ==", - "dev": true, - "requires": { - "@webassemblyjs/wast-printer": "1.8.5" - } - }, - "@webassemblyjs/helper-fsm": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-fsm/-/helper-fsm-1.8.5.tgz", - "integrity": "sha512-kRuX/saORcg8se/ft6Q2UbRpZwP4y7YrWsLXPbbmtepKr22i8Z4O3V5QE9DbZK908dh5Xya4Un57SDIKwB9eow==", - "dev": true - }, - "@webassemblyjs/helper-module-context": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-module-context/-/helper-module-context-1.8.5.tgz", - "integrity": "sha512-/O1B236mN7UNEU4t9X7Pj38i4VoU8CcMHyy3l2cV/kIF4U5KoHXDVqcDuOs1ltkac90IM4vZdHc52t1x8Yfs3g==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "mamacro": "^0.0.3" - } - }, - "@webassemblyjs/helper-wasm-bytecode": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.8.5.tgz", - "integrity": "sha512-Cu4YMYG3Ddl72CbmpjU/wbP6SACcOPVbHN1dI4VJNJVgFwaKf1ppeFJrwydOG3NDHxVGuCfPlLZNyEdIYlQ6QQ==", - "dev": true - }, - "@webassemblyjs/helper-wasm-section": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.8.5.tgz", - "integrity": "sha512-VV083zwR+VTrIWWtgIUpqfvVdK4ff38loRmrdDBgBT8ADXYsEZ5mPQ4Nde90N3UYatHdYoDIFb7oHzMncI02tA==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-buffer": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/wasm-gen": "1.8.5" - } - }, - "@webassemblyjs/ieee754": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.8.5.tgz", - "integrity": "sha512-aaCvQYrvKbY/n6wKHb/ylAJr27GglahUO89CcGXMItrOBqRarUMxWLJgxm9PJNuKULwN5n1csT9bYoMeZOGF3g==", - "dev": true, - "requires": { - "@xtuc/ieee754": "^1.2.0" - } - }, - "@webassemblyjs/leb128": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.8.5.tgz", - "integrity": "sha512-plYUuUwleLIziknvlP8VpTgO4kqNaH57Y3JnNa6DLpu/sGcP6hbVdfdX5aHAV716pQBKrfuU26BJK29qY37J7A==", - "dev": true, - "requires": { - "@xtuc/long": "4.2.2" - } - }, - "@webassemblyjs/utf8": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.8.5.tgz", - "integrity": "sha512-U7zgftmQriw37tfD934UNInokz6yTmn29inT2cAetAsaU9YeVCveWEwhKL1Mg4yS7q//NGdzy79nlXh3bT8Kjw==", - "dev": true - }, - "@webassemblyjs/wasm-edit": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.8.5.tgz", - "integrity": "sha512-A41EMy8MWw5yvqj7MQzkDjU29K7UJq1VrX2vWLzfpRHt3ISftOXqrtojn7nlPsZ9Ijhp5NwuODuycSvfAO/26Q==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-buffer": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/helper-wasm-section": "1.8.5", - "@webassemblyjs/wasm-gen": "1.8.5", - "@webassemblyjs/wasm-opt": "1.8.5", - "@webassemblyjs/wasm-parser": "1.8.5", - "@webassemblyjs/wast-printer": "1.8.5" - } - }, - "@webassemblyjs/wasm-gen": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.8.5.tgz", - "integrity": "sha512-BCZBT0LURC0CXDzj5FXSc2FPTsxwp3nWcqXQdOZE4U7h7i8FqtFK5Egia6f9raQLpEKT1VL7zr4r3+QX6zArWg==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/ieee754": "1.8.5", - "@webassemblyjs/leb128": "1.8.5", - "@webassemblyjs/utf8": "1.8.5" - } - }, - "@webassemblyjs/wasm-opt": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.8.5.tgz", - "integrity": "sha512-HKo2mO/Uh9A6ojzu7cjslGaHaUU14LdLbGEKqTR7PBKwT6LdPtLLh9fPY33rmr5wcOMrsWDbbdCHq4hQUdd37Q==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-buffer": "1.8.5", - "@webassemblyjs/wasm-gen": "1.8.5", - "@webassemblyjs/wasm-parser": "1.8.5" - } - }, - "@webassemblyjs/wasm-parser": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.8.5.tgz", - "integrity": "sha512-pi0SYE9T6tfcMkthwcgCpL0cM9nRYr6/6fjgDtL6q/ZqKHdMWvxitRi5JcZ7RI4SNJJYnYNaWy5UUrHQy998lw==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-api-error": "1.8.5", - "@webassemblyjs/helper-wasm-bytecode": "1.8.5", - "@webassemblyjs/ieee754": "1.8.5", - "@webassemblyjs/leb128": "1.8.5", - "@webassemblyjs/utf8": "1.8.5" - } - }, - "@webassemblyjs/wast-parser": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-parser/-/wast-parser-1.8.5.tgz", - "integrity": "sha512-daXC1FyKWHF1i11obK086QRlsMsY4+tIOKgBqI1lxAnkp9xe9YMcgOxm9kLe+ttjs5aWV2KKE1TWJCN57/Btsg==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/floating-point-hex-parser": "1.8.5", - "@webassemblyjs/helper-api-error": "1.8.5", - "@webassemblyjs/helper-code-frame": "1.8.5", - "@webassemblyjs/helper-fsm": "1.8.5", - "@xtuc/long": "4.2.2" - } - }, - "@webassemblyjs/wast-printer": { - "version": "1.8.5", - "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.8.5.tgz", - "integrity": "sha512-w0U0pD4EhlnvRyeJzBqaVSJAo9w/ce7/WPogeXLzGkO6hzhr4GnQIZ4W4uUt5b9ooAaXPtnXlj0gzsXEOUNYMg==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/wast-parser": "1.8.5", - "@xtuc/long": "4.2.2" - } - }, - "@xtuc/ieee754": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", - "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==", - "dev": true - }, - "@xtuc/long": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", - "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==", - "dev": true - }, - "acorn": { - "version": "6.4.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.0.tgz", - "integrity": "sha512-gac8OEcQ2Li1dxIEWGZzsp2BitJxwkwcOm0zHAJLcPJaVvm58FRnk6RkuLRpU1EujipU2ZFODv2P9DLMfnV8mw==", - "dev": true - }, - "ajv": { - "version": "6.10.2", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.2.tgz", - "integrity": "sha512-TXtUUEYHuaTEbLZWIKUr5pmBuhDLy+8KYtPYdcV8qC+pOZL+NKqYwvWSRrVXHn+ZmRRAu8vJTAznH7Oag6RVRw==", - "dev": true, - "requires": { - "fast-deep-equal": "^2.0.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - } - }, - "ajv-errors": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/ajv-errors/-/ajv-errors-1.0.1.tgz", - "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", - "dev": true - }, - "ajv-keywords": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.4.1.tgz", - "integrity": "sha512-RO1ibKvd27e6FEShVFfPALuHI3WjSVNeK5FIsmme/LYRNxjKuNj+Dt7bucLa6NdSv3JcVTyMlm9kGR84z1XpaQ==", - "dev": true - }, - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "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" - } - }, - "anymatch": { - "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" - }, - "dependencies": { - "normalize-path": { - "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" - } - } - } - }, - "aproba": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", - "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==", - "dev": true - }, - "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 - }, - "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 - }, - "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 - }, - "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 - }, - "asn1.js": { - "version": "4.10.1", - "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", - "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", - "dev": true, - "requires": { - "bn.js": "^4.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0" - } - }, - "assert": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/assert/-/assert-1.5.0.tgz", - "integrity": "sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==", - "dev": true, - "requires": { - "object-assign": "^4.1.1", - "util": "0.10.3" - }, - "dependencies": { - "inherits": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz", - "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=", - "dev": true - }, - "util": { - "version": "0.10.3", - "resolved": "https://registry.npmjs.org/util/-/util-0.10.3.tgz", - "integrity": "sha1-evsa/lCAUkZInj23/g7TeTNqwPk=", - "dev": true, - "requires": { - "inherits": "2.0.1" - } - } - } - }, - "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 - }, - "async-each": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", - "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", - "dev": true - }, - "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 - }, - "balanced-match": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", - "dev": true - }, - "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", - "component-emitter": "^1.2.1", - "define-property": "^1.0.0", - "isobject": "^3.0.1", - "mixin-deep": "^1.2.0", - "pascalcase": "^0.1.1" - }, - "dependencies": { - "define-property": { - "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" - } - }, - "is-accessor-descriptor": { - "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" - } - }, - "is-data-descriptor": { - "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" - } - }, - "is-descriptor": { - "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", - "kind-of": "^6.0.2" - } - } - } - }, - "base64-js": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.1.tgz", - "integrity": "sha512-mLQ4i2QO1ytvGWFWmcngKO//JXAQueZvwEKtjgQFM4jIK0kU+ytMfplL8j+n5mspOfjHwoAg+9yhb7BwAHm36g==", - "dev": true - }, - "big.js": { - "version": "5.2.2", - "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", - "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", - "dev": true - }, - "binary-extensions": { - "version": "1.13.1", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", - "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", - "dev": true - }, - "bindings": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", - "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", - "dev": true, - "optional": true, - "requires": { - "file-uri-to-path": "1.0.0" - } - }, - "bluebird": { - "version": "3.7.2", - "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", - "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", - "dev": true - }, - "bn.js": { - "version": "4.11.8", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", - "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", - "dev": true - }, - "brace-expansion": { - "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" - } - }, - "braces": { - "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", - "extend-shallow": "^2.0.1", - "fill-range": "^4.0.0", - "isobject": "^3.0.1", - "repeat-element": "^1.1.2", - "snapdragon": "^0.8.1", - "snapdragon-node": "^2.0.1", - "split-string": "^3.0.2", - "to-regex": "^3.0.1" - }, - "dependencies": { - "extend-shallow": { - "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" - } - } - } - }, - "brorand": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", - "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=", - "dev": true - }, - "browserify-aes": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", - "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", - "dev": true, - "requires": { - "buffer-xor": "^1.0.3", - "cipher-base": "^1.0.0", - "create-hash": "^1.1.0", - "evp_bytestokey": "^1.0.3", - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "browserify-cipher": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/browserify-cipher/-/browserify-cipher-1.0.1.tgz", - "integrity": "sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==", - "dev": true, - "requires": { - "browserify-aes": "^1.0.4", - "browserify-des": "^1.0.0", - "evp_bytestokey": "^1.0.0" - } - }, - "browserify-des": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/browserify-des/-/browserify-des-1.0.2.tgz", - "integrity": "sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==", - "dev": true, - "requires": { - "cipher-base": "^1.0.1", - "des.js": "^1.0.0", - "inherits": "^2.0.1", - "safe-buffer": "^5.1.2" - } - }, - "browserify-rsa": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.0.1.tgz", - "integrity": "sha1-IeCr+vbyApzy+vsTNWenAdQTVSQ=", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "randombytes": "^2.0.1" - } - }, - "browserify-sign": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/browserify-sign/-/browserify-sign-4.0.4.tgz", - "integrity": "sha1-qk62jl17ZYuqa/alfmMMvXqT0pg=", - "dev": true, - "requires": { - "bn.js": "^4.1.1", - "browserify-rsa": "^4.0.0", - "create-hash": "^1.1.0", - "create-hmac": "^1.1.2", - "elliptic": "^6.0.0", - "inherits": "^2.0.1", - "parse-asn1": "^5.0.0" - } - }, - "browserify-zlib": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/browserify-zlib/-/browserify-zlib-0.2.0.tgz", - "integrity": "sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==", - "dev": true, - "requires": { - "pako": "~1.0.5" - } - }, - "buffer": { - "version": "4.9.2", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-4.9.2.tgz", - "integrity": "sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==", - "dev": true, - "requires": { - "base64-js": "^1.0.2", - "ieee754": "^1.1.4", - "isarray": "^1.0.0" - } - }, - "buffer-from": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", - "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", - "dev": true - }, - "buffer-xor": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz", - "integrity": "sha1-JuYe0UIvtw3ULm42cp7VHYVf6Nk=", - "dev": true - }, - "builtin-status-codes": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz", - "integrity": "sha1-hZgoeOIbmOHGZCXgPQF0eI9Wnug=", - "dev": true - }, - "cacache": { - "version": "12.0.3", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.3.tgz", - "integrity": "sha512-kqdmfXEGFepesTuROHMs3MpFLWrPkSSpRqOw80RCflZXy/khxaArvFrQ7uJxSUduzAufc6G0g1VUCOZXxWavPw==", - "dev": true, - "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" - } - }, - "cache-base": { - "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", - "get-value": "^2.0.6", - "has-value": "^1.0.0", - "isobject": "^3.0.1", - "set-value": "^2.0.0", - "to-object-path": "^0.3.0", - "union-value": "^1.0.0", - "unset-value": "^1.0.0" - } - }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, - "chalk": { - "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", - "supports-color": "^5.3.0" - }, - "dependencies": { - "supports-color": { - "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" - } - } - } - }, - "chokidar": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", - "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", - "dev": true, - "requires": { - "anymatch": "^2.0.0", - "async-each": "^1.0.1", - "braces": "^2.3.2", - "fsevents": "^1.2.7", - "glob-parent": "^3.1.0", - "inherits": "^2.0.3", - "is-binary-path": "^1.0.0", - "is-glob": "^4.0.0", - "normalize-path": "^3.0.0", - "path-is-absolute": "^1.0.0", - "readdirp": "^2.2.1", - "upath": "^1.1.1" - } - }, - "chownr": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.3.tgz", - "integrity": "sha512-i70fVHhmV3DtTl6nqvZOnIjbY0Pe4kAUjwHj8z0zAdgBtYrJyYwLKCCuRBQ5ppkyL0AkN7HKRnETdmdp1zqNXw==", - "dev": true - }, - "chrome-trace-event": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.2.tgz", - "integrity": "sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "cipher-base": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz", - "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "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", - "isobject": "^3.0.0", - "static-extend": "^0.1.1" - }, - "dependencies": { - "define-property": { - "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" - } - } - } - }, - "cliui": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", - "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", - "dev": true, - "requires": { - "string-width": "^3.1.0", - "strip-ansi": "^5.2.0", - "wrap-ansi": "^5.1.0" - } - }, - "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" - } - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, - "commander": { - "version": "2.20.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", - "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", - "dev": true - }, - "commondir": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", - "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", - "dev": true - }, - "component-emitter": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", - "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", - "dev": true - }, - "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 - }, - "concat-stream": { - "version": "1.6.2", - "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", - "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", - "dev": true, - "requires": { - "buffer-from": "^1.0.0", - "inherits": "^2.0.3", - "readable-stream": "^2.2.2", - "typedarray": "^0.0.6" - } - }, - "console-browserify": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/console-browserify/-/console-browserify-1.2.0.tgz", - "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==", - "dev": true - }, - "constants-browserify": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/constants-browserify/-/constants-browserify-1.0.0.tgz", - "integrity": "sha1-wguW2MYXdIqvHBYCF2DNJ/y4y3U=", - "dev": true - }, - "copy-concurrently": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/copy-concurrently/-/copy-concurrently-1.0.5.tgz", - "integrity": "sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A==", - "dev": true, - "requires": { - "aproba": "^1.1.1", - "fs-write-stream-atomic": "^1.0.8", - "iferr": "^0.1.5", - "mkdirp": "^0.5.1", - "rimraf": "^2.5.4", - "run-queue": "^1.0.0" - } - }, - "copy-descriptor": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", - "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", - "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 - }, - "create-ecdh": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/create-ecdh/-/create-ecdh-4.0.3.tgz", - "integrity": "sha512-GbEHQPMOswGpKXM9kCWVrremUcBmjteUaQ01T9rkKCPDXfUHX0IoP9LpHYo2NPFampa4e+/pFDc3jQdxrxQLaw==", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "elliptic": "^6.0.0" - } - }, - "create-hash": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", - "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", - "dev": true, - "requires": { - "cipher-base": "^1.0.1", - "inherits": "^2.0.1", - "md5.js": "^1.3.4", - "ripemd160": "^2.0.1", - "sha.js": "^2.4.0" - } - }, - "create-hmac": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", - "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", - "dev": true, - "requires": { - "cipher-base": "^1.0.3", - "create-hash": "^1.1.0", - "inherits": "^2.0.1", - "ripemd160": "^2.0.0", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - }, - "cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", - "dev": true, - "requires": { - "nice-try": "^1.0.4", - "path-key": "^2.0.1", - "semver": "^5.5.0", - "shebang-command": "^1.2.0", - "which": "^1.2.9" - } - }, - "crypto-browserify": { - "version": "3.12.0", - "resolved": "https://registry.npmjs.org/crypto-browserify/-/crypto-browserify-3.12.0.tgz", - "integrity": "sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==", - "dev": true, - "requires": { - "browserify-cipher": "^1.0.0", - "browserify-sign": "^4.0.0", - "create-ecdh": "^4.0.0", - "create-hash": "^1.1.0", - "create-hmac": "^1.1.0", - "diffie-hellman": "^5.0.0", - "inherits": "^2.0.1", - "pbkdf2": "^3.0.3", - "public-encrypt": "^4.0.0", - "randombytes": "^2.0.0", - "randomfill": "^1.0.3" - } - }, - "cyclist": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/cyclist/-/cyclist-1.0.1.tgz", - "integrity": "sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=", - "dev": true - }, - "debug": { - "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" - } - }, - "decamelize": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", - "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", - "dev": true - }, - "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 - }, - "define-property": { - "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" - }, - "dependencies": { - "is-accessor-descriptor": { - "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" - } - }, - "is-data-descriptor": { - "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" - } - }, - "is-descriptor": { - "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", - "kind-of": "^6.0.2" - } - } - } - }, - "des.js": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/des.js/-/des.js-1.0.1.tgz", - "integrity": "sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0" - } - }, - "detect-file": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/detect-file/-/detect-file-1.0.0.tgz", - "integrity": "sha1-8NZtA2cqglyxtzvbP+YjEMjlUrc=", - "dev": true - }, - "diffie-hellman": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", - "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "miller-rabin": "^4.0.0", - "randombytes": "^2.0.0" - } - }, - "domain-browser": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/domain-browser/-/domain-browser-1.2.0.tgz", - "integrity": "sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==", - "dev": true - }, - "duplexify": { - "version": "3.7.1", - "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-3.7.1.tgz", - "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", - "dev": true, - "requires": { - "end-of-stream": "^1.0.0", - "inherits": "^2.0.1", - "readable-stream": "^2.0.0", - "stream-shift": "^1.0.0" - } - }, - "elliptic": { - "version": "6.5.2", - "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.2.tgz", - "integrity": "sha512-f4x70okzZbIQl/NSRLkI/+tteV/9WqL98zx+SQ69KbXxmVrmjwsNUPn/gYJJ0sHvEak24cZgHIPegRePAtA/xw==", - "dev": true, - "requires": { - "bn.js": "^4.4.0", - "brorand": "^1.0.1", - "hash.js": "^1.0.0", - "hmac-drbg": "^1.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0", - "minimalistic-crypto-utils": "^1.0.0" - } - }, - "emoji-regex": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", - "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", - "dev": true - }, - "emojis-list": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-2.1.0.tgz", - "integrity": "sha1-TapNnbAPmBmIDHn6RXrlsJof04k=", - "dev": true - }, - "end-of-stream": { - "version": "1.4.4", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", - "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", - "dev": true, - "requires": { - "once": "^1.4.0" - } - }, - "enhanced-resolve": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.1.1.tgz", - "integrity": "sha512-98p2zE+rL7/g/DzMHMTF4zZlCgeVdJ7yr6xzEpJRYwFYrGi9ANdn5DnJURg6RpBkyk60XYDnWIv51VfIhfNGuA==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "memory-fs": "^0.5.0", - "tapable": "^1.0.0" - }, - "dependencies": { - "memory-fs": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.5.0.tgz", - "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - } - } - }, - "errno": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.7.tgz", - "integrity": "sha512-MfrRBDWzIWifgq6tJj60gkAwtLNb6sQPlcFrSOflcP1aFmmruKQ2wRnze/8V6kgyz7H3FF8Npzv78mZ7XLLflg==", - "dev": true, - "requires": { - "prr": "~1.0.1" - } - }, - "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 - }, - "eslint-scope": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.3.tgz", - "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", - "dev": true, - "requires": { - "esrecurse": "^4.1.0", - "estraverse": "^4.1.1" - } - }, - "esrecurse": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz", - "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==", - "dev": true, - "requires": { - "estraverse": "^4.1.0" - } - }, - "estraverse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", - "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", - "dev": true - }, - "events": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/events/-/events-3.1.0.tgz", - "integrity": "sha512-Rv+u8MLHNOdMjTAFeT3nCjHn2aGlx435FP/sDHNaRhDEMwyI/aB22Kj2qIN8R0cw3z28psEQLYwxVKLsKrMgWg==", - "dev": true - }, - "evp_bytestokey": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz", - "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==", - "dev": true, - "requires": { - "md5.js": "^1.3.4", - "safe-buffer": "^5.1.1" - } - }, - "execa": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", - "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", - "dev": true, - "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" - } - }, - "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", - "extend-shallow": "^2.0.1", - "posix-character-classes": "^0.1.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - }, - "dependencies": { - "define-property": { - "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" - } - }, - "extend-shallow": { - "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-tilde": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/expand-tilde/-/expand-tilde-2.0.2.tgz", - "integrity": "sha1-l+gBqgUt8CRU3kawK/YhZCzchQI=", - "dev": true, - "requires": { - "homedir-polyfill": "^1.0.1" - } - }, - "extend-shallow": { - "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" - }, - "dependencies": { - "is-extendable": { - "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" - } - } - } - }, - "extglob": { - "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", - "expand-brackets": "^2.1.4", - "extend-shallow": "^2.0.1", - "fragment-cache": "^0.2.1", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - }, - "dependencies": { - "define-property": { - "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" - } - }, - "extend-shallow": { - "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" - } - }, - "is-accessor-descriptor": { - "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" - } - }, - "is-data-descriptor": { - "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" - } - }, - "is-descriptor": { - "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", - "kind-of": "^6.0.2" - } - } - } - }, - "fast-deep-equal": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz", - "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=", - "dev": true - }, - "fast-json-stable-stringify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true - }, - "figgy-pudding": { - "version": "3.5.1", - "resolved": "https://registry.npmjs.org/figgy-pudding/-/figgy-pudding-3.5.1.tgz", - "integrity": "sha512-vNKxJHTEKNThjfrdJwHc7brvM6eVevuO5nTj6ez8ZQ1qbXTvGthucRF7S4vf2cr71QVnT70V34v0S1DyQsti0w==", - "dev": true - }, - "file-uri-to-path": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", - "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", - "dev": true, - "optional": true - }, - "fill-range": { - "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", - "repeat-string": "^1.6.1", - "to-regex-range": "^2.1.0" - }, - "dependencies": { - "extend-shallow": { - "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" - } - } - } - }, - "find-cache-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.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==", - "dev": true, - "requires": { - "locate-path": "^3.0.0" - } - }, - "findup-sync": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/findup-sync/-/findup-sync-3.0.0.tgz", - "integrity": "sha512-YbffarhcicEhOrm4CtrwdKBdCuz576RLdhJDsIfvNtxUuhdRet1qZcsMjqbePtAseKdAnDyM/IyXbu7PRPRLYg==", - "dev": true, - "requires": { - "detect-file": "^1.0.0", - "is-glob": "^4.0.0", - "micromatch": "^3.0.4", - "resolve-dir": "^1.0.1" - } - }, - "flush-write-stream": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/flush-write-stream/-/flush-write-stream-1.1.1.tgz", - "integrity": "sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "readable-stream": "^2.3.6" - } - }, - "fontfaceobserver": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fontfaceobserver/-/fontfaceobserver-2.1.0.tgz", - "integrity": "sha512-ReOsO2F66jUa0jmv2nlM/s1MiutJx/srhAe2+TE8dJCMi02ZZOcCTxTCQFr3Yet+uODUtnr4Mewg+tNQ+4V1Ng==" - }, - "for-in": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", - "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", - "dev": true - }, - "fragment-cache": { - "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" - } - }, - "from2": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", - "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "readable-stream": "^2.0.0" - } - }, - "fs-write-stream-atomic": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz", - "integrity": "sha1-tH31NJPvkR33VzHnCp3tAYnbQMk=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "iferr": "^0.1.5", - "imurmurhash": "^0.1.4", - "readable-stream": "1 || 2" - } - }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", - "dev": true - }, - "fsevents": { - "version": "1.2.11", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.11.tgz", - "integrity": "sha512-+ux3lx6peh0BpvY0JebGyZoiR4D+oYzdPZMKJwkZ+sFkNJzpL7tXc/wehS49gUAxg3tmMHPHZkA8JU2rhhgDHw==", - "dev": true, - "optional": true, - "requires": { - "bindings": "^1.5.0", - "nan": "^2.12.1", - "node-pre-gyp": "*" - }, - "dependencies": { - "abbrev": { - "version": "1.1.1", - "bundled": true, - "dev": true, - "optional": true - }, - "ansi-regex": { - "version": "2.1.1", - "bundled": true, - "dev": 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", - "readable-stream": "^2.0.6" - } - }, - "balanced-match": { - "version": "1.0.0", - "bundled": true, - "dev": true - }, - "brace-expansion": { - "version": "1.1.11", - "bundled": true, - "dev": true, - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "chownr": { - "version": "1.1.3", - "bundled": true, - "dev": true, - "optional": true - }, - "code-point-at": { - "version": "1.1.0", - "bundled": true, - "dev": true - }, - "concat-map": { - "version": "0.0.1", - "bundled": true, - "dev": true - }, - "console-control-strings": { - "version": "1.1.0", - "bundled": true, - "dev": true - }, - "core-util-is": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "debug": { - "version": "3.2.6", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "ms": "^2.1.1" - } - }, - "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.7", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minipass": "^2.6.0" - } - }, - "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", - "console-control-strings": "^1.0.0", - "has-unicode": "^2.0.0", - "object-assign": "^4.1.0", - "signal-exit": "^3.0.0", - "string-width": "^1.0.1", - "strip-ansi": "^3.0.1", - "wide-align": "^1.1.0" - } - }, - "glob": { - "version": "7.1.6", - "bundled": true, - "dev": true, - "optional": true, - "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" - } - }, - "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" - } - }, - "ignore-walk": { - "version": "3.0.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minimatch": "^3.0.4" - } - }, - "inflight": { - "version": "1.0.6", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "once": "^1.3.0", - "wrappy": "1" - } - }, - "inherits": { - "version": "2.0.4", - "bundled": true, - "dev": true - }, - "ini": { - "version": "1.3.5", - "bundled": true, - "dev": true, - "optional": true - }, - "is-fullwidth-code-point": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "requires": { - "number-is-nan": "^1.0.0" - } - }, - "isarray": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "optional": true - }, - "minimatch": { - "version": "3.0.4", - "bundled": true, - "dev": true, - "requires": { - "brace-expansion": "^1.1.7" - } - }, - "minimist": { - "version": "0.0.8", - "bundled": true, - "dev": true - }, - "minipass": { - "version": "2.9.0", - "bundled": true, - "dev": true, - "requires": { - "safe-buffer": "^5.1.2", - "yallist": "^3.0.0" - } - }, - "minizlib": { - "version": "1.3.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minipass": "^2.9.0" - } - }, - "mkdirp": { - "version": "0.5.1", - "bundled": true, - "dev": true, - "requires": { - "minimist": "0.0.8" - } - }, - "ms": { - "version": "2.1.2", - "bundled": true, - "dev": true, - "optional": true - }, - "needle": { - "version": "2.4.0", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "debug": "^3.2.6", - "iconv-lite": "^0.4.4", - "sax": "^1.2.4" - } - }, - "node-pre-gyp": { - "version": "0.14.0", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "detect-libc": "^1.0.2", - "mkdirp": "^0.5.1", - "needle": "^2.2.1", - "nopt": "^4.0.1", - "npm-packlist": "^1.1.6", - "npmlog": "^4.0.2", - "rc": "^1.2.7", - "rimraf": "^2.6.1", - "semver": "^5.3.0", - "tar": "^4.4.2" - } - }, - "nopt": { - "version": "4.0.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "abbrev": "1", - "osenv": "^0.1.4" - } - }, - "npm-bundled": { - "version": "1.1.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npm-normalize-package-bin": { - "version": "1.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "npm-packlist": { - "version": "1.4.7", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "ignore-walk": "^3.0.1", - "npm-bundled": "^1.0.1" - } - }, - "npmlog": { - "version": "4.1.2", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "are-we-there-yet": "~1.1.2", - "console-control-strings": "~1.1.0", - "gauge": "~2.7.3", - "set-blocking": "~2.0.0" - } - }, - "number-is-nan": { - "version": "1.0.1", - "bundled": true, - "dev": true - }, - "object-assign": { - "version": "4.1.1", - "bundled": true, - "dev": true, - "optional": true - }, - "once": { - "version": "1.4.0", - "bundled": true, - "dev": true, - "requires": { - "wrappy": "1" - } - }, - "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", - "os-tmpdir": "^1.0.0" - } - }, - "path-is-absolute": { - "version": "1.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "process-nextick-args": { - "version": "2.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "rc": { - "version": "1.2.8", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "deep-extend": "^0.6.0", - "ini": "~1.3.0", - "minimist": "^1.2.0", - "strip-json-comments": "~2.0.1" - }, - "dependencies": { - "minimist": { - "version": "1.2.0", - "bundled": true, - "dev": true, - "optional": true - } - } - }, - "readable-stream": { - "version": "2.3.6", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" - } - }, - "rimraf": { - "version": "2.7.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "glob": "^7.1.3" - } - }, - "safe-buffer": { - "version": "5.1.2", - "bundled": true, - "dev": 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.7.1", - "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, - "requires": { - "code-point-at": "^1.0.0", - "is-fullwidth-code-point": "^1.0.0", - "strip-ansi": "^3.0.0" - } - }, - "string_decoder": { - "version": "1.1.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "safe-buffer": "~5.1.0" - } - }, - "strip-ansi": { - "version": "3.0.1", - "bundled": true, - "dev": true, - "requires": { - "ansi-regex": "^2.0.0" - } - }, - "strip-json-comments": { - "version": "2.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "tar": { - "version": "4.4.13", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "chownr": "^1.1.1", - "fs-minipass": "^1.2.5", - "minipass": "^2.8.6", - "minizlib": "^1.2.1", - "mkdirp": "^0.5.0", - "safe-buffer": "^5.1.2", - "yallist": "^3.0.3" - } - }, - "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" - } - }, - "wrappy": { - "version": "1.0.2", - "bundled": true, - "dev": true - }, - "yallist": { - "version": "3.1.1", - "bundled": true, - "dev": true - } - } - }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "dev": true - }, - "get-stream": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", - "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", - "dev": true, - "requires": { - "pump": "^3.0.0" - } - }, - "get-value": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", - "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", - "dev": true - }, - "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", - "dev": true, - "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" - } - }, - "glob-parent": { - "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" - }, - "dependencies": { - "is-glob": { - "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" - } - } - } - }, - "global-modules": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-2.0.0.tgz", - "integrity": "sha512-NGbfmJBp9x8IxyJSd1P+otYK8vonoJactOogrVfFRIAEY1ukil8RSKDz2Yo7wh1oihl51l/r6W4epkeKJHqL8A==", - "dev": true, - "requires": { - "global-prefix": "^3.0.0" - }, - "dependencies": { - "global-prefix": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-3.0.0.tgz", - "integrity": "sha512-awConJSVCHVGND6x3tmMaKcQvwXLhjdkmomy2W+Goaui8YPgYgXJZewhg3fWC+DlfqqQuWg8AwqjGTD2nAPVWg==", - "dev": true, - "requires": { - "ini": "^1.3.5", - "kind-of": "^6.0.2", - "which": "^1.3.1" - } - } - } - }, - "global-prefix": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/global-prefix/-/global-prefix-1.0.2.tgz", - "integrity": "sha1-2/dDxsFJklk8ZVVoy2btMsASLr4=", - "dev": true, - "requires": { - "expand-tilde": "^2.0.2", - "homedir-polyfill": "^1.0.1", - "ini": "^1.3.4", - "is-windows": "^1.0.1", - "which": "^1.2.14" - } - }, - "graceful-fs": { - "version": "4.2.3", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", - "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==", - "dev": true - }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true - }, - "has-value": { - "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", - "isobject": "^3.0.0" - } - }, - "has-values": { - "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" - }, - "dependencies": { - "kind-of": { - "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" - } - } - } - }, - "hash-base": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.0.4.tgz", - "integrity": "sha1-X8hoaEfs1zSZQDMZprCj8/auSRg=", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "hash.js": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", - "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "minimalistic-assert": "^1.0.1" - } - }, - "hmac-drbg": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", - "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=", - "dev": true, - "requires": { - "hash.js": "^1.0.3", - "minimalistic-assert": "^1.0.0", - "minimalistic-crypto-utils": "^1.0.1" - } - }, - "homedir-polyfill": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/homedir-polyfill/-/homedir-polyfill-1.0.3.tgz", - "integrity": "sha512-eSmmWE5bZTK2Nou4g0AI3zZ9rswp7GRKoKXS1BLUkvPviOqs4YTN1djQIqrXy9k5gEtdLPy86JjRwsNM9tnDcA==", - "dev": true, - "requires": { - "parse-passwd": "^1.0.0" - } - }, - "https-browserify": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/https-browserify/-/https-browserify-1.0.0.tgz", - "integrity": "sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM=", - "dev": true - }, - "ieee754": { - "version": "1.1.13", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz", - "integrity": "sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==", - "dev": true - }, - "iferr": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/iferr/-/iferr-0.1.5.tgz", - "integrity": "sha1-xg7taebY/bazEEofy8ocGS3FtQE=", - "dev": true - }, - "import-local": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/import-local/-/import-local-2.0.0.tgz", - "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", - "dev": true, - "requires": { - "pkg-dir": "^3.0.0", - "resolve-cwd": "^2.0.0" - } - }, - "imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", - "dev": true - }, - "infer-owner": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/infer-owner/-/infer-owner-1.0.4.tgz", - "integrity": "sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==", - "dev": true - }, - "inflight": { - "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" - } - }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", - "dev": true - }, - "ini": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", - "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", - "dev": true - }, - "interpret": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.2.0.tgz", - "integrity": "sha512-mT34yGKMNceBQUoVn7iCDKDntA7SC6gycMAWzGx1z/CMCTV7b2AAtXlo3nRyHZ1FelRkQbQjprHSYGwzLtkVbw==", - "dev": true - }, - "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==", - "dev": true - }, - "is-accessor-descriptor": { - "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" - }, - "dependencies": { - "kind-of": { - "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" - } - } - } - }, - "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" - } - }, - "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 - }, - "is-data-descriptor": { - "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" - }, - "dependencies": { - "kind-of": { - "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" - } - } - } - }, - "is-descriptor": { - "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", - "kind-of": "^5.0.0" - }, - "dependencies": { - "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 - } - } - }, - "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 - }, - "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 - }, - "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 - }, - "is-glob": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", - "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", - "dev": true, - "requires": { - "is-extglob": "^2.1.1" - } - }, - "is-number": { - "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" - }, - "dependencies": { - "kind-of": { - "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" - } - } - } - }, - "is-plain-object": { - "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" - } - }, - "is-stream": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", - "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", - "dev": true - }, - "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 - }, - "is-wsl": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", - "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", - "dev": true - }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", - "dev": true - }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "dev": true - }, - "isobject": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", - "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", - "dev": true - }, - "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 - }, - "json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true - }, - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "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 - }, - "lcid": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/lcid/-/lcid-2.0.0.tgz", - "integrity": "sha512-avPEb8P8EGnwXKClwsNUgryVjllcRqtMYa49NTsbQagYuT1DcXnl1915oxWjoyGrXR6zH/Y0Zc96xWsPcoDKeA==", - "dev": true, - "requires": { - "invert-kv": "^2.0.0" - } - }, - "loader-runner": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-2.4.0.tgz", - "integrity": "sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw==", - "dev": true - }, - "loader-utils": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.2.3.tgz", - "integrity": "sha512-fkpz8ejdnEMG3s37wGL07iSBDg99O9D5yflE9RGNH3hRdx9SOwYfnGYdZOUIZitN8E+E2vkq3MUMYMvPYl5ZZA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^2.0.0", - "json5": "^1.0.1" - } - }, - "locate-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", - "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", - "dev": true, - "requires": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" - } - }, - "lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "requires": { - "yallist": "^3.0.2" - } - }, - "make-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", - "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", - "dev": true, - "requires": { - "pify": "^4.0.1", - "semver": "^5.6.0" - } - }, - "mamacro": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/mamacro/-/mamacro-0.0.3.tgz", - "integrity": "sha512-qMEwh+UujcQ+kbz3T6V+wAmO2U8veoq2w+3wY8MquqwVA3jChfwY+Tk52GZKDfACEPjuZ7r2oJLejwpt8jtwTA==", - "dev": true - }, - "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==", - "dev": true, - "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 - }, - "map-visit": { - "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" - } - }, - "md5.js": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", - "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", - "dev": true, - "requires": { - "hash-base": "^3.0.0", - "inherits": "^2.0.1", - "safe-buffer": "^5.1.2" - } - }, - "mem": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/mem/-/mem-4.3.0.tgz", - "integrity": "sha512-qX2bG48pTqYRVmDB37rn/6PT7LcR8T7oAX3bf99u1Tt1nzxYfxkgqDwUwolPlXweM0XzBOBFzSx4kfp7KP1s/w==", - "dev": true, - "requires": { - "map-age-cleaner": "^0.1.1", - "mimic-fn": "^2.0.0", - "p-is-promise": "^2.0.0" - } - }, - "memory-fs": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", - "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - }, - "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", - "braces": "^2.3.1", - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "extglob": "^2.0.4", - "fragment-cache": "^0.2.1", - "kind-of": "^6.0.2", - "nanomatch": "^1.2.9", - "object.pick": "^1.3.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.2" - } - }, - "miller-rabin": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/miller-rabin/-/miller-rabin-4.0.1.tgz", - "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", - "dev": true, - "requires": { - "bn.js": "^4.0.0", - "brorand": "^1.0.1" - } - }, - "mimic-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", - "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", - "dev": true - }, - "minimalistic-assert": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", - "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==", - "dev": true - }, - "minimalistic-crypto-utils": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", - "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=", - "dev": true - }, - "minimatch": { - "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" - } - }, - "minimist": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.0.tgz", - "integrity": "sha1-o1AIsg9BOD7sH7kU9M1d95omQoQ=", - "dev": true - }, - "mississippi": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/mississippi/-/mississippi-3.0.0.tgz", - "integrity": "sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA==", - "dev": true, - "requires": { - "concat-stream": "^1.5.0", - "duplexify": "^3.4.2", - "end-of-stream": "^1.1.0", - "flush-write-stream": "^1.0.0", - "from2": "^2.1.0", - "parallel-transform": "^1.1.0", - "pump": "^3.0.0", - "pumpify": "^1.3.3", - "stream-each": "^1.1.0", - "through2": "^2.0.0" - } - }, - "mixin-deep": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", - "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", - "dev": true, - "requires": { - "for-in": "^1.0.2", - "is-extendable": "^1.0.1" - }, - "dependencies": { - "is-extendable": { - "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" - } - } - } - }, - "mkdirp": { - "version": "0.5.1", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", - "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", - "dev": true, - "requires": { - "minimist": "0.0.8" - }, - "dependencies": { - "minimist": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", - "dev": true - } - } - }, - "move-concurrently": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz", - "integrity": "sha1-viwAX9oy4LKa8fBdfEszIUxwH5I=", - "dev": true, - "requires": { - "aproba": "^1.1.1", - "copy-concurrently": "^1.0.0", - "fs-write-stream-atomic": "^1.0.8", - "mkdirp": "^0.5.1", - "rimraf": "^2.5.4", - "run-queue": "^1.0.3" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "nan": { - "version": "2.14.0", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.0.tgz", - "integrity": "sha512-INOFj37C7k3AfaNTtX8RhsTw7qRy7eLET14cROi9+5HAVbbHuIWUHEauBv5qT4Av2tWasiTY1Jw6puUNqRJXQg==", - "dev": true, - "optional": true - }, - "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", - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "fragment-cache": "^0.2.1", - "is-windows": "^1.0.2", - "kind-of": "^6.0.2", - "object.pick": "^1.3.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - } - }, - "neo-async": { - "version": "2.6.1", - "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.1.tgz", - "integrity": "sha512-iyam8fBuCUpWeKPGpaNMetEocMt364qkCsfL9JuhjXX6dRnguRVOfk2GZaDpPjcOKiiXCPINZC1GczQ7iTq3Zw==", - "dev": true - }, - "nice-try": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", - "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", - "dev": true - }, - "node-libs-browser": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/node-libs-browser/-/node-libs-browser-2.2.1.tgz", - "integrity": "sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q==", - "dev": true, - "requires": { - "assert": "^1.1.1", - "browserify-zlib": "^0.2.0", - "buffer": "^4.3.0", - "console-browserify": "^1.1.0", - "constants-browserify": "^1.0.0", - "crypto-browserify": "^3.11.0", - "domain-browser": "^1.1.1", - "events": "^3.0.0", - "https-browserify": "^1.0.0", - "os-browserify": "^0.3.0", - "path-browserify": "0.0.1", - "process": "^0.11.10", - "punycode": "^1.2.4", - "querystring-es3": "^0.2.0", - "readable-stream": "^2.3.3", - "stream-browserify": "^2.0.1", - "stream-http": "^2.7.2", - "string_decoder": "^1.0.0", - "timers-browserify": "^2.0.4", - "tty-browserify": "0.0.0", - "url": "^0.11.0", - "util": "^0.11.0", - "vm-browserify": "^1.0.1" - }, - "dependencies": { - "punycode": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", - "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", - "dev": true - } - } - }, - "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 - }, - "npm-run-path": { - "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" - } - }, - "object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", - "dev": true - }, - "object-copy": { - "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", - "kind-of": "^3.0.3" - }, - "dependencies": { - "define-property": { - "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" - } - }, - "kind-of": { - "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" - } - } - } - }, - "object-visit": { - "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" - } - }, - "object.pick": { - "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" - } - }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "dev": true, - "requires": { - "wrappy": "1" - } - }, - "os-browserify": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/os-browserify/-/os-browserify-0.3.0.tgz", - "integrity": "sha1-hUNzx/XCMVkU/Jv8a9gjj92h7Cc=", - "dev": true - }, - "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==", - "dev": true, - "requires": { - "execa": "^1.0.0", - "lcid": "^2.0.0", - "mem": "^4.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=", - "dev": true - }, - "p-finally": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", - "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", - "dev": true - }, - "p-is-promise": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-2.1.0.tgz", - "integrity": "sha512-Y3W0wlRPK8ZMRbNq97l4M5otioeA5lm1z7bkNkxCka8HSPjR0xRWmpCmc9utiaLP9Jb1eD8BgeIxTW4AIF45Pg==", - "dev": true - }, - "p-limit": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.2.2.tgz", - "integrity": "sha512-WGR+xHecKTr7EbUEhyLSh5Dube9JtdiG78ufaeLxTgpudf/20KqyMioIUZJAezlTIi6evxuoUs9YXc11cU+yzQ==", - "dev": true, - "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==", - "dev": true, - "requires": { - "p-limit": "^2.0.0" - } - }, - "p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", - "dev": true - }, - "pako": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.10.tgz", - "integrity": "sha512-0DTvPVU3ed8+HNXOu5Bs+o//Mbdj9VNQMUOe9oKCwh8l0GNwpTDMKCWbRjgtD291AWnkAgkqA/LOnQS8AmS1tw==", - "dev": true - }, - "parallel-transform": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/parallel-transform/-/parallel-transform-1.2.0.tgz", - "integrity": "sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg==", - "dev": true, - "requires": { - "cyclist": "^1.0.1", - "inherits": "^2.0.3", - "readable-stream": "^2.1.5" - } - }, - "parse-asn1": { - "version": "5.1.5", - "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.5.tgz", - "integrity": "sha512-jkMYn1dcJqF6d5CpU689bq7w/b5ALS9ROVSpQDPrZsqqesUJii9qutvoT5ltGedNXMO2e16YUWIghG9KxaViTQ==", - "dev": true, - "requires": { - "asn1.js": "^4.0.0", - "browserify-aes": "^1.0.0", - "create-hash": "^1.1.0", - "evp_bytestokey": "^1.0.0", - "pbkdf2": "^3.0.3", - "safe-buffer": "^5.1.1" - } - }, - "parse-passwd": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/parse-passwd/-/parse-passwd-1.0.0.tgz", - "integrity": "sha1-bVuTSkVpk7I9N/QKOC1vFmao5cY=", - "dev": true - }, - "pascalcase": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", - "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", - "dev": true - }, - "path-browserify": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-0.0.1.tgz", - "integrity": "sha512-BapA40NHICOS+USX9SN4tyhq+A2RrN/Ws5F0Z5aMHDp98Fl86lX8Oti8B7uN93L4Ifv4fHOEA+pQw87gmMO/lQ==", - "dev": true - }, - "path-dirname": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", - "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", - "dev": true - }, - "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", - "dev": true - }, - "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 - }, - "path-key": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", - "dev": true - }, - "pbkdf2": { - "version": "3.0.17", - "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.17.tgz", - "integrity": "sha512-U/il5MsrZp7mGg3mSQfn742na2T+1/vHDCG5/iTI3X9MKUuYUZVLQhyRsg06mCgDBTd57TxzgZt7P+fYfjRLtA==", - "dev": true, - "requires": { - "create-hash": "^1.1.2", - "create-hmac": "^1.1.4", - "ripemd160": "^2.0.1", - "safe-buffer": "^5.0.1", - "sha.js": "^2.4.8" - } - }, - "pify": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", - "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", - "dev": true - }, - "pkg-dir": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", - "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", - "dev": true, - "requires": { - "find-up": "^3.0.0" - } - }, - "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 - }, - "process": { - "version": "0.11.10", - "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", - "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=", - "dev": true - }, - "process-nextick-args": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", - "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", - "dev": true - }, - "promise-inflight": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/promise-inflight/-/promise-inflight-1.0.1.tgz", - "integrity": "sha1-mEcocL8igTL8vdhoEputEsPAKeM=", - "dev": true - }, - "prr": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", - "integrity": "sha1-0/wRS6BplaRexok/SEzrHXj19HY=", - "dev": true - }, - "public-encrypt": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", - "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", - "dev": true, - "requires": { - "bn.js": "^4.1.0", - "browserify-rsa": "^4.0.0", - "create-hash": "^1.1.0", - "parse-asn1": "^5.0.0", - "randombytes": "^2.0.1", - "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==", - "dev": true, - "requires": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" - } - }, - "pumpify": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/pumpify/-/pumpify-1.5.1.tgz", - "integrity": "sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==", - "dev": true, - "requires": { - "duplexify": "^3.6.0", - "inherits": "^2.0.3", - "pump": "^2.0.0" - }, - "dependencies": { - "pump": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/pump/-/pump-2.0.1.tgz", - "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", - "dev": true, - "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", - "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", - "dev": true - }, - "querystring": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", - "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", - "dev": true - }, - "querystring-es3": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/querystring-es3/-/querystring-es3-0.2.1.tgz", - "integrity": "sha1-nsYfeQSYdXB9aUFFlv2Qek1xHnM=", - "dev": true - }, - "randombytes": { - "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" - } - }, - "randomfill": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", - "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", - "dev": true, - "requires": { - "randombytes": "^2.0.5", - "safe-buffer": "^5.1.0" - } - }, - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "dev": true, - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" - } - }, - "readdirp": { - "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", - "readable-stream": "^2.0.2" - } - }, - "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" - } - }, - "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 - }, - "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 - }, - "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 - }, - "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 - }, - "require-main-filename": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", - "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", - "dev": true - }, - "resolve-cwd": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-2.0.0.tgz", - "integrity": "sha1-AKn3OHVW4nA46uIyyqNypqWbZlo=", - "dev": true, - "requires": { - "resolve-from": "^3.0.0" - } - }, - "resolve-dir": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/resolve-dir/-/resolve-dir-1.0.1.tgz", - "integrity": "sha1-eaQGRMNivoLybv/nOcm7U4IEb0M=", - "dev": true, - "requires": { - "expand-tilde": "^2.0.0", - "global-modules": "^1.0.0" - }, - "dependencies": { - "global-modules": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/global-modules/-/global-modules-1.0.0.tgz", - "integrity": "sha512-sKzpEkf11GpOFuw0Zzjzmt4B4UZwjOcG757PPvrfhxcLFbq0wpsgpOqxpxtxFiCG4DtG93M6XRVbF2oGdev7bg==", - "dev": true, - "requires": { - "global-prefix": "^1.0.1", - "is-windows": "^1.0.1", - "resolve-dir": "^1.0.0" - } - } - } - }, - "resolve-from": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", - "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", - "dev": true - }, - "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 - }, - "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 - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "ripemd160": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", - "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", - "dev": true, - "requires": { - "hash-base": "^3.0.0", - "inherits": "^2.0.1" - } - }, - "rot-js": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/rot-js/-/rot-js-2.1.1.tgz", - "integrity": "sha512-CtqgIm/gM2Rfo2yi4AktYcVWD0VU2UxmJsV9F+aI0LBI6Y4jhsX/SlLEDdmVqY3YmMUw8jWKONYfPb/jOfS3rQ==" - }, - "run-queue": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/run-queue/-/run-queue-1.0.3.tgz", - "integrity": "sha1-6Eg5bwV9Ij8kOGkkYY4laUFh7Ec=", - "dev": true, - "requires": { - "aproba": "^1.1.1" - } - }, - "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 - }, - "safe-regex": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", - "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", - "dev": true, - "requires": { - "ret": "~0.1.10" - } - }, - "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", - "dev": true, - "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "serialize-javascript": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-2.1.2.tgz", - "integrity": "sha512-rs9OggEUF0V4jUSecXazOYsLfu7OGK2qIn3c7IPBiffz32XniEp/TX9Xmc9LQfK2nQ2QKHvZ2oygKUGU0lG4jQ==", - "dev": true - }, - "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 - }, - "set-value": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", - "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-extendable": "^0.1.1", - "is-plain-object": "^2.0.3", - "split-string": "^3.0.1" - }, - "dependencies": { - "extend-shallow": { - "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" - } - } - } - }, - "setimmediate": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", - "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", - "dev": true - }, - "sha.js": { - "version": "2.4.11", - "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", - "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "shebang-command": { - "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" - } - }, - "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 - }, - "signal-exit": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz", - "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=", - "dev": true - }, - "snapdragon": { - "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", - "define-property": "^0.2.5", - "extend-shallow": "^2.0.1", - "map-cache": "^0.2.2", - "source-map": "^0.5.6", - "source-map-resolve": "^0.5.0", - "use": "^3.1.0" - }, - "dependencies": { - "define-property": { - "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" - } - }, - "extend-shallow": { - "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" - } - } - } - }, - "snapdragon-node": { - "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", - "snapdragon-util": "^3.0.1" - }, - "dependencies": { - "define-property": { - "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" - } - }, - "is-accessor-descriptor": { - "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" - } - }, - "is-data-descriptor": { - "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" - } - }, - "is-descriptor": { - "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", - "kind-of": "^6.0.2" - } - } - } - }, - "snapdragon-util": { - "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" - }, - "dependencies": { - "kind-of": { - "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" - } - } - } - }, - "source-list-map": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz", - "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==", - "dev": true - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - }, - "source-map-resolve": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", - "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", - "dev": true, - "requires": { - "atob": "^2.1.2", - "decode-uri-component": "^0.2.0", - "resolve-url": "^0.2.1", - "source-map-url": "^0.4.0", - "urix": "^0.1.0" - } - }, - "source-map-support": { - "version": "0.5.16", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.16.tgz", - "integrity": "sha512-efyLRJDr68D9hBBNIPWFjhpFzURh+KJykQwvMyW5UiZzYwoF6l4YMMDIJJEyFWxWCqfyxLzz6tSfUFR+kXXsVQ==", - "dev": true, - "requires": { - "buffer-from": "^1.0.0", - "source-map": "^0.6.0" - }, - "dependencies": { - "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 - } - } - }, - "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 - }, - "split-string": { - "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" - } - }, - "ssri": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.1.tgz", - "integrity": "sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - }, - "static-extend": { - "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" - }, - "dependencies": { - "define-property": { - "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" - } - } - } - }, - "stream-browserify": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/stream-browserify/-/stream-browserify-2.0.2.tgz", - "integrity": "sha512-nX6hmklHs/gr2FuxYDltq8fJA1GDlxKQCz8O/IM4atRqBH8OORmBNgfvW5gG10GT/qQ9u0CzIvr2X5Pkt6ntqg==", - "dev": true, - "requires": { - "inherits": "~2.0.1", - "readable-stream": "^2.0.2" - } - }, - "stream-each": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/stream-each/-/stream-each-1.2.3.tgz", - "integrity": "sha512-vlMC2f8I2u/bZGqkdfLQW/13Zihpej/7PmSiMQsbYddxuTsJp8vRe2x2FvVExZg7FaOds43ROAuFJwPR4MTZLw==", - "dev": true, - "requires": { - "end-of-stream": "^1.1.0", - "stream-shift": "^1.0.0" - } - }, - "stream-http": { - "version": "2.8.3", - "resolved": "https://registry.npmjs.org/stream-http/-/stream-http-2.8.3.tgz", - "integrity": "sha512-+TSkfINHDo4J+ZobQLWiMouQYB+UVYFttRA94FpEzzJ7ZdqcL4uUUQ7WkdkI4DSozGmgBUE/a47L+38PenXhUw==", - "dev": true, - "requires": { - "builtin-status-codes": "^3.0.0", - "inherits": "^2.0.1", - "readable-stream": "^2.3.6", - "to-arraybuffer": "^1.0.0", - "xtend": "^4.0.0" - } - }, - "stream-shift": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/stream-shift/-/stream-shift-1.0.1.tgz", - "integrity": "sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ==", - "dev": true - }, - "string-width": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", - "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", - "dev": true, - "requires": { - "emoji-regex": "^7.0.1", - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^5.1.0" - } - }, - "string_decoder": { - "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" - } - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } - }, - "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 - }, - "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==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - }, - "tapable": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", - "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", - "dev": true - }, - "terser": { - "version": "4.6.3", - "resolved": "https://registry.npmjs.org/terser/-/terser-4.6.3.tgz", - "integrity": "sha512-Lw+ieAXmY69d09IIc/yqeBqXpEQIpDGZqT34ui1QWXIUpR2RjbqEkT8X7Lgex19hslSqcWM5iMN2kM11eMsESQ==", - "dev": true, - "requires": { - "commander": "^2.20.0", - "source-map": "~0.6.1", - "source-map-support": "~0.5.12" - }, - "dependencies": { - "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 - } - } - }, - "terser-webpack-plugin": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-1.4.3.tgz", - "integrity": "sha512-QMxecFz/gHQwteWwSo5nTc6UaICqN1bMedC5sMtUc7y3Ha3Q8y6ZO0iCR8pq4RJC8Hjf0FEPEHZqcMB/+DFCrA==", - "dev": true, - "requires": { - "cacache": "^12.0.2", - "find-cache-dir": "^2.1.0", - "is-wsl": "^1.1.0", - "schema-utils": "^1.0.0", - "serialize-javascript": "^2.1.2", - "source-map": "^0.6.1", - "terser": "^4.1.2", - "webpack-sources": "^1.4.0", - "worker-farm": "^1.7.0" - }, - "dependencies": { - "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 - } - } - }, - "through2": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", - "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", - "dev": true, - "requires": { - "readable-stream": "~2.3.6", - "xtend": "~4.0.1" - } - }, - "timers-browserify": { - "version": "2.0.11", - "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-2.0.11.tgz", - "integrity": "sha512-60aV6sgJ5YEbzUdn9c8kYGIqOubPoUdqQCul3SBAsRCZ40s6Y5cMcrW4dt3/k/EsbLVJNl9n6Vz3fTc+k2GeKQ==", - "dev": true, - "requires": { - "setimmediate": "^1.0.4" - } - }, - "to-arraybuffer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", - "integrity": "sha1-fSKbH8xjfkZsoIEYCDanqr/4P0M=", - "dev": true - }, - "to-object-path": { - "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" - }, - "dependencies": { - "kind-of": { - "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" - } - } - } - }, - "to-regex": { - "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", - "regex-not": "^1.0.2", - "safe-regex": "^1.1.0" - } - }, - "to-regex-range": { - "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" - } - }, - "tslib": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.10.0.tgz", - "integrity": "sha512-qOebF53frne81cf0S9B41ByenJ3/IuH8yJKngAX35CmiZySA0khhkovshKK+jGCaMnVomla7gVlIcc3EvKPbTQ==", - "dev": true - }, - "tty-browserify": { - "version": "0.0.0", - "resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.0.tgz", - "integrity": "sha1-oVe6QC2iTpv5V/mqadUk7tQpAaY=", - "dev": true - }, - "typedarray": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", - "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", - "dev": true - }, - "union-value": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", - "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", - "dev": true, - "requires": { - "arr-union": "^3.1.0", - "get-value": "^2.0.6", - "is-extendable": "^0.1.1", - "set-value": "^2.0.1" - } - }, - "unique-filename": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-1.1.1.tgz", - "integrity": "sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==", - "dev": true, - "requires": { - "unique-slug": "^2.0.0" - } - }, - "unique-slug": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/unique-slug/-/unique-slug-2.0.2.tgz", - "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", - "dev": true, - "requires": { - "imurmurhash": "^0.1.4" - } - }, - "unset-value": { - "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" - }, - "dependencies": { - "has-value": { - "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", - "isobject": "^2.0.0" - }, - "dependencies": { - "isobject": { - "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" - } - } - } - }, - "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 - } - } - }, - "upath": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", - "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", - "dev": true - }, - "uri-js": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", - "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", - "dev": true, - "requires": { - "punycode": "^2.1.0" - } - }, - "urix": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", - "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", - "dev": true - }, - "url": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", - "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", - "dev": true, - "requires": { - "punycode": "1.3.2", - "querystring": "0.2.0" - }, - "dependencies": { - "punycode": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", - "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", - "dev": true - } - } - }, - "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 - }, - "util": { - "version": "0.11.1", - "resolved": "https://registry.npmjs.org/util/-/util-0.11.1.tgz", - "integrity": "sha512-HShAsny+zS2TZfaXxD9tYj4HQGlBezXZMZuM/S5PKLLoZkShZiGk9o5CzukI1LVHZvjdvZ2Sj1aW/Ndn2NB/HQ==", - "dev": true, - "requires": { - "inherits": "2.0.3" - }, - "dependencies": { - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true - } - } - }, - "util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", - "dev": true - }, - "v8-compile-cache": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.0.3.tgz", - "integrity": "sha512-CNmdbwQMBjwr9Gsmohvm0pbL954tJrNzf6gWL3K+QMQf00PF7ERGrEiLgjuU3mKreLC2MeGhUsNV9ybTbLgd3w==", - "dev": true - }, - "vm-browserify": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vm-browserify/-/vm-browserify-1.1.2.tgz", - "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==", - "dev": true - }, - "watchpack": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.6.0.tgz", - "integrity": "sha512-i6dHe3EyLjMmDlU1/bGQpEw25XSjkJULPuAVKCbNRefQVq48yXKUpwg538F7AZTf9kyr57zj++pQFltUa5H7yA==", - "dev": true, - "requires": { - "chokidar": "^2.0.2", - "graceful-fs": "^4.1.2", - "neo-async": "^2.5.0" - } - }, - "webpack": { - "version": "4.41.5", - "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.41.5.tgz", - "integrity": "sha512-wp0Co4vpyumnp3KlkmpM5LWuzvZYayDwM2n17EHFr4qxBBbRokC7DJawPJC7TfSFZ9HZ6GsdH40EBj4UV0nmpw==", - "dev": true, - "requires": { - "@webassemblyjs/ast": "1.8.5", - "@webassemblyjs/helper-module-context": "1.8.5", - "@webassemblyjs/wasm-edit": "1.8.5", - "@webassemblyjs/wasm-parser": "1.8.5", - "acorn": "^6.2.1", - "ajv": "^6.10.2", - "ajv-keywords": "^3.4.1", - "chrome-trace-event": "^1.0.2", - "enhanced-resolve": "^4.1.0", - "eslint-scope": "^4.0.3", - "json-parse-better-errors": "^1.0.2", - "loader-runner": "^2.4.0", - "loader-utils": "^1.2.3", - "memory-fs": "^0.4.1", - "micromatch": "^3.1.10", - "mkdirp": "^0.5.1", - "neo-async": "^2.6.1", - "node-libs-browser": "^2.2.1", - "schema-utils": "^1.0.0", - "tapable": "^1.1.3", - "terser-webpack-plugin": "^1.4.3", - "watchpack": "^1.6.0", - "webpack-sources": "^1.4.1" - } - }, - "webpack-cli": { - "version": "3.3.10", - "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-3.3.10.tgz", - "integrity": "sha512-u1dgND9+MXaEt74sJR4PR7qkPxXUSQ0RXYq8x1L6Jg1MYVEmGPrH6Ah6C4arD4r0J1P5HKjRqpab36k0eIzPqg==", - "dev": true, - "requires": { - "chalk": "2.4.2", - "cross-spawn": "6.0.5", - "enhanced-resolve": "4.1.0", - "findup-sync": "3.0.0", - "global-modules": "2.0.0", - "import-local": "2.0.0", - "interpret": "1.2.0", - "loader-utils": "1.2.3", - "supports-color": "6.1.0", - "v8-compile-cache": "2.0.3", - "yargs": "13.2.4" - }, - "dependencies": { - "enhanced-resolve": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.1.0.tgz", - "integrity": "sha512-F/7vkyTtyc/llOIn8oWclcB25KdRaiPBpZYDgJHgh/UHtpgT2p2eldQgtQnLtUvfMKPKxbRaQM/hHkvLHt1Vng==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "memory-fs": "^0.4.0", - "tapable": "^1.0.0" - } - } - } - }, - "webpack-sources": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", - "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", - "dev": true, - "requires": { - "source-list-map": "^2.0.0", - "source-map": "~0.6.1" - }, - "dependencies": { - "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 - } - } - }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } - }, - "which-module": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", - "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", - "dev": true - }, - "worker-farm": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/worker-farm/-/worker-farm-1.7.0.tgz", - "integrity": "sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw==", - "dev": true, - "requires": { - "errno": "~0.1.7" - } - }, - "wrap-ansi": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", - "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", - "dev": true, - "requires": { - "ansi-styles": "^3.2.0", - "string-width": "^3.0.0", - "strip-ansi": "^5.0.0" - } - }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "dev": true - }, - "xtend": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", - "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", - "dev": true - }, - "y18n": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", - "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", - "dev": true - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - }, - "yargs": { - "version": "13.2.4", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.2.4.tgz", - "integrity": "sha512-HG/DWAJa1PAnHT9JAhNa8AbAv3FPaiLzioSjCcmuXXhP8MlpHO5vwls4g4j6n30Z74GVQj8Xa62dWVx1QCGklg==", - "dev": true, - "requires": { - "cliui": "^5.0.0", - "find-up": "^3.0.0", - "get-caller-file": "^2.0.1", - "os-locale": "^3.1.0", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^3.0.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^13.1.0" - } - }, - "yargs-parser": { - "version": "13.1.1", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.1.tgz", - "integrity": "sha512-oVAVsHz6uFrg3XQheFII8ESO2ssAf9luWuAd6Wexsu4F3OtIW0o8IribPXYrD4WC24LWtPrJlGy87y5udK+dxQ==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } - } - } -} From acf3308e66b3a51c6ef77dd623ff5824b72ef253 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:22:57 -0700 Subject: [PATCH 03/12] bump v0.6.0 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 38fb34e..63c3f0b 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "rote", - "version": "0.5.0", + "version": "0.6.0", "description": "Tools for making Rogue-like games", "private": true, "main": "src/index.js", From 16ecad7b0f18a94ccc4b2e2c83b94510135b7658 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:27:33 -0700 Subject: [PATCH 04/12] add hook for afterHeroDeath --- src/Game.js | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/src/Game.js b/src/Game.js index d6332d1..97815e4 100644 --- a/src/Game.js +++ b/src/Game.js @@ -139,7 +139,7 @@ class Game { showInventory() { const items = this.hero.inventory.getString(); - this.print('Inventory: ' + items); + this.print('Inventory: ' + items); } draw() { @@ -297,7 +297,7 @@ class Game { // TODO: allow pushes based on authority/size } else { // just blocked return { x, y, moved: false }; - } + } } static canBumpSwitch(actor, blocker) { @@ -384,8 +384,9 @@ class Game { const isDamaged = (startHp > this.hero.hp); this.display.drawDamage(isDamaged); if (this.hero.dead()) { - this.print('R.I.P. Congratulations! YOU HAVE DIED!', 'plot'); - this.print('Reload the page to play again.', 'tip'); + this.hook('afterHeroDeath', {}); + // this.print('R.I.P. Congratulations! YOU HAVE DIED!', 'plot'); + // this.print('Reload the page to play again.', 'tip'); } this.draw(); } From 0077032c76f95c081061c33e36e4e9f4d28a0ed4 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:36:37 -0700 Subject: [PATCH 05/12] code cleanup --- src/Game.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/Game.js b/src/Game.js index 97815e4..c3c95c6 100644 --- a/src/Game.js +++ b/src/Game.js @@ -385,8 +385,6 @@ class Game { this.display.drawDamage(isDamaged); if (this.hero.dead()) { this.hook('afterHeroDeath', {}); - // this.print('R.I.P. Congratulations! YOU HAVE DIED!', 'plot'); - // this.print('Reload the page to play again.', 'tip'); } this.draw(); } From 99c35aeddb796dca3e9d2cd6f10f2c5adcb0f035 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:42:05 -0700 Subject: [PATCH 06/12] add support for getArmorDefense --- src/Actor.js | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/src/Actor.js b/src/Actor.js index 6e51864..f8c1661 100644 --- a/src/Actor.js +++ b/src/Actor.js @@ -299,7 +299,7 @@ class Actor { const costs = Object.keys(ability.readyCost); costs.forEach((key) => { fn(key, parseInt(ability.readyCost[key], 10)); - }); + }); } static getAbilityEffectsString(ability) { @@ -364,6 +364,19 @@ class Actor { return null; // ? } + getArmorDefense() { + if (!this.isHero) { + return 1; // TODO: change this so there is some kind of natural defense for monsters + } + let highestDefense = 0; + this.inventory.loopOverContents((item) => { + if ((item.armor || item.defense) > highestDefense) { + highestDefense = item.armor || item.defense; + } + }); + return highestDefense; + } + getWeaponDamage() { if (!this.isHero) { return 1; // TODO: change this so there is some kind of natural damage for monsters From 5f8356ea9ed226802e66c3de2fb114a3c7756ccf Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 14:51:25 -0700 Subject: [PATCH 07/12] add support for defense --- src/Actor.js | 4 ++-- src/Item.js | 2 ++ 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/src/Actor.js b/src/Actor.js index f8c1661..2824178 100644 --- a/src/Actor.js +++ b/src/Actor.js @@ -370,8 +370,8 @@ class Actor { } let highestDefense = 0; this.inventory.loopOverContents((item) => { - if ((item.armor || item.defense) > highestDefense) { - highestDefense = item.armor || item.defense; + if (item.defense > highestDefense) { + highestDefense = item.defense; } }); return highestDefense; diff --git a/src/Item.js b/src/Item.js index a9396ab..683af83 100644 --- a/src/Item.js +++ b/src/Item.js @@ -16,6 +16,8 @@ class Item { }); this.isWeapon = Boolean(options.weapon); this.damage = parseInt(options.weapon, 10) || 0; + this.isDefense = Boolean(options.defense); + this.defense = parseInt(options.defense, 10) || 0; this.illumination = options.illumination || 0; this.portable = (typeof options.portable === 'boolean') ? options.portable : true; this.containedIn = null; From 2079da249a6d0872ad54733099d320384cb01cbd Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 15:39:21 -0700 Subject: [PATCH 08/12] update teleport message --- src/Game.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Game.js b/src/Game.js index c3c95c6..5c63bbd 100644 --- a/src/Game.js +++ b/src/Game.js @@ -418,7 +418,7 @@ class Game { message = outcome.message; break; case 'teleport': - message = `${actor.name} travels to a new location: `; + message = `${actor.name} now entering: `; this.teleportActor(actor, action.teleport); { const newLevel = this.getActiveLevel(); From c65fc9b270c42db23086b7003c66be352443dd2d Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 15:50:43 -0700 Subject: [PATCH 09/12] add currency support --- src/Actor.js | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/Actor.js b/src/Actor.js index 2824178..b7509a9 100644 --- a/src/Actor.js +++ b/src/Actor.js @@ -54,6 +54,12 @@ class Actor { this.abilityList = []; // temporary this.initiativeBoost = 0; + + // all actors can carry currency + // everything is boiled down to copper pieces + // 100 copper = 1 silver + // 100 silver = 1 gold + this.currency = 0; } draw(display, lighting = {}, inView = false) { From 352d2823f16e9d5ab4028a6c4d7078defdbc5993 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 16:27:54 -0700 Subject: [PATCH 10/12] allow for options.currency --- src/Actor.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Actor.js b/src/Actor.js index b7509a9..dd801c0 100644 --- a/src/Actor.js +++ b/src/Actor.js @@ -59,7 +59,7 @@ class Actor { // everything is boiled down to copper pieces // 100 copper = 1 silver // 100 silver = 1 gold - this.currency = 0; + this.currency = options.currency || 0; } draw(display, lighting = {}, inView = false) { From 104a0ebb70ded4485f48738d91f977e376d545b5 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 19:04:41 -0700 Subject: [PATCH 11/12] reworked some constants / enums --- src/Actor.js | 40 +++++++++++++++++++++++++++++++--------- 1 file changed, 31 insertions(+), 9 deletions(-) diff --git a/src/Actor.js b/src/Actor.js index dd801c0..a9fb2be 100644 --- a/src/Actor.js +++ b/src/Actor.js @@ -3,15 +3,37 @@ const Inventory = require('./Inventory'); const geometer = require('./geometer'); const random = require('./random'); -const MOVE = 'move'; -const WAIT = 'wait'; -const MONSTER_FACTION = 'monsters'; +class ActionVerbType { + static get MOVE () { + return 'move'; + } + + static get WAIT () { + return 'wait'; + } +} + +class FactionType { + + static get HUMAN () { + return 'human'; + } + + static get MONSTER () { + return 'monster'; + } + + static get NEUTRAL () { + return 'neutral'; + } +} + class Actor { constructor(options = {}) { this.type = options.type; this.name = options.name || null; - this.faction = options.faction || MONSTER_FACTION; + this.faction = options.faction || FactionType.MONSTER; this.isHero = Boolean(options.isHero); this.x = options.x || 0; this.y = options.y || 0; @@ -113,11 +135,11 @@ class Actor { doAction() { if (this.dead()) { return { verb: 'rot' }; } - const waitAction = { verb: WAIT }; + const waitAction = { verb: ActionVerbType.WAIT }; if (this.actionQueue.length === 0) { return waitAction; } let action = this.actionQueue.shift(); - const moveAlreadyThere = (action.verb === MOVE && action.x === this.x && action.y === this.y); - const moveTooFar = (action.verb === MOVE && this.getDistanceToNextMove(action) > this.maxMovement); + const moveAlreadyThere = (action.verb === ActionVerbType.MOVE && action.x === this.x && action.y === this.y); + const moveTooFar = (action.verb === ActionVerbType.MOVE && this.getDistanceToNextMove(action) > this.maxMovement); // console.log(this.name, this.x, this.y, action.verb, action.x, action.y, this.getDistanceToNextMove(), this.maxMovement, moveTooFar, 'q', this.actionQueue.length); if (moveAlreadyThere) { return this.doAction(); @@ -145,7 +167,7 @@ class Actor { atEndOfPath() { const nextAction = this.getNextAction(); if (!nextAction) { return true; } - return (nextAction.verb === MOVE) ? false : true; + return (nextAction.verb === ActionVerbType.MOVE) ? false : true; } wait() { @@ -410,7 +432,7 @@ class Actor { const astar = new ROT.Path.AStar(x, y, passableCallback, { topology: 4 }); const path = this.actionQueue; const pathCallback = function(x, y) { - path.push({ x, y, verb: MOVE }); + path.push({ x, y, verb: ActionVerbType.MOVE }); }; if (path[0] && path[0].x === this.x && path[0].y === this.y) { console.alert('removing first'); From 638cd0f7266e37fcd5c4f09b7bf5be8962062b84 Mon Sep 17 00:00:00 2001 From: erik yuzwa Date: Fri, 12 Mar 2021 20:17:28 -0700 Subject: [PATCH 12/12] latest build --- dist/rote-0.6.0.js | 8524 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 8524 insertions(+) create mode 100644 dist/rote-0.6.0.js diff --git a/dist/rote-0.6.0.js b/dist/rote-0.6.0.js new file mode 100644 index 0000000..72b052d --- /dev/null +++ b/dist/rote-0.6.0.js @@ -0,0 +1,8524 @@ +/******/ (function(modules) { // webpackBootstrap +/******/ // The module cache +/******/ var installedModules = {}; +/******/ +/******/ // The require function +/******/ function __webpack_require__(moduleId) { +/******/ +/******/ // Check if module is in cache +/******/ if(installedModules[moduleId]) { +/******/ return installedModules[moduleId].exports; +/******/ } +/******/ // Create a new module (and put it into the cache) +/******/ var module = installedModules[moduleId] = { +/******/ i: moduleId, +/******/ l: false, +/******/ exports: {} +/******/ }; +/******/ +/******/ // Execute the module function +/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); +/******/ +/******/ // Flag the module as loaded +/******/ module.l = true; +/******/ +/******/ // Return the exports of the module +/******/ return module.exports; +/******/ } +/******/ +/******/ +/******/ // expose the modules object (__webpack_modules__) +/******/ __webpack_require__.m = modules; +/******/ +/******/ // expose the module cache +/******/ __webpack_require__.c = installedModules; +/******/ +/******/ // define getter function for harmony exports +/******/ __webpack_require__.d = function(exports, name, getter) { +/******/ if(!__webpack_require__.o(exports, name)) { +/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter }); +/******/ } +/******/ }; +/******/ +/******/ // define __esModule on exports +/******/ __webpack_require__.r = function(exports) { +/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { +/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); +/******/ } +/******/ Object.defineProperty(exports, '__esModule', { value: true }); +/******/ }; +/******/ +/******/ // create a fake namespace object +/******/ // mode & 1: value is a module id, require it +/******/ // mode & 2: merge all properties of value into the ns +/******/ // mode & 4: return value when already ns object +/******/ // mode & 8|1: behave like require +/******/ __webpack_require__.t = function(value, mode) { +/******/ if(mode & 1) value = __webpack_require__(value); +/******/ if(mode & 8) return value; +/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value; +/******/ var ns = Object.create(null); +/******/ __webpack_require__.r(ns); +/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value }); +/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key)); +/******/ return ns; +/******/ }; +/******/ +/******/ // getDefaultExport function for compatibility with non-harmony modules +/******/ __webpack_require__.n = function(module) { +/******/ var getter = module && module.__esModule ? +/******/ function getDefault() { return module['default']; } : +/******/ function getModuleExports() { return module; }; +/******/ __webpack_require__.d(getter, 'a', getter); +/******/ return getter; +/******/ }; +/******/ +/******/ // Object.prototype.hasOwnProperty.call +/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); }; +/******/ +/******/ // __webpack_public_path__ +/******/ __webpack_require__.p = ""; +/******/ +/******/ +/******/ // Load entry module and return exports +/******/ return __webpack_require__(__webpack_require__.s = 17); +/******/ }) +/************************************************************************/ +/******/ ([ +/* 0 */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +/** + * This code is an implementation of Alea algorithm; (C) 2010 Johannes Baagøe. + * Alea is licensed according to the http://en.wikipedia.org/wiki/MIT_License. + */ +const FRAC = 2.3283064365386963e-10; /* 2^-32 */ +class RNG { + constructor() { + this._seed = 0; + this._s0 = 0; + this._s1 = 0; + this._s2 = 0; + this._c = 0; + } + getSeed() { return this._seed; } + /** + * Seed the number generator + */ + setSeed(seed) { + seed = (seed < 1 ? 1 / seed : seed); + this._seed = seed; + this._s0 = (seed >>> 0) * FRAC; + seed = (seed * 69069 + 1) >>> 0; + this._s1 = seed * FRAC; + seed = (seed * 69069 + 1) >>> 0; + this._s2 = seed * FRAC; + this._c = 1; + return this; + } + /** + * @returns Pseudorandom value [0,1), uniformly distributed + */ + getUniform() { + let t = 2091639 * this._s0 + this._c * FRAC; + this._s0 = this._s1; + this._s1 = this._s2; + this._c = t | 0; + this._s2 = t - this._c; + return this._s2; + } + /** + * @param lowerBound The lower end of the range to return a value from, inclusive + * @param upperBound The upper end of the range to return a value from, inclusive + * @returns Pseudorandom value [lowerBound, upperBound], using ROT.RNG.getUniform() to distribute the value + */ + getUniformInt(lowerBound, upperBound) { + let max = Math.max(lowerBound, upperBound); + let min = Math.min(lowerBound, upperBound); + return Math.floor(this.getUniform() * (max - min + 1)) + min; + } + /** + * @param mean Mean value + * @param stddev Standard deviation. ~95% of the absolute values will be lower than 2*stddev. + * @returns A normally distributed pseudorandom value + */ + getNormal(mean = 0, stddev = 1) { + let u, v, r; + do { + u = 2 * this.getUniform() - 1; + v = 2 * this.getUniform() - 1; + r = u * u + v * v; + } while (r > 1 || r == 0); + let gauss = u * Math.sqrt(-2 * Math.log(r) / r); + return mean + gauss * stddev; + } + /** + * @returns Pseudorandom value [1,100] inclusive, uniformly distributed + */ + getPercentage() { + return 1 + Math.floor(this.getUniform() * 100); + } + /** + * @returns Randomly picked item, null when length=0 + */ + getItem(array) { + if (!array.length) { + return null; + } + return array[Math.floor(this.getUniform() * array.length)]; + } + /** + * @returns New array with randomized items + */ + shuffle(array) { + let result = []; + let clone = array.slice(); + while (clone.length) { + let index = clone.indexOf(this.getItem(clone)); + result.push(clone.splice(index, 1)[0]); + } + return result; + } + /** + * @param data key=whatever, value=weight (relative probability) + * @returns whatever + */ + getWeightedValue(data) { + let total = 0; + for (let id in data) { + total += data[id]; + } + let random = this.getUniform() * total; + let id, part = 0; + for (id in data) { + part += data[id]; + if (random < part) { + return id; + } + } + // If by some floating-point annoyance we have + // random >= total, just return the last id. + return id; + } + /** + * Get RNG state. Useful for storing the state and re-setting it via setState. + * @returns Internal state + */ + getState() { return [this._s0, this._s1, this._s2, this._c]; } + /** + * Set a previously retrieved state. + */ + setState(state) { + this._s0 = state[0]; + this._s1 = state[1]; + this._s2 = state[2]; + this._c = state[3]; + return this; + } + /** + * Returns a cloned RNG + */ + clone() { + let clone = new RNG(); + return clone.setState(this.getState()); + } +} +/* harmony default export */ __webpack_exports__["a"] = (new RNG().setSeed(Date.now())); + + +/***/ }), +/* 1 */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +__webpack_require__.r(__webpack_exports__); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "mod", function() { return mod; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "clamp", function() { return clamp; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "capitalize", function() { return capitalize; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "format", function() { return format; }); +/** + * Always positive modulus + * @param x Operand + * @param n Modulus + * @returns x modulo n + */ +function mod(x, n) { + return (x % n + n) % n; +} +function clamp(val, min = 0, max = 1) { + if (val < min) + return min; + if (val > max) + return max; + return val; +} +function capitalize(string) { + return string.charAt(0).toUpperCase() + string.substring(1); +} +/** + * Format a string in a flexible way. Scans for %s strings and replaces them with arguments. List of patterns is modifiable via String.format.map. + * @param {string} template + * @param {any} [argv] + */ +function format(template, ...args) { + let map = format.map; + let replacer = function (match, group1, group2, index) { + if (template.charAt(index - 1) == "%") { + return match.substring(1); + } + if (!args.length) { + return match; + } + let obj = args[0]; + let group = group1 || group2; + let parts = group.split(","); + let name = parts.shift() || ""; + let method = map[name.toLowerCase()]; + if (!method) { + return match; + } + obj = args.shift(); + let replaced = obj[method].apply(obj, parts); + let first = name.charAt(0); + if (first != first.toLowerCase()) { + replaced = capitalize(replaced); + } + return replaced; + }; + return template.replace(/%(?:([a-z]+)|(?:{([^}]+)}))/gi, replacer); +} +format.map = { + "s": "toString" +}; + + +/***/ }), +/* 2 */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +__webpack_require__.r(__webpack_exports__); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "fromString", function() { return fromString; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "add", function() { return add; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "add_", function() { return add_; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "multiply", function() { return multiply; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "multiply_", function() { return multiply_; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "interpolate", function() { return interpolate; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "lerp", function() { return lerp; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "interpolateHSL", function() { return interpolateHSL; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "lerpHSL", function() { return lerpHSL; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "randomize", function() { return randomize; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "rgb2hsl", function() { return rgb2hsl; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "hsl2rgb", function() { return hsl2rgb; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "toRGB", function() { return toRGB; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "toHex", function() { return toHex; }); +/* harmony import */ var _util_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(1); +/* harmony import */ var _rng_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(0); + + +function fromString(str) { + let cached, r; + if (str in CACHE) { + cached = CACHE[str]; + } + else { + if (str.charAt(0) == "#") { // hex rgb + let matched = str.match(/[0-9a-f]/gi) || []; + let values = matched.map((x) => parseInt(x, 16)); + if (values.length == 3) { + cached = values.map((x) => x * 17); + } + else { + for (let i = 0; i < 3; i++) { + values[i + 1] += 16 * values[i]; + values.splice(i, 1); + } + cached = values; + } + } + else if ((r = str.match(/rgb\(([0-9, ]+)\)/i))) { // decimal rgb + cached = r[1].split(/\s*,\s*/).map((x) => parseInt(x)); + } + else { // html name + cached = [0, 0, 0]; + } + CACHE[str] = cached; + } + return cached.slice(); +} +/** + * Add two or more colors + */ +function add(color1, ...colors) { + let result = color1.slice(); + for (let i = 0; i < 3; i++) { + for (let j = 0; j < colors.length; j++) { + result[i] += colors[j][i]; + } + } + return result; +} +/** + * Add two or more colors, MODIFIES FIRST ARGUMENT + */ +function add_(color1, ...colors) { + for (let i = 0; i < 3; i++) { + for (let j = 0; j < colors.length; j++) { + color1[i] += colors[j][i]; + } + } + return color1; +} +/** + * Multiply (mix) two or more colors + */ +function multiply(color1, ...colors) { + let result = color1.slice(); + for (let i = 0; i < 3; i++) { + for (let j = 0; j < colors.length; j++) { + result[i] *= colors[j][i] / 255; + } + result[i] = Math.round(result[i]); + } + return result; +} +/** + * Multiply (mix) two or more colors, MODIFIES FIRST ARGUMENT + */ +function multiply_(color1, ...colors) { + for (let i = 0; i < 3; i++) { + for (let j = 0; j < colors.length; j++) { + color1[i] *= colors[j][i] / 255; + } + color1[i] = Math.round(color1[i]); + } + return color1; +} +/** + * Interpolate (blend) two colors with a given factor + */ +function interpolate(color1, color2, factor = 0.5) { + let result = color1.slice(); + for (let i = 0; i < 3; i++) { + result[i] = Math.round(result[i] + factor * (color2[i] - color1[i])); + } + return result; +} +const lerp = interpolate; +/** + * Interpolate (blend) two colors with a given factor in HSL mode + */ +function interpolateHSL(color1, color2, factor = 0.5) { + let hsl1 = rgb2hsl(color1); + let hsl2 = rgb2hsl(color2); + for (let i = 0; i < 3; i++) { + hsl1[i] += factor * (hsl2[i] - hsl1[i]); + } + return hsl2rgb(hsl1); +} +const lerpHSL = interpolateHSL; +/** + * Create a new random color based on this one + * @param color + * @param diff Set of standard deviations + */ +function randomize(color, diff) { + if (!(diff instanceof Array)) { + diff = Math.round(_rng_js__WEBPACK_IMPORTED_MODULE_1__[/* default */ "a"].getNormal(0, diff)); + } + let result = color.slice(); + for (let i = 0; i < 3; i++) { + result[i] += (diff instanceof Array ? Math.round(_rng_js__WEBPACK_IMPORTED_MODULE_1__[/* default */ "a"].getNormal(0, diff[i])) : diff); + } + return result; +} +/** + * Converts an RGB color value to HSL. Expects 0..255 inputs, produces 0..1 outputs. + */ +function rgb2hsl(color) { + let r = color[0] / 255; + let g = color[1] / 255; + let b = color[2] / 255; + let max = Math.max(r, g, b), min = Math.min(r, g, b); + let h = 0, s, l = (max + min) / 2; + if (max == min) { + s = 0; // achromatic + } + else { + let d = max - min; + s = (l > 0.5 ? d / (2 - max - min) : d / (max + min)); + switch (max) { + case r: + h = (g - b) / d + (g < b ? 6 : 0); + break; + case g: + h = (b - r) / d + 2; + break; + case b: + h = (r - g) / d + 4; + break; + } + h /= 6; + } + return [h, s, l]; +} +function hue2rgb(p, q, t) { + if (t < 0) + t += 1; + if (t > 1) + t -= 1; + if (t < 1 / 6) + return p + (q - p) * 6 * t; + if (t < 1 / 2) + return q; + if (t < 2 / 3) + return p + (q - p) * (2 / 3 - t) * 6; + return p; +} +/** + * Converts an HSL color value to RGB. Expects 0..1 inputs, produces 0..255 outputs. + */ +function hsl2rgb(color) { + let l = color[2]; + if (color[1] == 0) { + l = Math.round(l * 255); + return [l, l, l]; + } + else { + let s = color[1]; + let q = (l < 0.5 ? l * (1 + s) : l + s - l * s); + let p = 2 * l - q; + let r = hue2rgb(p, q, color[0] + 1 / 3); + let g = hue2rgb(p, q, color[0]); + let b = hue2rgb(p, q, color[0] - 1 / 3); + return [Math.round(r * 255), Math.round(g * 255), Math.round(b * 255)]; + } +} +function toRGB(color) { + let clamped = color.map(x => Object(_util_js__WEBPACK_IMPORTED_MODULE_0__["clamp"])(x, 0, 255)); + return `rgb(${clamped.join(",")})`; +} +function toHex(color) { + let clamped = color.map(x => Object(_util_js__WEBPACK_IMPORTED_MODULE_0__["clamp"])(x, 0, 255).toString(16).padStart(2, "0")); + return `#${clamped.join("")}`; +} +const CACHE = { + "black": [0, 0, 0], + "navy": [0, 0, 128], + "darkblue": [0, 0, 139], + "mediumblue": [0, 0, 205], + "blue": [0, 0, 255], + "darkgreen": [0, 100, 0], + "green": [0, 128, 0], + "teal": [0, 128, 128], + "darkcyan": [0, 139, 139], + "deepskyblue": [0, 191, 255], + "darkturquoise": [0, 206, 209], + "mediumspringgreen": [0, 250, 154], + "lime": [0, 255, 0], + "springgreen": [0, 255, 127], + "aqua": [0, 255, 255], + "cyan": [0, 255, 255], + "midnightblue": [25, 25, 112], + "dodgerblue": [30, 144, 255], + "forestgreen": [34, 139, 34], + "seagreen": [46, 139, 87], + "darkslategray": [47, 79, 79], + "darkslategrey": [47, 79, 79], + "limegreen": [50, 205, 50], + "mediumseagreen": [60, 179, 113], + "turquoise": [64, 224, 208], + "royalblue": [65, 105, 225], + "steelblue": [70, 130, 180], + "darkslateblue": [72, 61, 139], + "mediumturquoise": [72, 209, 204], + "indigo": [75, 0, 130], + "darkolivegreen": [85, 107, 47], + "cadetblue": [95, 158, 160], + "cornflowerblue": [100, 149, 237], + "mediumaquamarine": [102, 205, 170], + "dimgray": [105, 105, 105], + "dimgrey": [105, 105, 105], + "slateblue": [106, 90, 205], + "olivedrab": [107, 142, 35], + "slategray": [112, 128, 144], + "slategrey": [112, 128, 144], + "lightslategray": [119, 136, 153], + "lightslategrey": [119, 136, 153], + "mediumslateblue": [123, 104, 238], + "lawngreen": [124, 252, 0], + "chartreuse": [127, 255, 0], + "aquamarine": [127, 255, 212], + "maroon": [128, 0, 0], + "purple": [128, 0, 128], + "olive": [128, 128, 0], + "gray": [128, 128, 128], + "grey": [128, 128, 128], + "skyblue": [135, 206, 235], + "lightskyblue": [135, 206, 250], + "blueviolet": [138, 43, 226], + "darkred": [139, 0, 0], + "darkmagenta": [139, 0, 139], + "saddlebrown": [139, 69, 19], + "darkseagreen": [143, 188, 143], + "lightgreen": [144, 238, 144], + "mediumpurple": [147, 112, 216], + "darkviolet": [148, 0, 211], + "palegreen": [152, 251, 152], + "darkorchid": [153, 50, 204], + "yellowgreen": [154, 205, 50], + "sienna": [160, 82, 45], + "brown": [165, 42, 42], + "darkgray": [169, 169, 169], + "darkgrey": [169, 169, 169], + "lightblue": [173, 216, 230], + "greenyellow": [173, 255, 47], + "paleturquoise": [175, 238, 238], + "lightsteelblue": [176, 196, 222], + "powderblue": [176, 224, 230], + "firebrick": [178, 34, 34], + "darkgoldenrod": [184, 134, 11], + "mediumorchid": [186, 85, 211], + "rosybrown": [188, 143, 143], + "darkkhaki": [189, 183, 107], + "silver": [192, 192, 192], + "mediumvioletred": [199, 21, 133], + "indianred": [205, 92, 92], + "peru": [205, 133, 63], + "chocolate": [210, 105, 30], + "tan": [210, 180, 140], + "lightgray": [211, 211, 211], + "lightgrey": [211, 211, 211], + "palevioletred": [216, 112, 147], + "thistle": [216, 191, 216], + "orchid": [218, 112, 214], + "goldenrod": [218, 165, 32], + "crimson": [220, 20, 60], + "gainsboro": [220, 220, 220], + "plum": [221, 160, 221], + "burlywood": [222, 184, 135], + "lightcyan": [224, 255, 255], + "lavender": [230, 230, 250], + "darksalmon": [233, 150, 122], + "violet": [238, 130, 238], + "palegoldenrod": [238, 232, 170], + "lightcoral": [240, 128, 128], + "khaki": [240, 230, 140], + "aliceblue": [240, 248, 255], + "honeydew": [240, 255, 240], + "azure": [240, 255, 255], + "sandybrown": [244, 164, 96], + "wheat": [245, 222, 179], + "beige": [245, 245, 220], + "whitesmoke": [245, 245, 245], + "mintcream": [245, 255, 250], + "ghostwhite": [248, 248, 255], + "salmon": [250, 128, 114], + "antiquewhite": [250, 235, 215], + "linen": [250, 240, 230], + "lightgoldenrodyellow": [250, 250, 210], + "oldlace": [253, 245, 230], + "red": [255, 0, 0], + "fuchsia": [255, 0, 255], + "magenta": [255, 0, 255], + "deeppink": [255, 20, 147], + "orangered": [255, 69, 0], + "tomato": [255, 99, 71], + "hotpink": [255, 105, 180], + "coral": [255, 127, 80], + "darkorange": [255, 140, 0], + "lightsalmon": [255, 160, 122], + "orange": [255, 165, 0], + "lightpink": [255, 182, 193], + "pink": [255, 192, 203], + "gold": [255, 215, 0], + "peachpuff": [255, 218, 185], + "navajowhite": [255, 222, 173], + "moccasin": [255, 228, 181], + "bisque": [255, 228, 196], + "mistyrose": [255, 228, 225], + "blanchedalmond": [255, 235, 205], + "papayawhip": [255, 239, 213], + "lavenderblush": [255, 240, 245], + "seashell": [255, 245, 238], + "cornsilk": [255, 248, 220], + "lemonchiffon": [255, 250, 205], + "floralwhite": [255, 250, 240], + "snow": [255, 250, 250], + "yellow": [255, 255, 0], + "lightyellow": [255, 255, 224], + "ivory": [255, 255, 240], + "white": [255, 255, 255] +}; + + +/***/ }), +/* 3 */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +__webpack_require__.r(__webpack_exports__); +var text_namespaceObject = {}; +__webpack_require__.r(text_namespaceObject); +__webpack_require__.d(text_namespaceObject, "TYPE_TEXT", function() { return TYPE_TEXT; }); +__webpack_require__.d(text_namespaceObject, "TYPE_NEWLINE", function() { return TYPE_NEWLINE; }); +__webpack_require__.d(text_namespaceObject, "TYPE_FG", function() { return TYPE_FG; }); +__webpack_require__.d(text_namespaceObject, "TYPE_BG", function() { return TYPE_BG; }); +__webpack_require__.d(text_namespaceObject, "measure", function() { return measure; }); +__webpack_require__.d(text_namespaceObject, "tokenize", function() { return tokenize; }); + +// EXTERNAL MODULE: ./node_modules/rot-js/lib/rng.js +var rng = __webpack_require__(0); + +// EXTERNAL MODULE: ./node_modules/rot-js/lib/display/backend.js +var backend = __webpack_require__(4); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/display/canvas.js + +class canvas_Canvas extends backend["a" /* default */] { + constructor() { + super(); + this._ctx = document.createElement("canvas").getContext("2d"); + } + schedule(cb) { requestAnimationFrame(cb); } + getContainer() { return this._ctx.canvas; } + setOptions(opts) { + super.setOptions(opts); + const style = (opts.fontStyle ? `${opts.fontStyle} ` : ``); + const font = `${style} ${opts.fontSize}px ${opts.fontFamily}`; + this._ctx.font = font; + this._updateSize(); + this._ctx.font = font; + this._ctx.textAlign = "center"; + this._ctx.textBaseline = "middle"; + } + clear() { + this._ctx.fillStyle = this._options.bg; + this._ctx.fillRect(0, 0, this._ctx.canvas.width, this._ctx.canvas.height); + } + eventToPosition(x, y) { + let canvas = this._ctx.canvas; + let rect = canvas.getBoundingClientRect(); + x -= rect.left; + y -= rect.top; + x *= canvas.width / rect.width; + y *= canvas.height / rect.height; + if (x < 0 || y < 0 || x >= canvas.width || y >= canvas.height) { + return [-1, -1]; + } + return this._normalizedEventToPosition(x, y); + } +} + +// EXTERNAL MODULE: ./node_modules/rot-js/lib/util.js +var util = __webpack_require__(1); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/display/hex.js + + +/** + * @class Hexagonal backend + * @private + */ +class hex_Hex extends canvas_Canvas { + constructor() { + super(); + this._spacingX = 0; + this._spacingY = 0; + this._hexSize = 0; + } + draw(data, clearBefore) { + let [x, y, ch, fg, bg] = data; + let px = [ + (x + 1) * this._spacingX, + y * this._spacingY + this._hexSize + ]; + if (this._options.transpose) { + px.reverse(); + } + if (clearBefore) { + this._ctx.fillStyle = bg; + this._fill(px[0], px[1]); + } + if (!ch) { + return; + } + this._ctx.fillStyle = fg; + let chars = [].concat(ch); + for (let i = 0; i < chars.length; i++) { + this._ctx.fillText(chars[i], px[0], Math.ceil(px[1])); + } + } + computeSize(availWidth, availHeight) { + if (this._options.transpose) { + availWidth += availHeight; + availHeight = availWidth - availHeight; + availWidth -= availHeight; + } + let width = Math.floor(availWidth / this._spacingX) - 1; + let height = Math.floor((availHeight - 2 * this._hexSize) / this._spacingY + 1); + return [width, height]; + } + computeFontSize(availWidth, availHeight) { + if (this._options.transpose) { + availWidth += availHeight; + availHeight = availWidth - availHeight; + availWidth -= availHeight; + } + let hexSizeWidth = 2 * availWidth / ((this._options.width + 1) * Math.sqrt(3)) - 1; + let hexSizeHeight = availHeight / (2 + 1.5 * (this._options.height - 1)); + let hexSize = Math.min(hexSizeWidth, hexSizeHeight); + // compute char ratio + let oldFont = this._ctx.font; + this._ctx.font = "100px " + this._options.fontFamily; + let width = Math.ceil(this._ctx.measureText("W").width); + this._ctx.font = oldFont; + let ratio = width / 100; + hexSize = Math.floor(hexSize) + 1; // closest larger hexSize + // FIXME char size computation does not respect transposed hexes + let fontSize = 2 * hexSize / (this._options.spacing * (1 + ratio / Math.sqrt(3))); + // closest smaller fontSize + return Math.ceil(fontSize) - 1; + } + _normalizedEventToPosition(x, y) { + let nodeSize; + if (this._options.transpose) { + x += y; + y = x - y; + x -= y; + nodeSize = this._ctx.canvas.width; + } + else { + nodeSize = this._ctx.canvas.height; + } + let size = nodeSize / this._options.height; + y = Math.floor(y / size); + if (Object(util["mod"])(y, 2)) { /* odd row */ + x -= this._spacingX; + x = 1 + 2 * Math.floor(x / (2 * this._spacingX)); + } + else { + x = 2 * Math.floor(x / (2 * this._spacingX)); + } + return [x, y]; + } + /** + * Arguments are pixel values. If "transposed" mode is enabled, then these two are already swapped. + */ + _fill(cx, cy) { + let a = this._hexSize; + let b = this._options.border; + const ctx = this._ctx; + ctx.beginPath(); + if (this._options.transpose) { + ctx.moveTo(cx - a + b, cy); + ctx.lineTo(cx - a / 2 + b, cy + this._spacingX - b); + ctx.lineTo(cx + a / 2 - b, cy + this._spacingX - b); + ctx.lineTo(cx + a - b, cy); + ctx.lineTo(cx + a / 2 - b, cy - this._spacingX + b); + ctx.lineTo(cx - a / 2 + b, cy - this._spacingX + b); + ctx.lineTo(cx - a + b, cy); + } + else { + ctx.moveTo(cx, cy - a + b); + ctx.lineTo(cx + this._spacingX - b, cy - a / 2 + b); + ctx.lineTo(cx + this._spacingX - b, cy + a / 2 - b); + ctx.lineTo(cx, cy + a - b); + ctx.lineTo(cx - this._spacingX + b, cy + a / 2 - b); + ctx.lineTo(cx - this._spacingX + b, cy - a / 2 + b); + ctx.lineTo(cx, cy - a + b); + } + ctx.fill(); + } + _updateSize() { + const opts = this._options; + const charWidth = Math.ceil(this._ctx.measureText("W").width); + this._hexSize = Math.floor(opts.spacing * (opts.fontSize + charWidth / Math.sqrt(3)) / 2); + this._spacingX = this._hexSize * Math.sqrt(3) / 2; + this._spacingY = this._hexSize * 1.5; + let xprop; + let yprop; + if (opts.transpose) { + xprop = "height"; + yprop = "width"; + } + else { + xprop = "width"; + yprop = "height"; + } + this._ctx.canvas[xprop] = Math.ceil((opts.width + 1) * this._spacingX); + this._ctx.canvas[yprop] = Math.ceil((opts.height - 1) * this._spacingY + 2 * this._hexSize); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/display/rect.js + +/** + * @class Rectangular backend + * @private + */ +class rect_Rect extends canvas_Canvas { + constructor() { + super(); + this._spacingX = 0; + this._spacingY = 0; + this._canvasCache = {}; + } + setOptions(options) { + super.setOptions(options); + this._canvasCache = {}; + } + draw(data, clearBefore) { + if (rect_Rect.cache) { + this._drawWithCache(data); + } + else { + this._drawNoCache(data, clearBefore); + } + } + _drawWithCache(data) { + let [x, y, ch, fg, bg] = data; + let hash = "" + ch + fg + bg; + let canvas; + if (hash in this._canvasCache) { + canvas = this._canvasCache[hash]; + } + else { + let b = this._options.border; + canvas = document.createElement("canvas"); + let ctx = canvas.getContext("2d"); + canvas.width = this._spacingX; + canvas.height = this._spacingY; + ctx.fillStyle = bg; + ctx.fillRect(b, b, canvas.width - b, canvas.height - b); + if (ch) { + ctx.fillStyle = fg; + ctx.font = this._ctx.font; + ctx.textAlign = "center"; + ctx.textBaseline = "middle"; + let chars = [].concat(ch); + for (let i = 0; i < chars.length; i++) { + ctx.fillText(chars[i], this._spacingX / 2, Math.ceil(this._spacingY / 2)); + } + } + this._canvasCache[hash] = canvas; + } + this._ctx.drawImage(canvas, x * this._spacingX, y * this._spacingY); + } + _drawNoCache(data, clearBefore) { + let [x, y, ch, fg, bg] = data; + if (clearBefore) { + let b = this._options.border; + this._ctx.fillStyle = bg; + this._ctx.fillRect(x * this._spacingX + b, y * this._spacingY + b, this._spacingX - b, this._spacingY - b); + } + if (!ch) { + return; + } + this._ctx.fillStyle = fg; + let chars = [].concat(ch); + for (let i = 0; i < chars.length; i++) { + this._ctx.fillText(chars[i], (x + 0.5) * this._spacingX, Math.ceil((y + 0.5) * this._spacingY)); + } + } + computeSize(availWidth, availHeight) { + let width = Math.floor(availWidth / this._spacingX); + let height = Math.floor(availHeight / this._spacingY); + return [width, height]; + } + computeFontSize(availWidth, availHeight) { + let boxWidth = Math.floor(availWidth / this._options.width); + let boxHeight = Math.floor(availHeight / this._options.height); + /* compute char ratio */ + let oldFont = this._ctx.font; + this._ctx.font = "100px " + this._options.fontFamily; + let width = Math.ceil(this._ctx.measureText("W").width); + this._ctx.font = oldFont; + let ratio = width / 100; + let widthFraction = ratio * boxHeight / boxWidth; + if (widthFraction > 1) { /* too wide with current aspect ratio */ + boxHeight = Math.floor(boxHeight / widthFraction); + } + return Math.floor(boxHeight / this._options.spacing); + } + _normalizedEventToPosition(x, y) { + return [Math.floor(x / this._spacingX), Math.floor(y / this._spacingY)]; + } + _updateSize() { + const opts = this._options; + const charWidth = Math.ceil(this._ctx.measureText("W").width); + this._spacingX = Math.ceil(opts.spacing * charWidth); + this._spacingY = Math.ceil(opts.spacing * opts.fontSize); + if (opts.forceSquareRatio) { + this._spacingX = this._spacingY = Math.max(this._spacingX, this._spacingY); + } + this._ctx.canvas.width = opts.width * this._spacingX; + this._ctx.canvas.height = opts.height * this._spacingY; + } +} +rect_Rect.cache = false; + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/display/tile.js + +/** + * @class Tile backend + * @private + */ +class tile_Tile extends canvas_Canvas { + constructor() { + super(); + this._colorCanvas = document.createElement("canvas"); + } + draw(data, clearBefore) { + let [x, y, ch, fg, bg] = data; + let tileWidth = this._options.tileWidth; + let tileHeight = this._options.tileHeight; + if (clearBefore) { + if (this._options.tileColorize) { + this._ctx.clearRect(x * tileWidth, y * tileHeight, tileWidth, tileHeight); + } + else { + this._ctx.fillStyle = bg; + this._ctx.fillRect(x * tileWidth, y * tileHeight, tileWidth, tileHeight); + } + } + if (!ch) { + return; + } + let chars = [].concat(ch); + let fgs = [].concat(fg); + let bgs = [].concat(bg); + for (let i = 0; i < chars.length; i++) { + let tile = this._options.tileMap[chars[i]]; + if (!tile) { + throw new Error(`Char "${chars[i]}" not found in tileMap`); + } + if (this._options.tileColorize) { // apply colorization + let canvas = this._colorCanvas; + let context = canvas.getContext("2d"); + context.globalCompositeOperation = "source-over"; + context.clearRect(0, 0, tileWidth, tileHeight); + let fg = fgs[i]; + let bg = bgs[i]; + context.drawImage(this._options.tileSet, tile[0], tile[1], tileWidth, tileHeight, 0, 0, tileWidth, tileHeight); + if (fg != "transparent") { + context.fillStyle = fg; + context.globalCompositeOperation = "source-atop"; + context.fillRect(0, 0, tileWidth, tileHeight); + } + if (bg != "transparent") { + context.fillStyle = bg; + context.globalCompositeOperation = "destination-over"; + context.fillRect(0, 0, tileWidth, tileHeight); + } + this._ctx.drawImage(canvas, x * tileWidth, y * tileHeight, tileWidth, tileHeight); + } + else { // no colorizing, easy + this._ctx.drawImage(this._options.tileSet, tile[0], tile[1], tileWidth, tileHeight, x * tileWidth, y * tileHeight, tileWidth, tileHeight); + } + } + } + computeSize(availWidth, availHeight) { + let width = Math.floor(availWidth / this._options.tileWidth); + let height = Math.floor(availHeight / this._options.tileHeight); + return [width, height]; + } + computeFontSize() { + throw new Error("Tile backend does not understand font size"); + } + _normalizedEventToPosition(x, y) { + return [Math.floor(x / this._options.tileWidth), Math.floor(y / this._options.tileHeight)]; + } + _updateSize() { + const opts = this._options; + this._ctx.canvas.width = opts.width * opts.tileWidth; + this._ctx.canvas.height = opts.height * opts.tileHeight; + this._colorCanvas.width = opts.tileWidth; + this._colorCanvas.height = opts.tileHeight; + } +} + +// EXTERNAL MODULE: ./node_modules/rot-js/lib/color.js +var lib_color = __webpack_require__(2); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/display/tile-gl.js + + +/** + * @class Tile backend + * @private + */ +class tile_gl_TileGL extends backend["a" /* default */] { + static isSupported() { + return !!document.createElement("canvas").getContext("webgl2", { preserveDrawingBuffer: true }); + } + constructor() { + super(); + this._uniforms = {}; + try { + this._gl = this._initWebGL(); + } + catch (e) { + alert(e.message); + } + } + schedule(cb) { requestAnimationFrame(cb); } + getContainer() { return this._gl.canvas; } + setOptions(opts) { + super.setOptions(opts); + this._updateSize(); + let tileSet = this._options.tileSet; + if (tileSet && "complete" in tileSet && !tileSet.complete) { + tileSet.addEventListener("load", () => this._updateTexture(tileSet)); + } + else { + this._updateTexture(tileSet); + } + } + draw(data, clearBefore) { + const gl = this._gl; + const opts = this._options; + let [x, y, ch, fg, bg] = data; + let scissorY = gl.canvas.height - (y + 1) * opts.tileHeight; + gl.scissor(x * opts.tileWidth, scissorY, opts.tileWidth, opts.tileHeight); + if (clearBefore) { + if (opts.tileColorize) { + gl.clearColor(0, 0, 0, 0); + } + else { + gl.clearColor(...parseColor(bg)); + } + gl.clear(gl.COLOR_BUFFER_BIT); + } + if (!ch) { + return; + } + let chars = [].concat(ch); + let bgs = [].concat(bg); + let fgs = [].concat(fg); + gl.uniform2fv(this._uniforms["targetPosRel"], [x, y]); + for (let i = 0; i < chars.length; i++) { + let tile = this._options.tileMap[chars[i]]; + if (!tile) { + throw new Error(`Char "${chars[i]}" not found in tileMap`); + } + gl.uniform1f(this._uniforms["colorize"], opts.tileColorize ? 1 : 0); + gl.uniform2fv(this._uniforms["tilesetPosAbs"], tile); + if (opts.tileColorize) { + gl.uniform4fv(this._uniforms["tint"], parseColor(fgs[i])); + gl.uniform4fv(this._uniforms["bg"], parseColor(bgs[i])); + } + gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4); + } + /* + + + for (let i=0;i= canvas.width || y >= canvas.height) { + return [-1, -1]; + } + return this._normalizedEventToPosition(x, y); + } + _initWebGL() { + let gl = document.createElement("canvas").getContext("webgl2", { preserveDrawingBuffer: true }); + window.gl = gl; + let program = createProgram(gl, VS, FS); + gl.useProgram(program); + createQuad(gl); + UNIFORMS.forEach(name => this._uniforms[name] = gl.getUniformLocation(program, name)); + this._program = program; + gl.enable(gl.BLEND); + gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA); + gl.enable(gl.SCISSOR_TEST); + return gl; + } + _normalizedEventToPosition(x, y) { + return [Math.floor(x / this._options.tileWidth), Math.floor(y / this._options.tileHeight)]; + } + _updateSize() { + const gl = this._gl; + const opts = this._options; + const canvasSize = [opts.width * opts.tileWidth, opts.height * opts.tileHeight]; + gl.canvas.width = canvasSize[0]; + gl.canvas.height = canvasSize[1]; + gl.viewport(0, 0, canvasSize[0], canvasSize[1]); + gl.uniform2fv(this._uniforms["tileSize"], [opts.tileWidth, opts.tileHeight]); + gl.uniform2fv(this._uniforms["targetSize"], canvasSize); + } + _updateTexture(tileSet) { + createTexture(this._gl, tileSet); + } +} +const UNIFORMS = ["targetPosRel", "tilesetPosAbs", "tileSize", "targetSize", "colorize", "bg", "tint"]; +const VS = ` +#version 300 es + +in vec2 tilePosRel; +out vec2 tilesetPosPx; + +uniform vec2 tilesetPosAbs; +uniform vec2 tileSize; +uniform vec2 targetSize; +uniform vec2 targetPosRel; + +void main() { + vec2 targetPosPx = (targetPosRel + tilePosRel) * tileSize; + vec2 targetPosNdc = ((targetPosPx / targetSize)-0.5)*2.0; + targetPosNdc.y *= -1.0; + + gl_Position = vec4(targetPosNdc, 0.0, 1.0); + tilesetPosPx = tilesetPosAbs + tilePosRel * tileSize; +}`.trim(); +const FS = ` +#version 300 es +precision highp float; + +in vec2 tilesetPosPx; +out vec4 fragColor; +uniform sampler2D image; +uniform bool colorize; +uniform vec4 bg; +uniform vec4 tint; + +void main() { + fragColor = vec4(0, 0, 0, 1); + + vec4 texel = texelFetch(image, ivec2(tilesetPosPx), 0); + + if (colorize) { + texel.rgb = tint.a * tint.rgb + (1.0-tint.a) * texel.rgb; + fragColor.rgb = texel.a*texel.rgb + (1.0-texel.a)*bg.rgb; + fragColor.a = texel.a + (1.0-texel.a)*bg.a; + } else { + fragColor = texel; + } +}`.trim(); +function createProgram(gl, vss, fss) { + const vs = gl.createShader(gl.VERTEX_SHADER); + gl.shaderSource(vs, vss); + gl.compileShader(vs); + if (!gl.getShaderParameter(vs, gl.COMPILE_STATUS)) { + throw new Error(gl.getShaderInfoLog(vs) || ""); + } + const fs = gl.createShader(gl.FRAGMENT_SHADER); + gl.shaderSource(fs, fss); + gl.compileShader(fs); + if (!gl.getShaderParameter(fs, gl.COMPILE_STATUS)) { + throw new Error(gl.getShaderInfoLog(fs) || ""); + } + const p = gl.createProgram(); + gl.attachShader(p, vs); + gl.attachShader(p, fs); + gl.linkProgram(p); + if (!gl.getProgramParameter(p, gl.LINK_STATUS)) { + throw new Error(gl.getProgramInfoLog(p) || ""); + } + return p; +} +function createQuad(gl) { + const pos = new Float32Array([0, 0, 1, 0, 0, 1, 1, 1]); + const buf = gl.createBuffer(); + gl.bindBuffer(gl.ARRAY_BUFFER, buf); + gl.bufferData(gl.ARRAY_BUFFER, pos, gl.STATIC_DRAW); + gl.enableVertexAttribArray(0); + gl.vertexAttribPointer(0, 2, gl.FLOAT, false, 0, 0); +} +function createTexture(gl, data) { + let t = gl.createTexture(); + gl.bindTexture(gl.TEXTURE_2D, t); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT); + gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT); + gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, 0); + gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, data); + return t; +} +let colorCache = {}; +function parseColor(color) { + if (!(color in colorCache)) { + let parsed; + if (color == "transparent") { + parsed = [0, 0, 0, 0]; + } + else if (color.indexOf("rgba") > -1) { + parsed = (color.match(/[\d.]+/g) || []).map(Number); + for (let i = 0; i < 3; i++) { + parsed[i] = parsed[i] / 255; + } + } + else { + parsed = lib_color["fromString"](color).map($ => $ / 255); + parsed.push(1); + } + colorCache[color] = parsed; + } + return colorCache[color]; +} + +// EXTERNAL MODULE: ./node_modules/rot-js/lib/display/term.js +var term = __webpack_require__(9); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/text.js +/** + * @namespace + * Contains text tokenization and breaking routines + */ +const RE_COLORS = /%([bc]){([^}]*)}/g; +// token types +const TYPE_TEXT = 0; +const TYPE_NEWLINE = 1; +const TYPE_FG = 2; +const TYPE_BG = 3; +/** + * Measure size of a resulting text block + */ +function measure(str, maxWidth) { + let result = { width: 0, height: 1 }; + let tokens = tokenize(str, maxWidth); + let lineWidth = 0; + for (let i = 0; i < tokens.length; i++) { + let token = tokens[i]; + switch (token.type) { + case TYPE_TEXT: + lineWidth += token.value.length; + break; + case TYPE_NEWLINE: + result.height++; + result.width = Math.max(result.width, lineWidth); + lineWidth = 0; + break; + } + } + result.width = Math.max(result.width, lineWidth); + return result; +} +/** + * Convert string to a series of a formatting commands + */ +function tokenize(str, maxWidth) { + let result = []; + /* first tokenization pass - split texts and color formatting commands */ + let offset = 0; + str.replace(RE_COLORS, function (match, type, name, index) { + /* string before */ + let part = str.substring(offset, index); + if (part.length) { + result.push({ + type: TYPE_TEXT, + value: part + }); + } + /* color command */ + result.push({ + type: (type == "c" ? TYPE_FG : TYPE_BG), + value: name.trim() + }); + offset = index + match.length; + return ""; + }); + /* last remaining part */ + let part = str.substring(offset); + if (part.length) { + result.push({ + type: TYPE_TEXT, + value: part + }); + } + return breakLines(result, maxWidth); +} +/* insert line breaks into first-pass tokenized data */ +function breakLines(tokens, maxWidth) { + if (!maxWidth) { + maxWidth = Infinity; + } + let i = 0; + let lineLength = 0; + let lastTokenWithSpace = -1; + while (i < tokens.length) { /* take all text tokens, remove space, apply linebreaks */ + let token = tokens[i]; + if (token.type == TYPE_NEWLINE) { /* reset */ + lineLength = 0; + lastTokenWithSpace = -1; + } + if (token.type != TYPE_TEXT) { /* skip non-text tokens */ + i++; + continue; + } + /* remove spaces at the beginning of line */ + while (lineLength == 0 && token.value.charAt(0) == " ") { + token.value = token.value.substring(1); + } + /* forced newline? insert two new tokens after this one */ + let index = token.value.indexOf("\n"); + if (index != -1) { + token.value = breakInsideToken(tokens, i, index, true); + /* if there are spaces at the end, we must remove them (we do not want the line too long) */ + let arr = token.value.split(""); + while (arr.length && arr[arr.length - 1] == " ") { + arr.pop(); + } + token.value = arr.join(""); + } + /* token degenerated? */ + if (!token.value.length) { + tokens.splice(i, 1); + continue; + } + if (lineLength + token.value.length > maxWidth) { /* line too long, find a suitable breaking spot */ + /* is it possible to break within this token? */ + let index = -1; + while (1) { + let nextIndex = token.value.indexOf(" ", index + 1); + if (nextIndex == -1) { + break; + } + if (lineLength + nextIndex > maxWidth) { + break; + } + index = nextIndex; + } + if (index != -1) { /* break at space within this one */ + token.value = breakInsideToken(tokens, i, index, true); + } + else if (lastTokenWithSpace != -1) { /* is there a previous token where a break can occur? */ + let token = tokens[lastTokenWithSpace]; + let breakIndex = token.value.lastIndexOf(" "); + token.value = breakInsideToken(tokens, lastTokenWithSpace, breakIndex, true); + i = lastTokenWithSpace; + } + else { /* force break in this token */ + token.value = breakInsideToken(tokens, i, maxWidth - lineLength, false); + } + } + else { /* line not long, continue */ + lineLength += token.value.length; + if (token.value.indexOf(" ") != -1) { + lastTokenWithSpace = i; + } + } + i++; /* advance to next token */ + } + tokens.push({ type: TYPE_NEWLINE }); /* insert fake newline to fix the last text line */ + /* remove trailing space from text tokens before newlines */ + let lastTextToken = null; + for (let i = 0; i < tokens.length; i++) { + let token = tokens[i]; + switch (token.type) { + case TYPE_TEXT: + lastTextToken = token; + break; + case TYPE_NEWLINE: + if (lastTextToken) { /* remove trailing space */ + let arr = lastTextToken.value.split(""); + while (arr.length && arr[arr.length - 1] == " ") { + arr.pop(); + } + lastTextToken.value = arr.join(""); + } + lastTextToken = null; + break; + } + } + tokens.pop(); /* remove fake token */ + return tokens; +} +/** + * Create new tokens and insert them into the stream + * @param {object[]} tokens + * @param {int} tokenIndex Token being processed + * @param {int} breakIndex Index within current token's value + * @param {bool} removeBreakChar Do we want to remove the breaking character? + * @returns {string} remaining unbroken token value + */ +function breakInsideToken(tokens, tokenIndex, breakIndex, removeBreakChar) { + let newBreakToken = { + type: TYPE_NEWLINE + }; + let newTextToken = { + type: TYPE_TEXT, + value: tokens[tokenIndex].value.substring(breakIndex + (removeBreakChar ? 1 : 0)) + }; + tokens.splice(tokenIndex + 1, 0, newBreakToken, newTextToken); + return tokens[tokenIndex].value.substring(0, breakIndex); +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/constants.js +/** Default with for display and map generators */ +let DEFAULT_WIDTH = 80; +/** Default height for display and map generators */ +let DEFAULT_HEIGHT = 25; +const DIRS = { + 4: [[0, -1], [1, 0], [0, 1], [-1, 0]], + 8: [[0, -1], [1, -1], [1, 0], [1, 1], [0, 1], [-1, 1], [-1, 0], [-1, -1]], + 6: [[-1, -1], [1, -1], [2, 0], [1, 1], [-1, 1], [-2, 0]] +}; +const KEYS = { + /** Cancel key. */ + VK_CANCEL: 3, + /** Help key. */ + VK_HELP: 6, + /** Backspace key. */ + VK_BACK_SPACE: 8, + /** Tab key. */ + VK_TAB: 9, + /** 5 key on Numpad when NumLock is unlocked. Or on Mac, clear key which is positioned at NumLock key. */ + VK_CLEAR: 12, + /** Return/enter key on the main keyboard. */ + VK_RETURN: 13, + /** Reserved, but not used. */ + VK_ENTER: 14, + /** Shift key. */ + VK_SHIFT: 16, + /** Control key. */ + VK_CONTROL: 17, + /** Alt (Option on Mac) key. */ + VK_ALT: 18, + /** Pause key. */ + VK_PAUSE: 19, + /** Caps lock. */ + VK_CAPS_LOCK: 20, + /** Escape key. */ + VK_ESCAPE: 27, + /** Space bar. */ + VK_SPACE: 32, + /** Page Up key. */ + VK_PAGE_UP: 33, + /** Page Down key. */ + VK_PAGE_DOWN: 34, + /** End key. */ + VK_END: 35, + /** Home key. */ + VK_HOME: 36, + /** Left arrow. */ + VK_LEFT: 37, + /** Up arrow. */ + VK_UP: 38, + /** Right arrow. */ + VK_RIGHT: 39, + /** Down arrow. */ + VK_DOWN: 40, + /** Print Screen key. */ + VK_PRINTSCREEN: 44, + /** Ins(ert) key. */ + VK_INSERT: 45, + /** Del(ete) key. */ + VK_DELETE: 46, + /***/ + VK_0: 48, + /***/ + VK_1: 49, + /***/ + VK_2: 50, + /***/ + VK_3: 51, + /***/ + VK_4: 52, + /***/ + VK_5: 53, + /***/ + VK_6: 54, + /***/ + VK_7: 55, + /***/ + VK_8: 56, + /***/ + VK_9: 57, + /** Colon (:) key. Requires Gecko 15.0 */ + VK_COLON: 58, + /** Semicolon (;) key. */ + VK_SEMICOLON: 59, + /** Less-than (<) key. Requires Gecko 15.0 */ + VK_LESS_THAN: 60, + /** Equals (=) key. */ + VK_EQUALS: 61, + /** Greater-than (>) key. Requires Gecko 15.0 */ + VK_GREATER_THAN: 62, + /** Question mark (?) key. Requires Gecko 15.0 */ + VK_QUESTION_MARK: 63, + /** Atmark (@) key. Requires Gecko 15.0 */ + VK_AT: 64, + /***/ + VK_A: 65, + /***/ + VK_B: 66, + /***/ + VK_C: 67, + /***/ + VK_D: 68, + /***/ + VK_E: 69, + /***/ + VK_F: 70, + /***/ + VK_G: 71, + /***/ + VK_H: 72, + /***/ + VK_I: 73, + /***/ + VK_J: 74, + /***/ + VK_K: 75, + /***/ + VK_L: 76, + /***/ + VK_M: 77, + /***/ + VK_N: 78, + /***/ + VK_O: 79, + /***/ + VK_P: 80, + /***/ + VK_Q: 81, + /***/ + VK_R: 82, + /***/ + VK_S: 83, + /***/ + VK_T: 84, + /***/ + VK_U: 85, + /***/ + VK_V: 86, + /***/ + VK_W: 87, + /***/ + VK_X: 88, + /***/ + VK_Y: 89, + /***/ + VK_Z: 90, + /***/ + VK_CONTEXT_MENU: 93, + /** 0 on the numeric keypad. */ + VK_NUMPAD0: 96, + /** 1 on the numeric keypad. */ + VK_NUMPAD1: 97, + /** 2 on the numeric keypad. */ + VK_NUMPAD2: 98, + /** 3 on the numeric keypad. */ + VK_NUMPAD3: 99, + /** 4 on the numeric keypad. */ + VK_NUMPAD4: 100, + /** 5 on the numeric keypad. */ + VK_NUMPAD5: 101, + /** 6 on the numeric keypad. */ + VK_NUMPAD6: 102, + /** 7 on the numeric keypad. */ + VK_NUMPAD7: 103, + /** 8 on the numeric keypad. */ + VK_NUMPAD8: 104, + /** 9 on the numeric keypad. */ + VK_NUMPAD9: 105, + /** * on the numeric keypad. */ + VK_MULTIPLY: 106, + /** + on the numeric keypad. */ + VK_ADD: 107, + /***/ + VK_SEPARATOR: 108, + /** - on the numeric keypad. */ + VK_SUBTRACT: 109, + /** Decimal point on the numeric keypad. */ + VK_DECIMAL: 110, + /** / on the numeric keypad. */ + VK_DIVIDE: 111, + /** F1 key. */ + VK_F1: 112, + /** F2 key. */ + VK_F2: 113, + /** F3 key. */ + VK_F3: 114, + /** F4 key. */ + VK_F4: 115, + /** F5 key. */ + VK_F5: 116, + /** F6 key. */ + VK_F6: 117, + /** F7 key. */ + VK_F7: 118, + /** F8 key. */ + VK_F8: 119, + /** F9 key. */ + VK_F9: 120, + /** F10 key. */ + VK_F10: 121, + /** F11 key. */ + VK_F11: 122, + /** F12 key. */ + VK_F12: 123, + /** F13 key. */ + VK_F13: 124, + /** F14 key. */ + VK_F14: 125, + /** F15 key. */ + VK_F15: 126, + /** F16 key. */ + VK_F16: 127, + /** F17 key. */ + VK_F17: 128, + /** F18 key. */ + VK_F18: 129, + /** F19 key. */ + VK_F19: 130, + /** F20 key. */ + VK_F20: 131, + /** F21 key. */ + VK_F21: 132, + /** F22 key. */ + VK_F22: 133, + /** F23 key. */ + VK_F23: 134, + /** F24 key. */ + VK_F24: 135, + /** Num Lock key. */ + VK_NUM_LOCK: 144, + /** Scroll Lock key. */ + VK_SCROLL_LOCK: 145, + /** Circumflex (^) key. Requires Gecko 15.0 */ + VK_CIRCUMFLEX: 160, + /** Exclamation (!) key. Requires Gecko 15.0 */ + VK_EXCLAMATION: 161, + /** Double quote () key. Requires Gecko 15.0 */ + VK_DOUBLE_QUOTE: 162, + /** Hash (#) key. Requires Gecko 15.0 */ + VK_HASH: 163, + /** Dollar sign ($) key. Requires Gecko 15.0 */ + VK_DOLLAR: 164, + /** Percent (%) key. Requires Gecko 15.0 */ + VK_PERCENT: 165, + /** Ampersand (&) key. Requires Gecko 15.0 */ + VK_AMPERSAND: 166, + /** Underscore (_) key. Requires Gecko 15.0 */ + VK_UNDERSCORE: 167, + /** Open parenthesis (() key. Requires Gecko 15.0 */ + VK_OPEN_PAREN: 168, + /** Close parenthesis ()) key. Requires Gecko 15.0 */ + VK_CLOSE_PAREN: 169, + /* Asterisk (*) key. Requires Gecko 15.0 */ + VK_ASTERISK: 170, + /** Plus (+) key. Requires Gecko 15.0 */ + VK_PLUS: 171, + /** Pipe (|) key. Requires Gecko 15.0 */ + VK_PIPE: 172, + /** Hyphen-US/docs/Minus (-) key. Requires Gecko 15.0 */ + VK_HYPHEN_MINUS: 173, + /** Open curly bracket ({) key. Requires Gecko 15.0 */ + VK_OPEN_CURLY_BRACKET: 174, + /** Close curly bracket (}) key. Requires Gecko 15.0 */ + VK_CLOSE_CURLY_BRACKET: 175, + /** Tilde (~) key. Requires Gecko 15.0 */ + VK_TILDE: 176, + /** Comma (,) key. */ + VK_COMMA: 188, + /** Period (.) key. */ + VK_PERIOD: 190, + /** Slash (/) key. */ + VK_SLASH: 191, + /** Back tick (`) key. */ + VK_BACK_QUOTE: 192, + /** Open square bracket ([) key. */ + VK_OPEN_BRACKET: 219, + /** Back slash (\) key. */ + VK_BACK_SLASH: 220, + /** Close square bracket (]) key. */ + VK_CLOSE_BRACKET: 221, + /** Quote (''') key. */ + VK_QUOTE: 222, + /** Meta key on Linux, Command key on Mac. */ + VK_META: 224, + /** AltGr key on Linux. Requires Gecko 15.0 */ + VK_ALTGR: 225, + /** Windows logo key on Windows. Or Super or Hyper key on Linux. Requires Gecko 15.0 */ + VK_WIN: 91, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_KANA: 21, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_HANGUL: 21, + /** 英数 key on Japanese Mac keyboard. Requires Gecko 15.0 */ + VK_EISU: 22, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_JUNJA: 23, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_FINAL: 24, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_HANJA: 25, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_KANJI: 25, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_CONVERT: 28, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_NONCONVERT: 29, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_ACCEPT: 30, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_MODECHANGE: 31, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_SELECT: 41, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_PRINT: 42, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_EXECUTE: 43, + /** Linux support for this keycode was added in Gecko 4.0. */ + VK_SLEEP: 95 +}; + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/display/display.js + + + + + + + +const BACKENDS = { + "hex": hex_Hex, + "rect": rect_Rect, + "tile": tile_Tile, + "tile-gl": tile_gl_TileGL, + "term": term["a" /* default */] +}; +const DEFAULT_OPTIONS = { + width: DEFAULT_WIDTH, + height: DEFAULT_HEIGHT, + transpose: false, + layout: "rect", + fontSize: 15, + spacing: 1, + border: 0, + forceSquareRatio: false, + fontFamily: "monospace", + fontStyle: "", + fg: "#ccc", + bg: "#000", + tileWidth: 32, + tileHeight: 32, + tileMap: {}, + tileSet: null, + tileColorize: false +}; +/** + * @class Visual map display + */ +class display_Display { + constructor(options = {}) { + this._data = {}; + this._dirty = false; // false = nothing, true = all, object = dirty cells + this._options = {}; + options = Object.assign({}, DEFAULT_OPTIONS, options); + this.setOptions(options); + this.DEBUG = this.DEBUG.bind(this); + this._tick = this._tick.bind(this); + this._backend.schedule(this._tick); + } + /** + * Debug helper, ideal as a map generator callback. Always bound to this. + * @param {int} x + * @param {int} y + * @param {int} what + */ + DEBUG(x, y, what) { + let colors = [this._options.bg, this._options.fg]; + this.draw(x, y, null, null, colors[what % colors.length]); + } + /** + * Clear the whole display (cover it with background color) + */ + clear() { + this._data = {}; + this._dirty = true; + } + /** + * @see ROT.Display + */ + setOptions(options) { + Object.assign(this._options, options); + if (options.width || options.height || options.fontSize || options.fontFamily || options.spacing || options.layout) { + if (options.layout) { + let ctor = BACKENDS[options.layout]; + this._backend = new ctor(); + } + this._backend.setOptions(this._options); + this._dirty = true; + } + return this; + } + /** + * Returns currently set options + */ + getOptions() { return this._options; } + /** + * Returns the DOM node of this display + */ + getContainer() { return this._backend.getContainer(); } + /** + * Compute the maximum width/height to fit into a set of given constraints + * @param {int} availWidth Maximum allowed pixel width + * @param {int} availHeight Maximum allowed pixel height + * @returns {int[2]} cellWidth,cellHeight + */ + computeSize(availWidth, availHeight) { + return this._backend.computeSize(availWidth, availHeight); + } + /** + * Compute the maximum font size to fit into a set of given constraints + * @param {int} availWidth Maximum allowed pixel width + * @param {int} availHeight Maximum allowed pixel height + * @returns {int} fontSize + */ + computeFontSize(availWidth, availHeight) { + return this._backend.computeFontSize(availWidth, availHeight); + } + computeTileSize(availWidth, availHeight) { + let width = Math.floor(availWidth / this._options.width); + let height = Math.floor(availHeight / this._options.height); + return [width, height]; + } + /** + * Convert a DOM event (mouse or touch) to map coordinates. Uses first touch for multi-touch. + * @param {Event} e event + * @returns {int[2]} -1 for values outside of the canvas + */ + eventToPosition(e) { + let x, y; + if ("touches" in e) { + x = e.touches[0].clientX; + y = e.touches[0].clientY; + } + else { + x = e.clientX; + y = e.clientY; + } + return this._backend.eventToPosition(x, y); + } + /** + * @param {int} x + * @param {int} y + * @param {string || string[]} ch One or more chars (will be overlapping themselves) + * @param {string} [fg] foreground color + * @param {string} [bg] background color + */ + draw(x, y, ch, fg, bg) { + if (!fg) { + fg = this._options.fg; + } + if (!bg) { + bg = this._options.bg; + } + let key = `${x},${y}`; + this._data[key] = [x, y, ch, fg, bg]; + if (this._dirty === true) { + return; + } // will already redraw everything + if (!this._dirty) { + this._dirty = {}; + } // first! + this._dirty[key] = true; + } + /** + * Draws a text at given position. Optionally wraps at a maximum length. Currently does not work with hex layout. + * @param {int} x + * @param {int} y + * @param {string} text May contain color/background format specifiers, %c{name}/%b{name}, both optional. %c{}/%b{} resets to default. + * @param {int} [maxWidth] wrap at what width? + * @returns {int} lines drawn + */ + drawText(x, y, text, maxWidth) { + let fg = null; + let bg = null; + let cx = x; + let cy = y; + let lines = 1; + if (!maxWidth) { + maxWidth = this._options.width - x; + } + let tokens = tokenize(text, maxWidth); + while (tokens.length) { // interpret tokenized opcode stream + let token = tokens.shift(); + switch (token.type) { + case TYPE_TEXT: + let isSpace = false, isPrevSpace = false, isFullWidth = false, isPrevFullWidth = false; + for (let i = 0; i < token.value.length; i++) { + let cc = token.value.charCodeAt(i); + let c = token.value.charAt(i); + // Assign to `true` when the current char is full-width. + isFullWidth = (cc > 0xff00 && cc < 0xff61) || (cc > 0xffdc && cc < 0xffe8) || cc > 0xffee; + // Current char is space, whatever full-width or half-width both are OK. + isSpace = (c.charCodeAt(0) == 0x20 || c.charCodeAt(0) == 0x3000); + // The previous char is full-width and + // current char is nether half-width nor a space. + if (isPrevFullWidth && !isFullWidth && !isSpace) { + cx++; + } // add an extra position + // The current char is full-width and + // the previous char is not a space. + if (isFullWidth && !isPrevSpace) { + cx++; + } // add an extra position + this.draw(cx++, cy, c, fg, bg); + isPrevSpace = isSpace; + isPrevFullWidth = isFullWidth; + } + break; + case TYPE_FG: + fg = token.value || null; + break; + case TYPE_BG: + bg = token.value || null; + break; + case TYPE_NEWLINE: + cx = x; + cy++; + lines++; + break; + } + } + return lines; + } + /** + * Timer tick: update dirty parts + */ + _tick() { + this._backend.schedule(this._tick); + if (!this._dirty) { + return; + } + if (this._dirty === true) { // draw all + this._backend.clear(); + for (let id in this._data) { + this._draw(id, false); + } // redraw cached data + } + else { // draw only dirty + for (let key in this._dirty) { + this._draw(key, true); + } + } + this._dirty = false; + } + /** + * @param {string} key What to draw + * @param {bool} clearBefore Is it necessary to clean before? + */ + _draw(key, clearBefore) { + let data = this._data[key]; + if (data[4] != this._options.bg) { + clearBefore = true; + } + this._backend.draw(data, clearBefore); + } +} +display_Display.Rect = rect_Rect; +display_Display.Hex = hex_Hex; +display_Display.Tile = tile_Tile; +display_Display.TileGL = tile_gl_TileGL; +display_Display.Term = term["a" /* default */]; + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/stringgenerator.js + +/** + * @class (Markov process)-based string generator. + * Copied from a RogueBasin article. + * Offers configurable order and prior. + */ +class stringgenerator_StringGenerator { + constructor(options) { + this._options = { + words: false, + order: 3, + prior: 0.001 + }; + Object.assign(this._options, options); + this._boundary = String.fromCharCode(0); + this._suffix = this._boundary; + this._prefix = []; + for (let i = 0; i < this._options.order; i++) { + this._prefix.push(this._boundary); + } + this._priorValues = {}; + this._priorValues[this._boundary] = this._options.prior; + this._data = {}; + } + /** + * Remove all learning data + */ + clear() { + this._data = {}; + this._priorValues = {}; + } + /** + * @returns {string} Generated string + */ + generate() { + let result = [this._sample(this._prefix)]; + while (result[result.length - 1] != this._boundary) { + result.push(this._sample(result)); + } + return this._join(result.slice(0, -1)); + } + /** + * Observe (learn) a string from a training set + */ + observe(string) { + let tokens = this._split(string); + for (let i = 0; i < tokens.length; i++) { + this._priorValues[tokens[i]] = this._options.prior; + } + tokens = this._prefix.concat(tokens).concat(this._suffix); /* add boundary symbols */ + for (let i = this._options.order; i < tokens.length; i++) { + let context = tokens.slice(i - this._options.order, i); + let event = tokens[i]; + for (let j = 0; j < context.length; j++) { + let subcontext = context.slice(j); + this._observeEvent(subcontext, event); + } + } + } + getStats() { + let parts = []; + let priorCount = Object.keys(this._priorValues).length; + priorCount--; // boundary + parts.push("distinct samples: " + priorCount); + let dataCount = Object.keys(this._data).length; + let eventCount = 0; + for (let p in this._data) { + eventCount += Object.keys(this._data[p]).length; + } + parts.push("dictionary size (contexts): " + dataCount); + parts.push("dictionary size (events): " + eventCount); + return parts.join(", "); + } + /** + * @param {string} + * @returns {string[]} + */ + _split(str) { + return str.split(this._options.words ? /\s+/ : ""); + } + /** + * @param {string[]} + * @returns {string} + */ + _join(arr) { + return arr.join(this._options.words ? " " : ""); + } + /** + * @param {string[]} context + * @param {string} event + */ + _observeEvent(context, event) { + let key = this._join(context); + if (!(key in this._data)) { + this._data[key] = {}; + } + let data = this._data[key]; + if (!(event in data)) { + data[event] = 0; + } + data[event]++; + } + /** + * @param {string[]} + * @returns {string} + */ + _sample(context) { + context = this._backoff(context); + let key = this._join(context); + let data = this._data[key]; + let available = {}; + if (this._options.prior) { + for (let event in this._priorValues) { + available[event] = this._priorValues[event]; + } + for (let event in data) { + available[event] += data[event]; + } + } + else { + available = data; + } + return rng["a" /* default */].getWeightedValue(available); + } + /** + * @param {string[]} + * @returns {string[]} + */ + _backoff(context) { + if (context.length > this._options.order) { + context = context.slice(-this._options.order); + } + else if (context.length < this._options.order) { + context = this._prefix.slice(0, this._options.order - context.length).concat(context); + } + while (!(this._join(context) in this._data) && context.length > 0) { + context = context.slice(1); + } + return context; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/eventqueue.js +class EventQueue { + /** + * @class Generic event queue: stores events and retrieves them based on their time + */ + constructor() { + this._time = 0; + this._events = []; + this._eventTimes = []; + } + /** + * @returns {number} Elapsed time + */ + getTime() { return this._time; } + /** + * Clear all scheduled events + */ + clear() { + this._events = []; + this._eventTimes = []; + return this; + } + /** + * @param {?} event + * @param {number} time + */ + add(event, time) { + let index = this._events.length; + for (let i = 0; i < this._eventTimes.length; i++) { + if (this._eventTimes[i] > time) { + index = i; + break; + } + } + this._events.splice(index, 0, event); + this._eventTimes.splice(index, 0, time); + } + /** + * Locates the nearest event, advances time if necessary. Returns that event and removes it from the queue. + * @returns {? || null} The event previously added by addEvent, null if no event available + */ + get() { + if (!this._events.length) { + return null; + } + let time = this._eventTimes.splice(0, 1)[0]; + if (time > 0) { /* advance */ + this._time += time; + for (let i = 0; i < this._eventTimes.length; i++) { + this._eventTimes[i] -= time; + } + } + return this._events.splice(0, 1)[0]; + } + /** + * Get the time associated with the given event + * @param {?} event + * @returns {number} time + */ + getEventTime(event) { + let index = this._events.indexOf(event); + if (index == -1) { + return undefined; + } + return this._eventTimes[index]; + } + /** + * Remove an event from the queue + * @param {?} event + * @returns {bool} success? + */ + remove(event) { + let index = this._events.indexOf(event); + if (index == -1) { + return false; + } + this._remove(index); + return true; + } + ; + /** + * Remove an event from the queue + * @param {int} index + */ + _remove(index) { + this._events.splice(index, 1); + this._eventTimes.splice(index, 1); + } + ; +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/scheduler/scheduler.js + +class scheduler_Scheduler { + /** + * @class Abstract scheduler + */ + constructor() { + this._queue = new EventQueue(); + this._repeat = []; + this._current = null; + } + /** + * @see ROT.EventQueue#getTime + */ + getTime() { return this._queue.getTime(); } + /** + * @param {?} item + * @param {bool} repeat + */ + add(item, repeat) { + if (repeat) { + this._repeat.push(item); + } + return this; + } + /** + * Get the time the given item is scheduled for + * @param {?} item + * @returns {number} time + */ + getTimeOf(item) { + return this._queue.getEventTime(item); + } + /** + * Clear all items + */ + clear() { + this._queue.clear(); + this._repeat = []; + this._current = null; + return this; + } + /** + * Remove a previously added item + * @param {?} item + * @returns {bool} successful? + */ + remove(item) { + let result = this._queue.remove(item); + let index = this._repeat.indexOf(item); + if (index != -1) { + this._repeat.splice(index, 1); + } + if (this._current == item) { + this._current = null; + } + return result; + } + /** + * Schedule next item + * @returns {?} + */ + next() { + this._current = this._queue.get(); + return this._current; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/scheduler/simple.js + +/** + * @class Simple fair scheduler (round-robin style) + */ +class simple_Simple extends scheduler_Scheduler { + add(item, repeat) { + this._queue.add(item, 0); + return super.add(item, repeat); + } + next() { + if (this._current !== null && this._repeat.indexOf(this._current) != -1) { + this._queue.add(this._current, 0); + } + return super.next(); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/scheduler/speed.js + +/** + * @class Speed-based scheduler + */ +class speed_Speed extends scheduler_Scheduler { + /** + * @param {object} item anything with "getSpeed" method + * @param {bool} repeat + * @param {number} [time=1/item.getSpeed()] + * @see ROT.Scheduler#add + */ + add(item, repeat, time) { + this._queue.add(item, time !== undefined ? time : 1 / item.getSpeed()); + return super.add(item, repeat); + } + /** + * @see ROT.Scheduler#next + */ + next() { + if (this._current && this._repeat.indexOf(this._current) != -1) { + this._queue.add(this._current, 1 / this._current.getSpeed()); + } + return super.next(); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/scheduler/action.js + +/** + * @class Action-based scheduler + * @augments ROT.Scheduler + */ +class action_Action extends scheduler_Scheduler { + constructor() { + super(); + this._defaultDuration = 1; /* for newly added */ + this._duration = this._defaultDuration; /* for this._current */ + } + /** + * @param {object} item + * @param {bool} repeat + * @param {number} [time=1] + * @see ROT.Scheduler#add + */ + add(item, repeat, time) { + this._queue.add(item, time || this._defaultDuration); + return super.add(item, repeat); + } + clear() { + this._duration = this._defaultDuration; + return super.clear(); + } + remove(item) { + if (item == this._current) { + this._duration = this._defaultDuration; + } + return super.remove(item); + } + /** + * @see ROT.Scheduler#next + */ + next() { + if (this._current !== null && this._repeat.indexOf(this._current) != -1) { + this._queue.add(this._current, this._duration || this._defaultDuration); + this._duration = this._defaultDuration; + } + return super.next(); + } + /** + * Set duration for the active item + */ + setDuration(time) { + if (this._current) { + this._duration = time; + } + return this; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/scheduler/index.js + + + +/* harmony default export */ var scheduler = ({ Simple: simple_Simple, Speed: speed_Speed, Action: action_Action }); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/fov/fov.js + +; +; +class fov_FOV { + /** + * @class Abstract FOV algorithm + * @param {function} lightPassesCallback Does the light pass through x,y? + * @param {object} [options] + * @param {int} [options.topology=8] 4/6/8 + */ + constructor(lightPassesCallback, options = {}) { + this._lightPasses = lightPassesCallback; + this._options = Object.assign({ topology: 8 }, options); + } + /** + * Return all neighbors in a concentric ring + * @param {int} cx center-x + * @param {int} cy center-y + * @param {int} r range + */ + _getCircle(cx, cy, r) { + let result = []; + let dirs, countFactor, startOffset; + switch (this._options.topology) { + case 4: + countFactor = 1; + startOffset = [0, 1]; + dirs = [ + DIRS[8][7], + DIRS[8][1], + DIRS[8][3], + DIRS[8][5] + ]; + break; + case 6: + dirs = DIRS[6]; + countFactor = 1; + startOffset = [-1, 1]; + break; + case 8: + dirs = DIRS[4]; + countFactor = 2; + startOffset = [-1, 1]; + break; + default: + throw new Error("Incorrect topology for FOV computation"); + break; + } + /* starting neighbor */ + let x = cx + startOffset[0] * r; + let y = cy + startOffset[1] * r; + /* circle */ + for (let i = 0; i < dirs.length; i++) { + for (let j = 0; j < r * countFactor; j++) { + result.push([x, y]); + x += dirs[i][0]; + y += dirs[i][1]; + } + } + return result; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/fov/discrete-shadowcasting.js + +/** + * @class Discrete shadowcasting algorithm. Obsoleted by Precise shadowcasting. + * @augments ROT.FOV + */ +class discrete_shadowcasting_DiscreteShadowcasting extends fov_FOV { + compute(x, y, R, callback) { + /* this place is always visible */ + callback(x, y, 0, 1); + /* standing in a dark place. FIXME is this a good idea? */ + if (!this._lightPasses(x, y)) { + return; + } + /* start and end angles */ + let DATA = []; + let A, B, cx, cy, blocks; + /* analyze surrounding cells in concentric rings, starting from the center */ + for (let r = 1; r <= R; r++) { + let neighbors = this._getCircle(x, y, r); + let angle = 360 / neighbors.length; + for (let i = 0; i < neighbors.length; i++) { + cx = neighbors[i][0]; + cy = neighbors[i][1]; + A = angle * (i - 0.5); + B = A + angle; + blocks = !this._lightPasses(cx, cy); + if (this._visibleCoords(Math.floor(A), Math.ceil(B), blocks, DATA)) { + callback(cx, cy, r, 1); + } + if (DATA.length == 2 && DATA[0] == 0 && DATA[1] == 360) { + return; + } /* cutoff? */ + } /* for all cells in this ring */ + } /* for all rings */ + } + /** + * @param {int} A start angle + * @param {int} B end angle + * @param {bool} blocks Does current cell block visibility? + * @param {int[][]} DATA shadowed angle pairs + */ + _visibleCoords(A, B, blocks, DATA) { + if (A < 0) { + let v1 = this._visibleCoords(0, B, blocks, DATA); + let v2 = this._visibleCoords(360 + A, 360, blocks, DATA); + return v1 || v2; + } + let index = 0; + while (index < DATA.length && DATA[index] < A) { + index++; + } + if (index == DATA.length) { /* completely new shadow */ + if (blocks) { + DATA.push(A, B); + } + return true; + } + let count = 0; + if (index % 2) { /* this shadow starts in an existing shadow, or within its ending boundary */ + while (index < DATA.length && DATA[index] < B) { + index++; + count++; + } + if (count == 0) { + return false; + } + if (blocks) { + if (count % 2) { + DATA.splice(index - count, count, B); + } + else { + DATA.splice(index - count, count); + } + } + return true; + } + else { /* this shadow starts outside an existing shadow, or within a starting boundary */ + while (index < DATA.length && DATA[index] < B) { + index++; + count++; + } + /* visible when outside an existing shadow, or when overlapping */ + if (A == DATA[index - count] && count == 1) { + return false; + } + if (blocks) { + if (count % 2) { + DATA.splice(index - count, count, A); + } + else { + DATA.splice(index - count, count, A, B); + } + } + return true; + } + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/fov/precise-shadowcasting.js + +/** + * @class Precise shadowcasting algorithm + * @augments ROT.FOV + */ +class precise_shadowcasting_PreciseShadowcasting extends fov_FOV { + compute(x, y, R, callback) { + /* this place is always visible */ + callback(x, y, 0, 1); + /* standing in a dark place. FIXME is this a good idea? */ + if (!this._lightPasses(x, y)) { + return; + } + /* list of all shadows */ + let SHADOWS = []; + let cx, cy, blocks, A1, A2, visibility; + /* analyze surrounding cells in concentric rings, starting from the center */ + for (let r = 1; r <= R; r++) { + let neighbors = this._getCircle(x, y, r); + let neighborCount = neighbors.length; + for (let i = 0; i < neighborCount; i++) { + cx = neighbors[i][0]; + cy = neighbors[i][1]; + /* shift half-an-angle backwards to maintain consistency of 0-th cells */ + A1 = [i ? 2 * i - 1 : 2 * neighborCount - 1, 2 * neighborCount]; + A2 = [2 * i + 1, 2 * neighborCount]; + blocks = !this._lightPasses(cx, cy); + visibility = this._checkVisibility(A1, A2, blocks, SHADOWS); + if (visibility) { + callback(cx, cy, r, visibility); + } + if (SHADOWS.length == 2 && SHADOWS[0][0] == 0 && SHADOWS[1][0] == SHADOWS[1][1]) { + return; + } /* cutoff? */ + } /* for all cells in this ring */ + } /* for all rings */ + } + /** + * @param {int[2]} A1 arc start + * @param {int[2]} A2 arc end + * @param {bool} blocks Does current arc block visibility? + * @param {int[][]} SHADOWS list of active shadows + */ + _checkVisibility(A1, A2, blocks, SHADOWS) { + if (A1[0] > A2[0]) { /* split into two sub-arcs */ + let v1 = this._checkVisibility(A1, [A1[1], A1[1]], blocks, SHADOWS); + let v2 = this._checkVisibility([0, 1], A2, blocks, SHADOWS); + return (v1 + v2) / 2; + } + /* index1: first shadow >= A1 */ + let index1 = 0, edge1 = false; + while (index1 < SHADOWS.length) { + let old = SHADOWS[index1]; + let diff = old[0] * A1[1] - A1[0] * old[1]; + if (diff >= 0) { /* old >= A1 */ + if (diff == 0 && !(index1 % 2)) { + edge1 = true; + } + break; + } + index1++; + } + /* index2: last shadow <= A2 */ + let index2 = SHADOWS.length, edge2 = false; + while (index2--) { + let old = SHADOWS[index2]; + let diff = A2[0] * old[1] - old[0] * A2[1]; + if (diff >= 0) { /* old <= A2 */ + if (diff == 0 && (index2 % 2)) { + edge2 = true; + } + break; + } + } + let visible = true; + if (index1 == index2 && (edge1 || edge2)) { /* subset of existing shadow, one of the edges match */ + visible = false; + } + else if (edge1 && edge2 && index1 + 1 == index2 && (index2 % 2)) { /* completely equivalent with existing shadow */ + visible = false; + } + else if (index1 > index2 && (index1 % 2)) { /* subset of existing shadow, not touching */ + visible = false; + } + if (!visible) { + return 0; + } /* fast case: not visible */ + let visibleLength; + /* compute the length of visible arc, adjust list of shadows (if blocking) */ + let remove = index2 - index1 + 1; + if (remove % 2) { + if (index1 % 2) { /* first edge within existing shadow, second outside */ + let P = SHADOWS[index1]; + visibleLength = (A2[0] * P[1] - P[0] * A2[1]) / (P[1] * A2[1]); + if (blocks) { + SHADOWS.splice(index1, remove, A2); + } + } + else { /* second edge within existing shadow, first outside */ + let P = SHADOWS[index2]; + visibleLength = (P[0] * A1[1] - A1[0] * P[1]) / (A1[1] * P[1]); + if (blocks) { + SHADOWS.splice(index1, remove, A1); + } + } + } + else { + if (index1 % 2) { /* both edges within existing shadows */ + let P1 = SHADOWS[index1]; + let P2 = SHADOWS[index2]; + visibleLength = (P2[0] * P1[1] - P1[0] * P2[1]) / (P1[1] * P2[1]); + if (blocks) { + SHADOWS.splice(index1, remove); + } + } + else { /* both edges outside existing shadows */ + if (blocks) { + SHADOWS.splice(index1, remove, A1, A2); + } + return 1; /* whole arc visible! */ + } + } + let arcLength = (A2[0] * A1[1] - A1[0] * A2[1]) / (A1[1] * A2[1]); + return visibleLength / arcLength; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/fov/recursive-shadowcasting.js + +/** Octants used for translating recursive shadowcasting offsets */ +const OCTANTS = [ + [-1, 0, 0, 1], + [0, -1, 1, 0], + [0, -1, -1, 0], + [-1, 0, 0, -1], + [1, 0, 0, -1], + [0, 1, -1, 0], + [0, 1, 1, 0], + [1, 0, 0, 1] +]; +/** + * @class Recursive shadowcasting algorithm + * Currently only supports 4/8 topologies, not hexagonal. + * Based on Peter Harkins' implementation of Björn Bergström's algorithm described here: http://www.roguebasin.com/index.php?title=FOV_using_recursive_shadowcasting + * @augments ROT.FOV + */ +class recursive_shadowcasting_RecursiveShadowcasting extends fov_FOV { + /** + * Compute visibility for a 360-degree circle + * @param {int} x + * @param {int} y + * @param {int} R Maximum visibility radius + * @param {function} callback + */ + compute(x, y, R, callback) { + //You can always see your own tile + callback(x, y, 0, 1); + for (let i = 0; i < OCTANTS.length; i++) { + this._renderOctant(x, y, OCTANTS[i], R, callback); + } + } + /** + * Compute visibility for a 180-degree arc + * @param {int} x + * @param {int} y + * @param {int} R Maximum visibility radius + * @param {int} dir Direction to look in (expressed in a ROT.DIRS value); + * @param {function} callback + */ + compute180(x, y, R, dir, callback) { + //You can always see your own tile + callback(x, y, 0, 1); + let previousOctant = (dir - 1 + 8) % 8; //Need to retrieve the previous octant to render a full 180 degrees + let nextPreviousOctant = (dir - 2 + 8) % 8; //Need to retrieve the previous two octants to render a full 180 degrees + let nextOctant = (dir + 1 + 8) % 8; //Need to grab to next octant to render a full 180 degrees + this._renderOctant(x, y, OCTANTS[nextPreviousOctant], R, callback); + this._renderOctant(x, y, OCTANTS[previousOctant], R, callback); + this._renderOctant(x, y, OCTANTS[dir], R, callback); + this._renderOctant(x, y, OCTANTS[nextOctant], R, callback); + } + ; + /** + * Compute visibility for a 90-degree arc + * @param {int} x + * @param {int} y + * @param {int} R Maximum visibility radius + * @param {int} dir Direction to look in (expressed in a ROT.DIRS value); + * @param {function} callback + */ + compute90(x, y, R, dir, callback) { + //You can always see your own tile + callback(x, y, 0, 1); + let previousOctant = (dir - 1 + 8) % 8; //Need to retrieve the previous octant to render a full 90 degrees + this._renderOctant(x, y, OCTANTS[dir], R, callback); + this._renderOctant(x, y, OCTANTS[previousOctant], R, callback); + } + /** + * Render one octant (45-degree arc) of the viewshed + * @param {int} x + * @param {int} y + * @param {int} octant Octant to be rendered + * @param {int} R Maximum visibility radius + * @param {function} callback + */ + _renderOctant(x, y, octant, R, callback) { + //Radius incremented by 1 to provide same coverage area as other shadowcasting radiuses + this._castVisibility(x, y, 1, 1.0, 0.0, R + 1, octant[0], octant[1], octant[2], octant[3], callback); + } + /** + * Actually calculates the visibility + * @param {int} startX The starting X coordinate + * @param {int} startY The starting Y coordinate + * @param {int} row The row to render + * @param {float} visSlopeStart The slope to start at + * @param {float} visSlopeEnd The slope to end at + * @param {int} radius The radius to reach out to + * @param {int} xx + * @param {int} xy + * @param {int} yx + * @param {int} yy + * @param {function} callback The callback to use when we hit a block that is visible + */ + _castVisibility(startX, startY, row, visSlopeStart, visSlopeEnd, radius, xx, xy, yx, yy, callback) { + if (visSlopeStart < visSlopeEnd) { + return; + } + for (let i = row; i <= radius; i++) { + let dx = -i - 1; + let dy = -i; + let blocked = false; + let newStart = 0; + //'Row' could be column, names here assume octant 0 and would be flipped for half the octants + while (dx <= 0) { + dx += 1; + //Translate from relative coordinates to map coordinates + let mapX = startX + dx * xx + dy * xy; + let mapY = startY + dx * yx + dy * yy; + //Range of the row + let slopeStart = (dx - 0.5) / (dy + 0.5); + let slopeEnd = (dx + 0.5) / (dy - 0.5); + //Ignore if not yet at left edge of Octant + if (slopeEnd > visSlopeStart) { + continue; + } + //Done if past right edge + if (slopeStart < visSlopeEnd) { + break; + } + //If it's in range, it's visible + if ((dx * dx + dy * dy) < (radius * radius)) { + callback(mapX, mapY, i, 1); + } + if (!blocked) { + //If tile is a blocking tile, cast around it + if (!this._lightPasses(mapX, mapY) && i < radius) { + blocked = true; + this._castVisibility(startX, startY, i + 1, visSlopeStart, slopeStart, radius, xx, xy, yx, yy, callback); + newStart = slopeEnd; + } + } + else { + //Keep narrowing if scanning across a block + if (!this._lightPasses(mapX, mapY)) { + newStart = slopeEnd; + continue; + } + //Block has ended + blocked = false; + visSlopeStart = newStart; + } + } + if (blocked) { + break; + } + } + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/fov/index.js + + + +/* harmony default export */ var fov = ({ DiscreteShadowcasting: discrete_shadowcasting_DiscreteShadowcasting, PreciseShadowcasting: precise_shadowcasting_PreciseShadowcasting, RecursiveShadowcasting: recursive_shadowcasting_RecursiveShadowcasting }); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/map.js + +; +class map_Map { + /** + * @class Base map generator + * @param {int} [width=ROT.DEFAULT_WIDTH] + * @param {int} [height=ROT.DEFAULT_HEIGHT] + */ + constructor(width = DEFAULT_WIDTH, height = DEFAULT_HEIGHT) { + this._width = width; + this._height = height; + } + ; + _fillMap(value) { + let map = []; + for (let i = 0; i < this._width; i++) { + map.push([]); + for (let j = 0; j < this._height; j++) { + map[i].push(value); + } + } + return map; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/arena.js + +/** + * @class Simple empty rectangular room + * @augments ROT.Map + */ +class arena_Arena extends map_Map { + create(callback) { + let w = this._width - 1; + let h = this._height - 1; + for (let i = 0; i <= w; i++) { + for (let j = 0; j <= h; j++) { + let empty = (i && j && i < w && j < h); + callback(i, j, empty ? 0 : 1); + } + } + return this; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/dungeon.js + +/** + * @class Dungeon map: has rooms and corridors + * @augments ROT.Map + */ +class dungeon_Dungeon extends map_Map { + constructor(width, height) { + super(width, height); + this._rooms = []; + this._corridors = []; + } + /** + * Get all generated rooms + * @returns {ROT.Map.Feature.Room[]} + */ + getRooms() { return this._rooms; } + /** + * Get all generated corridors + * @returns {ROT.Map.Feature.Corridor[]} + */ + getCorridors() { return this._corridors; } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/features.js + +; +/** + * @class Dungeon feature; has own .create() method + */ +class Feature { +} +/** + * @class Room + * @augments ROT.Map.Feature + * @param {int} x1 + * @param {int} y1 + * @param {int} x2 + * @param {int} y2 + * @param {int} [doorX] + * @param {int} [doorY] + */ +class features_Room extends Feature { + constructor(x1, y1, x2, y2, doorX, doorY) { + super(); + this._x1 = x1; + this._y1 = y1; + this._x2 = x2; + this._y2 = y2; + this._doors = {}; + if (doorX !== undefined && doorY !== undefined) { + this.addDoor(doorX, doorY); + } + } + ; + /** + * Room of random size, with a given doors and direction + */ + static createRandomAt(x, y, dx, dy, options) { + let min = options.roomWidth[0]; + let max = options.roomWidth[1]; + let width = rng["a" /* default */].getUniformInt(min, max); + min = options.roomHeight[0]; + max = options.roomHeight[1]; + let height = rng["a" /* default */].getUniformInt(min, max); + if (dx == 1) { /* to the right */ + let y2 = y - Math.floor(rng["a" /* default */].getUniform() * height); + return new this(x + 1, y2, x + width, y2 + height - 1, x, y); + } + if (dx == -1) { /* to the left */ + let y2 = y - Math.floor(rng["a" /* default */].getUniform() * height); + return new this(x - width, y2, x - 1, y2 + height - 1, x, y); + } + if (dy == 1) { /* to the bottom */ + let x2 = x - Math.floor(rng["a" /* default */].getUniform() * width); + return new this(x2, y + 1, x2 + width - 1, y + height, x, y); + } + if (dy == -1) { /* to the top */ + let x2 = x - Math.floor(rng["a" /* default */].getUniform() * width); + return new this(x2, y - height, x2 + width - 1, y - 1, x, y); + } + throw new Error("dx or dy must be 1 or -1"); + } + /** + * Room of random size, positioned around center coords + */ + static createRandomCenter(cx, cy, options) { + let min = options.roomWidth[0]; + let max = options.roomWidth[1]; + let width = rng["a" /* default */].getUniformInt(min, max); + min = options.roomHeight[0]; + max = options.roomHeight[1]; + let height = rng["a" /* default */].getUniformInt(min, max); + let x1 = cx - Math.floor(rng["a" /* default */].getUniform() * width); + let y1 = cy - Math.floor(rng["a" /* default */].getUniform() * height); + let x2 = x1 + width - 1; + let y2 = y1 + height - 1; + return new this(x1, y1, x2, y2); + } + /** + * Room of random size within a given dimensions + */ + static createRandom(availWidth, availHeight, options) { + let min = options.roomWidth[0]; + let max = options.roomWidth[1]; + let width = rng["a" /* default */].getUniformInt(min, max); + min = options.roomHeight[0]; + max = options.roomHeight[1]; + let height = rng["a" /* default */].getUniformInt(min, max); + let left = availWidth - width - 1; + let top = availHeight - height - 1; + let x1 = 1 + Math.floor(rng["a" /* default */].getUniform() * left); + let y1 = 1 + Math.floor(rng["a" /* default */].getUniform() * top); + let x2 = x1 + width - 1; + let y2 = y1 + height - 1; + return new this(x1, y1, x2, y2); + } + addDoor(x, y) { + this._doors[x + "," + y] = 1; + return this; + } + /** + * @param {function} + */ + getDoors(cb) { + for (let key in this._doors) { + let parts = key.split(","); + cb(parseInt(parts[0]), parseInt(parts[1])); + } + return this; + } + clearDoors() { + this._doors = {}; + return this; + } + addDoors(isWallCallback) { + let left = this._x1 - 1; + let right = this._x2 + 1; + let top = this._y1 - 1; + let bottom = this._y2 + 1; + for (let x = left; x <= right; x++) { + for (let y = top; y <= bottom; y++) { + if (x != left && x != right && y != top && y != bottom) { + continue; + } + if (isWallCallback(x, y)) { + continue; + } + this.addDoor(x, y); + } + } + return this; + } + debug() { + console.log("room", this._x1, this._y1, this._x2, this._y2); + } + isValid(isWallCallback, canBeDugCallback) { + let left = this._x1 - 1; + let right = this._x2 + 1; + let top = this._y1 - 1; + let bottom = this._y2 + 1; + for (let x = left; x <= right; x++) { + for (let y = top; y <= bottom; y++) { + if (x == left || x == right || y == top || y == bottom) { + if (!isWallCallback(x, y)) { + return false; + } + } + else { + if (!canBeDugCallback(x, y)) { + return false; + } + } + } + } + return true; + } + /** + * @param {function} digCallback Dig callback with a signature (x, y, value). Values: 0 = empty, 1 = wall, 2 = door. Multiple doors are allowed. + */ + create(digCallback) { + let left = this._x1 - 1; + let right = this._x2 + 1; + let top = this._y1 - 1; + let bottom = this._y2 + 1; + let value = 0; + for (let x = left; x <= right; x++) { + for (let y = top; y <= bottom; y++) { + if (x + "," + y in this._doors) { + value = 2; + } + else if (x == left || x == right || y == top || y == bottom) { + value = 1; + } + else { + value = 0; + } + digCallback(x, y, value); + } + } + } + getCenter() { + return [Math.round((this._x1 + this._x2) / 2), Math.round((this._y1 + this._y2) / 2)]; + } + getLeft() { return this._x1; } + getRight() { return this._x2; } + getTop() { return this._y1; } + getBottom() { return this._y2; } +} +/** + * @class Corridor + * @augments ROT.Map.Feature + * @param {int} startX + * @param {int} startY + * @param {int} endX + * @param {int} endY + */ +class features_Corridor extends Feature { + constructor(startX, startY, endX, endY) { + super(); + this._startX = startX; + this._startY = startY; + this._endX = endX; + this._endY = endY; + this._endsWithAWall = true; + } + static createRandomAt(x, y, dx, dy, options) { + let min = options.corridorLength[0]; + let max = options.corridorLength[1]; + let length = rng["a" /* default */].getUniformInt(min, max); + return new this(x, y, x + dx * length, y + dy * length); + } + debug() { + console.log("corridor", this._startX, this._startY, this._endX, this._endY); + } + isValid(isWallCallback, canBeDugCallback) { + let sx = this._startX; + let sy = this._startY; + let dx = this._endX - sx; + let dy = this._endY - sy; + let length = 1 + Math.max(Math.abs(dx), Math.abs(dy)); + if (dx) { + dx = dx / Math.abs(dx); + } + if (dy) { + dy = dy / Math.abs(dy); + } + let nx = dy; + let ny = -dx; + let ok = true; + for (let i = 0; i < length; i++) { + let x = sx + i * dx; + let y = sy + i * dy; + if (!canBeDugCallback(x, y)) { + ok = false; + } + if (!isWallCallback(x + nx, y + ny)) { + ok = false; + } + if (!isWallCallback(x - nx, y - ny)) { + ok = false; + } + if (!ok) { + length = i; + this._endX = x - dx; + this._endY = y - dy; + break; + } + } + /** + * If the length degenerated, this corridor might be invalid + */ + /* not supported */ + if (length == 0) { + return false; + } + /* length 1 allowed only if the next space is empty */ + if (length == 1 && isWallCallback(this._endX + dx, this._endY + dy)) { + return false; + } + /** + * We do not want the corridor to crash into a corner of a room; + * if any of the ending corners is empty, the N+1th cell of this corridor must be empty too. + * + * Situation: + * #######1 + * .......? + * #######2 + * + * The corridor was dug from left to right. + * 1, 2 - problematic corners, ? = N+1th cell (not dug) + */ + let firstCornerBad = !isWallCallback(this._endX + dx + nx, this._endY + dy + ny); + let secondCornerBad = !isWallCallback(this._endX + dx - nx, this._endY + dy - ny); + this._endsWithAWall = isWallCallback(this._endX + dx, this._endY + dy); + if ((firstCornerBad || secondCornerBad) && this._endsWithAWall) { + return false; + } + return true; + } + /** + * @param {function} digCallback Dig callback with a signature (x, y, value). Values: 0 = empty. + */ + create(digCallback) { + let sx = this._startX; + let sy = this._startY; + let dx = this._endX - sx; + let dy = this._endY - sy; + let length = 1 + Math.max(Math.abs(dx), Math.abs(dy)); + if (dx) { + dx = dx / Math.abs(dx); + } + if (dy) { + dy = dy / Math.abs(dy); + } + for (let i = 0; i < length; i++) { + let x = sx + i * dx; + let y = sy + i * dy; + digCallback(x, y, 0); + } + return true; + } + createPriorityWalls(priorityWallCallback) { + if (!this._endsWithAWall) { + return; + } + let sx = this._startX; + let sy = this._startY; + let dx = this._endX - sx; + let dy = this._endY - sy; + if (dx) { + dx = dx / Math.abs(dx); + } + if (dy) { + dy = dy / Math.abs(dy); + } + let nx = dy; + let ny = -dx; + priorityWallCallback(this._endX + dx, this._endY + dy); + priorityWallCallback(this._endX + nx, this._endY + ny); + priorityWallCallback(this._endX - nx, this._endY - ny); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/uniform.js + + + +; +/** + * @class Dungeon generator which tries to fill the space evenly. Generates independent rooms and tries to connect them. + * @augments ROT.Map.Dungeon + */ +class uniform_Uniform extends dungeon_Dungeon { + constructor(width, height, options) { + super(width, height); + this._options = { + roomWidth: [3, 9], + roomHeight: [3, 5], + roomDugPercentage: 0.1, + timeLimit: 1000 /* we stop after this much time has passed (msec) */ + }; + Object.assign(this._options, options); + this._map = []; + this._dug = 0; + this._roomAttempts = 20; /* new room is created N-times until is considered as impossible to generate */ + this._corridorAttempts = 20; /* corridors are tried N-times until the level is considered as impossible to connect */ + this._connected = []; /* list of already connected rooms */ + this._unconnected = []; /* list of remaining unconnected rooms */ + this._digCallback = this._digCallback.bind(this); + this._canBeDugCallback = this._canBeDugCallback.bind(this); + this._isWallCallback = this._isWallCallback.bind(this); + } + /** + * Create a map. If the time limit has been hit, returns null. + * @see ROT.Map#create + */ + create(callback) { + let t1 = Date.now(); + while (1) { + let t2 = Date.now(); + if (t2 - t1 > this._options.timeLimit) { + return null; + } /* time limit! */ + this._map = this._fillMap(1); + this._dug = 0; + this._rooms = []; + this._unconnected = []; + this._generateRooms(); + if (this._rooms.length < 2) { + continue; + } + if (this._generateCorridors()) { + break; + } + } + if (callback) { + for (let i = 0; i < this._width; i++) { + for (let j = 0; j < this._height; j++) { + callback(i, j, this._map[i][j]); + } + } + } + return this; + } + /** + * Generates a suitable amount of rooms + */ + _generateRooms() { + let w = this._width - 2; + let h = this._height - 2; + let room; + do { + room = this._generateRoom(); + if (this._dug / (w * h) > this._options.roomDugPercentage) { + break; + } /* achieved requested amount of free space */ + } while (room); + /* either enough rooms, or not able to generate more of them :) */ + } + /** + * Try to generate one room + */ + _generateRoom() { + let count = 0; + while (count < this._roomAttempts) { + count++; + let room = features_Room.createRandom(this._width, this._height, this._options); + if (!room.isValid(this._isWallCallback, this._canBeDugCallback)) { + continue; + } + room.create(this._digCallback); + this._rooms.push(room); + return room; + } + /* no room was generated in a given number of attempts */ + return null; + } + /** + * Generates connectors beween rooms + * @returns {bool} success Was this attempt successfull? + */ + _generateCorridors() { + let cnt = 0; + while (cnt < this._corridorAttempts) { + cnt++; + this._corridors = []; + /* dig rooms into a clear map */ + this._map = this._fillMap(1); + for (let i = 0; i < this._rooms.length; i++) { + let room = this._rooms[i]; + room.clearDoors(); + room.create(this._digCallback); + } + this._unconnected = rng["a" /* default */].shuffle(this._rooms.slice()); + this._connected = []; + if (this._unconnected.length) { + this._connected.push(this._unconnected.pop()); + } /* first one is always connected */ + while (1) { + /* 1. pick random connected room */ + let connected = rng["a" /* default */].getItem(this._connected); + if (!connected) { + break; + } + /* 2. find closest unconnected */ + let room1 = this._closestRoom(this._unconnected, connected); + if (!room1) { + break; + } + /* 3. connect it to closest connected */ + let room2 = this._closestRoom(this._connected, room1); + if (!room2) { + break; + } + let ok = this._connectRooms(room1, room2); + if (!ok) { + break; + } /* stop connecting, re-shuffle */ + if (!this._unconnected.length) { + return true; + } /* done; no rooms remain */ + } + } + return false; + } + ; + /** + * For a given room, find the closest one from the list + */ + _closestRoom(rooms, room) { + let dist = Infinity; + let center = room.getCenter(); + let result = null; + for (let i = 0; i < rooms.length; i++) { + let r = rooms[i]; + let c = r.getCenter(); + let dx = c[0] - center[0]; + let dy = c[1] - center[1]; + let d = dx * dx + dy * dy; + if (d < dist) { + dist = d; + result = r; + } + } + return result; + } + _connectRooms(room1, room2) { + /* + room1.debug(); + room2.debug(); + */ + let center1 = room1.getCenter(); + let center2 = room2.getCenter(); + let diffX = center2[0] - center1[0]; + let diffY = center2[1] - center1[1]; + let start; + let end; + let dirIndex1, dirIndex2, min, max, index; + if (Math.abs(diffX) < Math.abs(diffY)) { /* first try connecting north-south walls */ + dirIndex1 = (diffY > 0 ? 2 : 0); + dirIndex2 = (dirIndex1 + 2) % 4; + min = room2.getLeft(); + max = room2.getRight(); + index = 0; + } + else { /* first try connecting east-west walls */ + dirIndex1 = (diffX > 0 ? 1 : 3); + dirIndex2 = (dirIndex1 + 2) % 4; + min = room2.getTop(); + max = room2.getBottom(); + index = 1; + } + start = this._placeInWall(room1, dirIndex1); /* corridor will start here */ + if (!start) { + return false; + } + if (start[index] >= min && start[index] <= max) { /* possible to connect with straight line (I-like) */ + end = start.slice(); + let value = 0; + switch (dirIndex2) { + case 0: + value = room2.getTop() - 1; + break; + case 1: + value = room2.getRight() + 1; + break; + case 2: + value = room2.getBottom() + 1; + break; + case 3: + value = room2.getLeft() - 1; + break; + } + end[(index + 1) % 2] = value; + this._digLine([start, end]); + } + else if (start[index] < min - 1 || start[index] > max + 1) { /* need to switch target wall (L-like) */ + let diff = start[index] - center2[index]; + let rotation = 0; + switch (dirIndex2) { + case 0: + case 1: + rotation = (diff < 0 ? 3 : 1); + break; + case 2: + case 3: + rotation = (diff < 0 ? 1 : 3); + break; + } + dirIndex2 = (dirIndex2 + rotation) % 4; + end = this._placeInWall(room2, dirIndex2); + if (!end) { + return false; + } + let mid = [0, 0]; + mid[index] = start[index]; + let index2 = (index + 1) % 2; + mid[index2] = end[index2]; + this._digLine([start, mid, end]); + } + else { /* use current wall pair, but adjust the line in the middle (S-like) */ + let index2 = (index + 1) % 2; + end = this._placeInWall(room2, dirIndex2); + if (!end) { + return false; + } + let mid = Math.round((end[index2] + start[index2]) / 2); + let mid1 = [0, 0]; + let mid2 = [0, 0]; + mid1[index] = start[index]; + mid1[index2] = mid; + mid2[index] = end[index]; + mid2[index2] = mid; + this._digLine([start, mid1, mid2, end]); + } + room1.addDoor(start[0], start[1]); + room2.addDoor(end[0], end[1]); + index = this._unconnected.indexOf(room1); + if (index != -1) { + this._unconnected.splice(index, 1); + this._connected.push(room1); + } + index = this._unconnected.indexOf(room2); + if (index != -1) { + this._unconnected.splice(index, 1); + this._connected.push(room2); + } + return true; + } + _placeInWall(room, dirIndex) { + let start = [0, 0]; + let dir = [0, 0]; + let length = 0; + switch (dirIndex) { + case 0: + dir = [1, 0]; + start = [room.getLeft(), room.getTop() - 1]; + length = room.getRight() - room.getLeft() + 1; + break; + case 1: + dir = [0, 1]; + start = [room.getRight() + 1, room.getTop()]; + length = room.getBottom() - room.getTop() + 1; + break; + case 2: + dir = [1, 0]; + start = [room.getLeft(), room.getBottom() + 1]; + length = room.getRight() - room.getLeft() + 1; + break; + case 3: + dir = [0, 1]; + start = [room.getLeft() - 1, room.getTop()]; + length = room.getBottom() - room.getTop() + 1; + break; + } + let avail = []; + let lastBadIndex = -2; + for (let i = 0; i < length; i++) { + let x = start[0] + i * dir[0]; + let y = start[1] + i * dir[1]; + avail.push(null); + let isWall = (this._map[x][y] == 1); + if (isWall) { + if (lastBadIndex != i - 1) { + avail[i] = [x, y]; + } + } + else { + lastBadIndex = i; + if (i) { + avail[i - 1] = null; + } + } + } + for (let i = avail.length - 1; i >= 0; i--) { + if (!avail[i]) { + avail.splice(i, 1); + } + } + return (avail.length ? rng["a" /* default */].getItem(avail) : null); + } + /** + * Dig a polyline. + */ + _digLine(points) { + for (let i = 1; i < points.length; i++) { + let start = points[i - 1]; + let end = points[i]; + let corridor = new features_Corridor(start[0], start[1], end[0], end[1]); + corridor.create(this._digCallback); + this._corridors.push(corridor); + } + } + _digCallback(x, y, value) { + this._map[x][y] = value; + if (value == 0) { + this._dug++; + } + } + _isWallCallback(x, y) { + if (x < 0 || y < 0 || x >= this._width || y >= this._height) { + return false; + } + return (this._map[x][y] == 1); + } + _canBeDugCallback(x, y) { + if (x < 1 || y < 1 || x + 1 >= this._width || y + 1 >= this._height) { + return false; + } + return (this._map[x][y] == 1); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/cellular.js + + + +; +/** + * @class Cellular automaton map generator + * @augments ROT.Map + * @param {int} [width=ROT.DEFAULT_WIDTH] + * @param {int} [height=ROT.DEFAULT_HEIGHT] + * @param {object} [options] Options + * @param {int[]} [options.born] List of neighbor counts for a new cell to be born in empty space + * @param {int[]} [options.survive] List of neighbor counts for an existing cell to survive + * @param {int} [options.topology] Topology 4 or 6 or 8 + */ +class cellular_Cellular extends map_Map { + constructor(width, height, options = {}) { + super(width, height); + this._options = { + born: [5, 6, 7, 8], + survive: [4, 5, 6, 7, 8], + topology: 8 + }; + this.setOptions(options); + this._dirs = DIRS[this._options.topology]; + this._map = this._fillMap(0); + } + /** + * Fill the map with random values + * @param {float} probability Probability for a cell to become alive; 0 = all empty, 1 = all full + */ + randomize(probability) { + for (let i = 0; i < this._width; i++) { + for (let j = 0; j < this._height; j++) { + this._map[i][j] = (rng["a" /* default */].getUniform() < probability ? 1 : 0); + } + } + return this; + } + /** + * Change options. + * @see ROT.Map.Cellular + */ + setOptions(options) { Object.assign(this._options, options); } + set(x, y, value) { this._map[x][y] = value; } + create(callback) { + let newMap = this._fillMap(0); + let born = this._options.born; + let survive = this._options.survive; + for (let j = 0; j < this._height; j++) { + let widthStep = 1; + let widthStart = 0; + if (this._options.topology == 6) { + widthStep = 2; + widthStart = j % 2; + } + for (let i = widthStart; i < this._width; i += widthStep) { + let cur = this._map[i][j]; + let ncount = this._getNeighbors(i, j); + if (cur && survive.indexOf(ncount) != -1) { /* survive */ + newMap[i][j] = 1; + } + else if (!cur && born.indexOf(ncount) != -1) { /* born */ + newMap[i][j] = 1; + } + } + } + this._map = newMap; + callback && this._serviceCallback(callback); + } + _serviceCallback(callback) { + for (let j = 0; j < this._height; j++) { + let widthStep = 1; + let widthStart = 0; + if (this._options.topology == 6) { + widthStep = 2; + widthStart = j % 2; + } + for (let i = widthStart; i < this._width; i += widthStep) { + callback(i, j, this._map[i][j]); + } + } + } + /** + * Get neighbor count at [i,j] in this._map + */ + _getNeighbors(cx, cy) { + let result = 0; + for (let i = 0; i < this._dirs.length; i++) { + let dir = this._dirs[i]; + let x = cx + dir[0]; + let y = cy + dir[1]; + if (x < 0 || x >= this._width || y < 0 || y >= this._height) { + continue; + } + result += (this._map[x][y] == 1 ? 1 : 0); + } + return result; + } + /** + * Make sure every non-wall space is accessible. + * @param {function} callback to call to display map when do + * @param {int} value to consider empty space - defaults to 0 + * @param {function} callback to call when a new connection is made + */ + connect(callback, value, connectionCallback) { + if (!value) + value = 0; + let allFreeSpace = []; + let notConnected = {}; + // find all free space + let widthStep = 1; + let widthStarts = [0, 0]; + if (this._options.topology == 6) { + widthStep = 2; + widthStarts = [0, 1]; + } + for (let y = 0; y < this._height; y++) { + for (let x = widthStarts[y % 2]; x < this._width; x += widthStep) { + if (this._freeSpace(x, y, value)) { + let p = [x, y]; + notConnected[this._pointKey(p)] = p; + allFreeSpace.push([x, y]); + } + } + } + let start = allFreeSpace[rng["a" /* default */].getUniformInt(0, allFreeSpace.length - 1)]; + let key = this._pointKey(start); + let connected = {}; + connected[key] = start; + delete notConnected[key]; + // find what's connected to the starting point + this._findConnected(connected, notConnected, [start], false, value); + while (Object.keys(notConnected).length > 0) { + // find two points from notConnected to connected + let p = this._getFromTo(connected, notConnected); + let from = p[0]; // notConnected + let to = p[1]; // connected + // find everything connected to the starting point + let local = {}; + local[this._pointKey(from)] = from; + this._findConnected(local, notConnected, [from], true, value); + // connect to a connected cell + let tunnelFn = (this._options.topology == 6 ? this._tunnelToConnected6 : this._tunnelToConnected); + tunnelFn.call(this, to, from, connected, notConnected, value, connectionCallback); + // now all of local is connected + for (let k in local) { + let pp = local[k]; + this._map[pp[0]][pp[1]] = value; + connected[k] = pp; + delete notConnected[k]; + } + } + callback && this._serviceCallback(callback); + } + /** + * Find random points to connect. Search for the closest point in the larger space. + * This is to minimize the length of the passage while maintaining good performance. + */ + _getFromTo(connected, notConnected) { + let from = [0, 0], to = [0, 0], d; + let connectedKeys = Object.keys(connected); + let notConnectedKeys = Object.keys(notConnected); + for (let i = 0; i < 5; i++) { + if (connectedKeys.length < notConnectedKeys.length) { + let keys = connectedKeys; + to = connected[keys[rng["a" /* default */].getUniformInt(0, keys.length - 1)]]; + from = this._getClosest(to, notConnected); + } + else { + let keys = notConnectedKeys; + from = notConnected[keys[rng["a" /* default */].getUniformInt(0, keys.length - 1)]]; + to = this._getClosest(from, connected); + } + d = (from[0] - to[0]) * (from[0] - to[0]) + (from[1] - to[1]) * (from[1] - to[1]); + if (d < 64) { + break; + } + } + // console.log(">>> connected=" + to + " notConnected=" + from + " dist=" + d); + return [from, to]; + } + _getClosest(point, space) { + let minPoint = null; + let minDist = null; + for (let k in space) { + let p = space[k]; + let d = (p[0] - point[0]) * (p[0] - point[0]) + (p[1] - point[1]) * (p[1] - point[1]); + if (minDist == null || d < minDist) { + minDist = d; + minPoint = p; + } + } + return minPoint; + } + _findConnected(connected, notConnected, stack, keepNotConnected, value) { + while (stack.length > 0) { + let p = stack.splice(0, 1)[0]; + let tests; + if (this._options.topology == 6) { + tests = [ + [p[0] + 2, p[1]], + [p[0] + 1, p[1] - 1], + [p[0] - 1, p[1] - 1], + [p[0] - 2, p[1]], + [p[0] - 1, p[1] + 1], + [p[0] + 1, p[1] + 1], + ]; + } + else { + tests = [ + [p[0] + 1, p[1]], + [p[0] - 1, p[1]], + [p[0], p[1] + 1], + [p[0], p[1] - 1] + ]; + } + for (let i = 0; i < tests.length; i++) { + let key = this._pointKey(tests[i]); + if (connected[key] == null && this._freeSpace(tests[i][0], tests[i][1], value)) { + connected[key] = tests[i]; + if (!keepNotConnected) { + delete notConnected[key]; + } + stack.push(tests[i]); + } + } + } + } + _tunnelToConnected(to, from, connected, notConnected, value, connectionCallback) { + let a, b; + if (from[0] < to[0]) { + a = from; + b = to; + } + else { + a = to; + b = from; + } + for (let xx = a[0]; xx <= b[0]; xx++) { + this._map[xx][a[1]] = value; + let p = [xx, a[1]]; + let pkey = this._pointKey(p); + connected[pkey] = p; + delete notConnected[pkey]; + } + if (connectionCallback && a[0] < b[0]) { + connectionCallback(a, [b[0], a[1]]); + } + // x is now fixed + let x = b[0]; + if (from[1] < to[1]) { + a = from; + b = to; + } + else { + a = to; + b = from; + } + for (let yy = a[1]; yy < b[1]; yy++) { + this._map[x][yy] = value; + let p = [x, yy]; + let pkey = this._pointKey(p); + connected[pkey] = p; + delete notConnected[pkey]; + } + if (connectionCallback && a[1] < b[1]) { + connectionCallback([b[0], a[1]], [b[0], b[1]]); + } + } + _tunnelToConnected6(to, from, connected, notConnected, value, connectionCallback) { + let a, b; + if (from[0] < to[0]) { + a = from; + b = to; + } + else { + a = to; + b = from; + } + // tunnel diagonally until horizontally level + let xx = a[0]; + let yy = a[1]; + while (!(xx == b[0] && yy == b[1])) { + let stepWidth = 2; + if (yy < b[1]) { + yy++; + stepWidth = 1; + } + else if (yy > b[1]) { + yy--; + stepWidth = 1; + } + if (xx < b[0]) { + xx += stepWidth; + } + else if (xx > b[0]) { + xx -= stepWidth; + } + else if (b[1] % 2) { + // Won't step outside map if destination on is map's right edge + xx -= stepWidth; + } + else { + // ditto for left edge + xx += stepWidth; + } + this._map[xx][yy] = value; + let p = [xx, yy]; + let pkey = this._pointKey(p); + connected[pkey] = p; + delete notConnected[pkey]; + } + if (connectionCallback) { + connectionCallback(from, to); + } + } + _freeSpace(x, y, value) { + return x >= 0 && x < this._width && y >= 0 && y < this._height && this._map[x][y] == value; + } + _pointKey(p) { return p[0] + "." + p[1]; } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/digger.js + + + + +const FEATURES = { + "room": features_Room, + "corridor": features_Corridor +}; +/** + * Random dungeon generator using human-like digging patterns. + * Heavily based on Mike Anderson's ideas from the "Tyrant" algo, mentioned at + * http://www.roguebasin.roguelikedevelopment.org/index.php?title=Dungeon-Building_Algorithm. + */ +class digger_Digger extends dungeon_Dungeon { + constructor(width, height, options = {}) { + super(width, height); + this._options = Object.assign({ + roomWidth: [3, 9], + roomHeight: [3, 5], + corridorLength: [3, 10], + dugPercentage: 0.2, + timeLimit: 1000 /* we stop after this much time has passed (msec) */ + }, options); + this._features = { + "room": 4, + "corridor": 4 + }; + this._map = []; + this._featureAttempts = 20; /* how many times do we try to create a feature on a suitable wall */ + this._walls = {}; /* these are available for digging */ + this._dug = 0; + this._digCallback = this._digCallback.bind(this); + this._canBeDugCallback = this._canBeDugCallback.bind(this); + this._isWallCallback = this._isWallCallback.bind(this); + this._priorityWallCallback = this._priorityWallCallback.bind(this); + } + create(callback) { + this._rooms = []; + this._corridors = []; + this._map = this._fillMap(1); + this._walls = {}; + this._dug = 0; + let area = (this._width - 2) * (this._height - 2); + this._firstRoom(); + let t1 = Date.now(); + let priorityWalls; + do { + priorityWalls = 0; + let t2 = Date.now(); + if (t2 - t1 > this._options.timeLimit) { + break; + } + /* find a good wall */ + let wall = this._findWall(); + if (!wall) { + break; + } /* no more walls */ + let parts = wall.split(","); + let x = parseInt(parts[0]); + let y = parseInt(parts[1]); + let dir = this._getDiggingDirection(x, y); + if (!dir) { + continue; + } /* this wall is not suitable */ + // console.log("wall", x, y); + /* try adding a feature */ + let featureAttempts = 0; + do { + featureAttempts++; + if (this._tryFeature(x, y, dir[0], dir[1])) { /* feature added */ + //if (this._rooms.length + this._corridors.length == 2) { this._rooms[0].addDoor(x, y); } /* first room oficially has doors */ + this._removeSurroundingWalls(x, y); + this._removeSurroundingWalls(x - dir[0], y - dir[1]); + break; + } + } while (featureAttempts < this._featureAttempts); + for (let id in this._walls) { + if (this._walls[id] > 1) { + priorityWalls++; + } + } + } while (this._dug / area < this._options.dugPercentage || priorityWalls); /* fixme number of priority walls */ + this._addDoors(); + if (callback) { + for (let i = 0; i < this._width; i++) { + for (let j = 0; j < this._height; j++) { + callback(i, j, this._map[i][j]); + } + } + } + this._walls = {}; + this._map = []; + return this; + } + _digCallback(x, y, value) { + if (value == 0 || value == 2) { /* empty */ + this._map[x][y] = 0; + this._dug++; + } + else { /* wall */ + this._walls[x + "," + y] = 1; + } + } + _isWallCallback(x, y) { + if (x < 0 || y < 0 || x >= this._width || y >= this._height) { + return false; + } + return (this._map[x][y] == 1); + } + _canBeDugCallback(x, y) { + if (x < 1 || y < 1 || x + 1 >= this._width || y + 1 >= this._height) { + return false; + } + return (this._map[x][y] == 1); + } + _priorityWallCallback(x, y) { this._walls[x + "," + y] = 2; } + ; + _firstRoom() { + let cx = Math.floor(this._width / 2); + let cy = Math.floor(this._height / 2); + let room = features_Room.createRandomCenter(cx, cy, this._options); + this._rooms.push(room); + room.create(this._digCallback); + } + /** + * Get a suitable wall + */ + _findWall() { + let prio1 = []; + let prio2 = []; + for (let id in this._walls) { + let prio = this._walls[id]; + if (prio == 2) { + prio2.push(id); + } + else { + prio1.push(id); + } + } + let arr = (prio2.length ? prio2 : prio1); + if (!arr.length) { + return null; + } /* no walls :/ */ + let id = rng["a" /* default */].getItem(arr.sort()); // sort to make the order deterministic + delete this._walls[id]; + return id; + } + /** + * Tries adding a feature + * @returns {bool} was this a successful try? + */ + _tryFeature(x, y, dx, dy) { + let featureName = rng["a" /* default */].getWeightedValue(this._features); + let ctor = FEATURES[featureName]; + let feature = ctor.createRandomAt(x, y, dx, dy, this._options); + if (!feature.isValid(this._isWallCallback, this._canBeDugCallback)) { + // console.log("not valid"); + // feature.debug(); + return false; + } + feature.create(this._digCallback); + // feature.debug(); + if (feature instanceof features_Room) { + this._rooms.push(feature); + } + if (feature instanceof features_Corridor) { + feature.createPriorityWalls(this._priorityWallCallback); + this._corridors.push(feature); + } + return true; + } + _removeSurroundingWalls(cx, cy) { + let deltas = DIRS[4]; + for (let i = 0; i < deltas.length; i++) { + let delta = deltas[i]; + let x = cx + delta[0]; + let y = cy + delta[1]; + delete this._walls[x + "," + y]; + x = cx + 2 * delta[0]; + y = cy + 2 * delta[1]; + delete this._walls[x + "," + y]; + } + } + /** + * Returns vector in "digging" direction, or false, if this does not exist (or is not unique) + */ + _getDiggingDirection(cx, cy) { + if (cx <= 0 || cy <= 0 || cx >= this._width - 1 || cy >= this._height - 1) { + return null; + } + let result = null; + let deltas = DIRS[4]; + for (let i = 0; i < deltas.length; i++) { + let delta = deltas[i]; + let x = cx + delta[0]; + let y = cy + delta[1]; + if (!this._map[x][y]) { /* there already is another empty neighbor! */ + if (result) { + return null; + } + result = delta; + } + } + /* no empty neighbor */ + if (!result) { + return null; + } + return [-result[0], -result[1]]; + } + /** + * Find empty spaces surrounding rooms, and apply doors. + */ + _addDoors() { + let data = this._map; + function isWallCallback(x, y) { + return (data[x][y] == 1); + } + ; + for (let i = 0; i < this._rooms.length; i++) { + let room = this._rooms[i]; + room.clearDoors(); + room.addDoors(isWallCallback); + } + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/ellermaze.js + + +/** + * Join lists with "i" and "i+1" + */ +function addToList(i, L, R) { + R[L[i + 1]] = R[i]; + L[R[i]] = L[i + 1]; + R[i] = i + 1; + L[i + 1] = i; +} +/** + * Remove "i" from its list + */ +function removeFromList(i, L, R) { + R[L[i]] = R[i]; + L[R[i]] = L[i]; + R[i] = i; + L[i] = i; +} +/** + * Maze generator - Eller's algorithm + * See http://homepages.cwi.nl/~tromp/maze.html for explanation + */ +class ellermaze_EllerMaze extends map_Map { + create(callback) { + let map = this._fillMap(1); + let w = Math.ceil((this._width - 2) / 2); + let rand = 9 / 24; + let L = []; + let R = []; + for (let i = 0; i < w; i++) { + L.push(i); + R.push(i); + } + L.push(w - 1); /* fake stop-block at the right side */ + let j; + for (j = 1; j + 3 < this._height; j += 2) { + /* one row */ + for (let i = 0; i < w; i++) { + /* cell coords (will be always empty) */ + let x = 2 * i + 1; + let y = j; + map[x][y] = 0; + /* right connection */ + if (i != L[i + 1] && rng["a" /* default */].getUniform() > rand) { + addToList(i, L, R); + map[x + 1][y] = 0; + } + /* bottom connection */ + if (i != L[i] && rng["a" /* default */].getUniform() > rand) { + /* remove connection */ + removeFromList(i, L, R); + } + else { + /* create connection */ + map[x][y + 1] = 0; + } + } + } + /* last row */ + for (let i = 0; i < w; i++) { + /* cell coords (will be always empty) */ + let x = 2 * i + 1; + let y = j; + map[x][y] = 0; + /* right connection */ + if (i != L[i + 1] && (i == L[i] || rng["a" /* default */].getUniform() > rand)) { + /* dig right also if the cell is separated, so it gets connected to the rest of maze */ + addToList(i, L, R); + map[x + 1][y] = 0; + } + removeFromList(i, L, R); + } + for (let i = 0; i < this._width; i++) { + for (let j = 0; j < this._height; j++) { + callback(i, j, map[i][j]); + } + } + return this; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/dividedmaze.js + + +/** + * @class Recursively divided maze, http://en.wikipedia.org/wiki/Maze_generation_algorithm#Recursive_division_method + * @augments ROT.Map + */ +class dividedmaze_DividedMaze extends map_Map { + constructor() { + super(...arguments); + this._stack = []; + this._map = []; + } + create(callback) { + let w = this._width; + let h = this._height; + this._map = []; + for (let i = 0; i < w; i++) { + this._map.push([]); + for (let j = 0; j < h; j++) { + let border = (i == 0 || j == 0 || i + 1 == w || j + 1 == h); + this._map[i].push(border ? 1 : 0); + } + } + this._stack = [ + [1, 1, w - 2, h - 2] + ]; + this._process(); + for (let i = 0; i < w; i++) { + for (let j = 0; j < h; j++) { + callback(i, j, this._map[i][j]); + } + } + this._map = []; + return this; + } + _process() { + while (this._stack.length) { + let room = this._stack.shift(); /* [left, top, right, bottom] */ + this._partitionRoom(room); + } + } + _partitionRoom(room) { + let availX = []; + let availY = []; + for (let i = room[0] + 1; i < room[2]; i++) { + let top = this._map[i][room[1] - 1]; + let bottom = this._map[i][room[3] + 1]; + if (top && bottom && !(i % 2)) { + availX.push(i); + } + } + for (let j = room[1] + 1; j < room[3]; j++) { + let left = this._map[room[0] - 1][j]; + let right = this._map[room[2] + 1][j]; + if (left && right && !(j % 2)) { + availY.push(j); + } + } + if (!availX.length || !availY.length) { + return; + } + let x = rng["a" /* default */].getItem(availX); + let y = rng["a" /* default */].getItem(availY); + this._map[x][y] = 1; + let walls = []; + let w = []; + walls.push(w); /* left part */ + for (let i = room[0]; i < x; i++) { + this._map[i][y] = 1; + w.push([i, y]); + } + w = []; + walls.push(w); /* right part */ + for (let i = x + 1; i <= room[2]; i++) { + this._map[i][y] = 1; + w.push([i, y]); + } + w = []; + walls.push(w); /* top part */ + for (let j = room[1]; j < y; j++) { + this._map[x][j] = 1; + w.push([x, j]); + } + w = []; + walls.push(w); /* bottom part */ + for (let j = y + 1; j <= room[3]; j++) { + this._map[x][j] = 1; + w.push([x, j]); + } + let solid = rng["a" /* default */].getItem(walls); + for (let i = 0; i < walls.length; i++) { + let w = walls[i]; + if (w == solid) { + continue; + } + let hole = rng["a" /* default */].getItem(w); + this._map[hole[0]][hole[1]] = 0; + } + this._stack.push([room[0], room[1], x - 1, y - 1]); /* left top */ + this._stack.push([x + 1, room[1], room[2], y - 1]); /* right top */ + this._stack.push([room[0], y + 1, x - 1, room[3]]); /* left bottom */ + this._stack.push([x + 1, y + 1, room[2], room[3]]); /* right bottom */ + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/iceymaze.js + + +/** + * Icey's Maze generator + * See http://www.roguebasin.roguelikedevelopment.org/index.php?title=Simple_maze for explanation + */ +class iceymaze_IceyMaze extends map_Map { + constructor(width, height, regularity = 0) { + super(width, height); + this._regularity = regularity; + this._map = []; + } + create(callback) { + let width = this._width; + let height = this._height; + let map = this._fillMap(1); + width -= (width % 2 ? 1 : 2); + height -= (height % 2 ? 1 : 2); + let cx = 0; + let cy = 0; + let nx = 0; + let ny = 0; + let done = 0; + let blocked = false; + let dirs = [ + [0, 0], + [0, 0], + [0, 0], + [0, 0] + ]; + do { + cx = 1 + 2 * Math.floor(rng["a" /* default */].getUniform() * (width - 1) / 2); + cy = 1 + 2 * Math.floor(rng["a" /* default */].getUniform() * (height - 1) / 2); + if (!done) { + map[cx][cy] = 0; + } + if (!map[cx][cy]) { + this._randomize(dirs); + do { + if (Math.floor(rng["a" /* default */].getUniform() * (this._regularity + 1)) == 0) { + this._randomize(dirs); + } + blocked = true; + for (let i = 0; i < 4; i++) { + nx = cx + dirs[i][0] * 2; + ny = cy + dirs[i][1] * 2; + if (this._isFree(map, nx, ny, width, height)) { + map[nx][ny] = 0; + map[cx + dirs[i][0]][cy + dirs[i][1]] = 0; + cx = nx; + cy = ny; + blocked = false; + done++; + break; + } + } + } while (!blocked); + } + } while (done + 1 < width * height / 4); + for (let i = 0; i < this._width; i++) { + for (let j = 0; j < this._height; j++) { + callback(i, j, map[i][j]); + } + } + this._map = []; + return this; + } + _randomize(dirs) { + for (let i = 0; i < 4; i++) { + dirs[i][0] = 0; + dirs[i][1] = 0; + } + switch (Math.floor(rng["a" /* default */].getUniform() * 4)) { + case 0: + dirs[0][0] = -1; + dirs[1][0] = 1; + dirs[2][1] = -1; + dirs[3][1] = 1; + break; + case 1: + dirs[3][0] = -1; + dirs[2][0] = 1; + dirs[1][1] = -1; + dirs[0][1] = 1; + break; + case 2: + dirs[2][0] = -1; + dirs[3][0] = 1; + dirs[0][1] = -1; + dirs[1][1] = 1; + break; + case 3: + dirs[1][0] = -1; + dirs[0][0] = 1; + dirs[3][1] = -1; + dirs[2][1] = 1; + break; + } + } + _isFree(map, x, y, width, height) { + if (x < 1 || y < 1 || x >= width || y >= height) { + return false; + } + return map[x][y]; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/rogue.js + + + +/** + * Dungeon generator which uses the "orginal" Rogue dungeon generation algorithm. See http://kuoi.com/~kamikaze/GameDesign/art07_rogue_dungeon.php + * @author hyakugei + */ +class rogue_Rogue extends map_Map { + constructor(width, height, options) { + super(width, height); + this.map = []; + this.rooms = []; + this.connectedCells = []; + options = Object.assign({ + cellWidth: 3, + cellHeight: 3 // ie. as an array with min-max values for each direction.... + }, options); + /* + Set the room sizes according to the over-all width of the map, + and the cell sizes. + */ + if (!options.hasOwnProperty("roomWidth")) { + options["roomWidth"] = this._calculateRoomSize(this._width, options["cellWidth"]); + } + if (!options.hasOwnProperty("roomHeight")) { + options["roomHeight"] = this._calculateRoomSize(this._height, options["cellHeight"]); + } + this._options = options; + } + create(callback) { + this.map = this._fillMap(1); + this.rooms = []; + this.connectedCells = []; + this._initRooms(); + this._connectRooms(); + this._connectUnconnectedRooms(); + this._createRandomRoomConnections(); + this._createRooms(); + this._createCorridors(); + if (callback) { + for (let i = 0; i < this._width; i++) { + for (let j = 0; j < this._height; j++) { + callback(i, j, this.map[i][j]); + } + } + } + return this; + } + _calculateRoomSize(size, cell) { + let max = Math.floor((size / cell) * 0.8); + let min = Math.floor((size / cell) * 0.25); + if (min < 2) { + min = 2; + } + if (max < 2) { + max = 2; + } + return [min, max]; + } + _initRooms() { + // create rooms array. This is the "grid" list from the algo. + for (let i = 0; i < this._options.cellWidth; i++) { + this.rooms.push([]); + for (let j = 0; j < this._options.cellHeight; j++) { + this.rooms[i].push({ "x": 0, "y": 0, "width": 0, "height": 0, "connections": [], "cellx": i, "celly": j }); + } + } + } + _connectRooms() { + //pick random starting grid + let cgx = rng["a" /* default */].getUniformInt(0, this._options.cellWidth - 1); + let cgy = rng["a" /* default */].getUniformInt(0, this._options.cellHeight - 1); + let idx; + let ncgx; + let ncgy; + let found = false; + let room; + let otherRoom; + let dirToCheck; + // find unconnected neighbour cells + do { + //dirToCheck = [0, 1, 2, 3, 4, 5, 6, 7]; + dirToCheck = [0, 2, 4, 6]; + dirToCheck = rng["a" /* default */].shuffle(dirToCheck); + do { + found = false; + idx = dirToCheck.pop(); + ncgx = cgx + DIRS[8][idx][0]; + ncgy = cgy + DIRS[8][idx][1]; + if (ncgx < 0 || ncgx >= this._options.cellWidth) { + continue; + } + if (ncgy < 0 || ncgy >= this._options.cellHeight) { + continue; + } + room = this.rooms[cgx][cgy]; + if (room["connections"].length > 0) { + // as long as this room doesn't already coonect to me, we are ok with it. + if (room["connections"][0][0] == ncgx && room["connections"][0][1] == ncgy) { + break; + } + } + otherRoom = this.rooms[ncgx][ncgy]; + if (otherRoom["connections"].length == 0) { + otherRoom["connections"].push([cgx, cgy]); + this.connectedCells.push([ncgx, ncgy]); + cgx = ncgx; + cgy = ncgy; + found = true; + } + } while (dirToCheck.length > 0 && found == false); + } while (dirToCheck.length > 0); + } + _connectUnconnectedRooms() { + //While there are unconnected rooms, try to connect them to a random connected neighbor + //(if a room has no connected neighbors yet, just keep cycling, you'll fill out to it eventually). + let cw = this._options.cellWidth; + let ch = this._options.cellHeight; + this.connectedCells = rng["a" /* default */].shuffle(this.connectedCells); + let room; + let otherRoom; + let validRoom; + for (let i = 0; i < this._options.cellWidth; i++) { + for (let j = 0; j < this._options.cellHeight; j++) { + room = this.rooms[i][j]; + if (room["connections"].length == 0) { + let directions = [0, 2, 4, 6]; + directions = rng["a" /* default */].shuffle(directions); + validRoom = false; + do { + let dirIdx = directions.pop(); + let newI = i + DIRS[8][dirIdx][0]; + let newJ = j + DIRS[8][dirIdx][1]; + if (newI < 0 || newI >= cw || newJ < 0 || newJ >= ch) { + continue; + } + otherRoom = this.rooms[newI][newJ]; + validRoom = true; + if (otherRoom["connections"].length == 0) { + break; + } + for (let k = 0; k < otherRoom["connections"].length; k++) { + if (otherRoom["connections"][k][0] == i && otherRoom["connections"][k][1] == j) { + validRoom = false; + break; + } + } + if (validRoom) { + break; + } + } while (directions.length); + if (validRoom) { + room["connections"].push([otherRoom["cellx"], otherRoom["celly"]]); + } + else { + console.log("-- Unable to connect room."); + } + } + } + } + } + _createRandomRoomConnections() { + // Empty for now. + } + _createRooms() { + let w = this._width; + let h = this._height; + let cw = this._options.cellWidth; + let ch = this._options.cellHeight; + let cwp = Math.floor(this._width / cw); + let chp = Math.floor(this._height / ch); + let roomw; + let roomh; + let roomWidth = this._options["roomWidth"]; + let roomHeight = this._options["roomHeight"]; + let sx; + let sy; + let otherRoom; + for (let i = 0; i < cw; i++) { + for (let j = 0; j < ch; j++) { + sx = cwp * i; + sy = chp * j; + if (sx == 0) { + sx = 1; + } + if (sy == 0) { + sy = 1; + } + roomw = rng["a" /* default */].getUniformInt(roomWidth[0], roomWidth[1]); + roomh = rng["a" /* default */].getUniformInt(roomHeight[0], roomHeight[1]); + if (j > 0) { + otherRoom = this.rooms[i][j - 1]; + while (sy - (otherRoom["y"] + otherRoom["height"]) < 3) { + sy++; + } + } + if (i > 0) { + otherRoom = this.rooms[i - 1][j]; + while (sx - (otherRoom["x"] + otherRoom["width"]) < 3) { + sx++; + } + } + let sxOffset = Math.round(rng["a" /* default */].getUniformInt(0, cwp - roomw) / 2); + let syOffset = Math.round(rng["a" /* default */].getUniformInt(0, chp - roomh) / 2); + while (sx + sxOffset + roomw >= w) { + if (sxOffset) { + sxOffset--; + } + else { + roomw--; + } + } + while (sy + syOffset + roomh >= h) { + if (syOffset) { + syOffset--; + } + else { + roomh--; + } + } + sx = sx + sxOffset; + sy = sy + syOffset; + this.rooms[i][j]["x"] = sx; + this.rooms[i][j]["y"] = sy; + this.rooms[i][j]["width"] = roomw; + this.rooms[i][j]["height"] = roomh; + for (let ii = sx; ii < sx + roomw; ii++) { + for (let jj = sy; jj < sy + roomh; jj++) { + this.map[ii][jj] = 0; + } + } + } + } + } + _getWallPosition(aRoom, aDirection) { + let rx; + let ry; + let door; + if (aDirection == 1 || aDirection == 3) { + rx = rng["a" /* default */].getUniformInt(aRoom["x"] + 1, aRoom["x"] + aRoom["width"] - 2); + if (aDirection == 1) { + ry = aRoom["y"] - 2; + door = ry + 1; + } + else { + ry = aRoom["y"] + aRoom["height"] + 1; + door = ry - 1; + } + this.map[rx][door] = 0; // i'm not setting a specific 'door' tile value right now, just empty space. + } + else { + ry = rng["a" /* default */].getUniformInt(aRoom["y"] + 1, aRoom["y"] + aRoom["height"] - 2); + if (aDirection == 2) { + rx = aRoom["x"] + aRoom["width"] + 1; + door = rx - 1; + } + else { + rx = aRoom["x"] - 2; + door = rx + 1; + } + this.map[door][ry] = 0; // i'm not setting a specific 'door' tile value right now, just empty space. + } + return [rx, ry]; + } + _drawCorridor(startPosition, endPosition) { + let xOffset = endPosition[0] - startPosition[0]; + let yOffset = endPosition[1] - startPosition[1]; + let xpos = startPosition[0]; + let ypos = startPosition[1]; + let tempDist; + let xDir; + let yDir; + let move; // 2 element array, element 0 is the direction, element 1 is the total value to move. + let moves = []; // a list of 2 element arrays + let xAbs = Math.abs(xOffset); + let yAbs = Math.abs(yOffset); + let percent = rng["a" /* default */].getUniform(); // used to split the move at different places along the long axis + let firstHalf = percent; + let secondHalf = 1 - percent; + xDir = xOffset > 0 ? 2 : 6; + yDir = yOffset > 0 ? 4 : 0; + if (xAbs < yAbs) { + // move firstHalf of the y offset + tempDist = Math.ceil(yAbs * firstHalf); + moves.push([yDir, tempDist]); + // move all the x offset + moves.push([xDir, xAbs]); + // move sendHalf of the y offset + tempDist = Math.floor(yAbs * secondHalf); + moves.push([yDir, tempDist]); + } + else { + // move firstHalf of the x offset + tempDist = Math.ceil(xAbs * firstHalf); + moves.push([xDir, tempDist]); + // move all the y offset + moves.push([yDir, yAbs]); + // move secondHalf of the x offset. + tempDist = Math.floor(xAbs * secondHalf); + moves.push([xDir, tempDist]); + } + this.map[xpos][ypos] = 0; + while (moves.length > 0) { + move = moves.pop(); + while (move[1] > 0) { + xpos += DIRS[8][move[0]][0]; + ypos += DIRS[8][move[0]][1]; + this.map[xpos][ypos] = 0; + move[1] = move[1] - 1; + } + } + } + _createCorridors() { + // Draw Corridors between connected rooms + let cw = this._options.cellWidth; + let ch = this._options.cellHeight; + let room; + let connection; + let otherRoom; + let wall; + let otherWall; + for (let i = 0; i < cw; i++) { + for (let j = 0; j < ch; j++) { + room = this.rooms[i][j]; + for (let k = 0; k < room["connections"].length; k++) { + connection = room["connections"][k]; + otherRoom = this.rooms[connection[0]][connection[1]]; + // figure out what wall our corridor will start one. + // figure out what wall our corridor will end on. + if (otherRoom["cellx"] > room["cellx"]) { + wall = 2; + otherWall = 4; + } + else if (otherRoom["cellx"] < room["cellx"]) { + wall = 4; + otherWall = 2; + } + else if (otherRoom["celly"] > room["celly"]) { + wall = 3; + otherWall = 1; + } + else { + wall = 1; + otherWall = 3; + } + this._drawCorridor(this._getWallPosition(room, wall), this._getWallPosition(otherRoom, otherWall)); + } + } + } + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/map/index.js + + + + + + + + +/* harmony default export */ var lib_map = ({ Arena: arena_Arena, Uniform: uniform_Uniform, Cellular: cellular_Cellular, Digger: digger_Digger, EllerMaze: ellermaze_EllerMaze, DividedMaze: dividedmaze_DividedMaze, IceyMaze: iceymaze_IceyMaze, Rogue: rogue_Rogue }); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/noise/noise.js +/** + * Base noise generator + */ +class Noise { +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/noise/simplex.js + + + +const F2 = 0.5 * (Math.sqrt(3) - 1); +const G2 = (3 - Math.sqrt(3)) / 6; +/** + * A simple 2d implementation of simplex noise by Ondrej Zara + * + * Based on a speed-improved simplex noise algorithm for 2D, 3D and 4D in Java. + * Which is based on example code by Stefan Gustavson (stegu@itn.liu.se). + * With Optimisations by Peter Eastman (peastman@drizzle.stanford.edu). + * Better rank ordering method by Stefan Gustavson in 2012. + */ +class simplex_Simplex extends Noise { + /** + * @param gradients Random gradients + */ + constructor(gradients = 256) { + super(); + this._gradients = [ + [0, -1], + [1, -1], + [1, 0], + [1, 1], + [0, 1], + [-1, 1], + [-1, 0], + [-1, -1] + ]; + let permutations = []; + for (let i = 0; i < gradients; i++) { + permutations.push(i); + } + permutations = rng["a" /* default */].shuffle(permutations); + this._perms = []; + this._indexes = []; + for (let i = 0; i < 2 * gradients; i++) { + this._perms.push(permutations[i % gradients]); + this._indexes.push(this._perms[i] % this._gradients.length); + } + } + get(xin, yin) { + let perms = this._perms; + let indexes = this._indexes; + let count = perms.length / 2; + let n0 = 0, n1 = 0, n2 = 0, gi; // Noise contributions from the three corners + // Skew the input space to determine which simplex cell we're in + let s = (xin + yin) * F2; // Hairy factor for 2D + let i = Math.floor(xin + s); + let j = Math.floor(yin + s); + let t = (i + j) * G2; + let X0 = i - t; // Unskew the cell origin back to (x,y) space + let Y0 = j - t; + let x0 = xin - X0; // The x,y distances from the cell origin + let y0 = yin - Y0; + // For the 2D case, the simplex shape is an equilateral triangle. + // Determine which simplex we are in. + let i1, j1; // Offsets for second (middle) corner of simplex in (i,j) coords + if (x0 > y0) { + i1 = 1; + j1 = 0; + } + else { // lower triangle, XY order: (0,0)->(1,0)->(1,1) + i1 = 0; + j1 = 1; + } // upper triangle, YX order: (0,0)->(0,1)->(1,1) + // A step of (1,0) in (i,j) means a step of (1-c,-c) in (x,y), and + // a step of (0,1) in (i,j) means a step of (-c,1-c) in (x,y), where + // c = (3-sqrt(3))/6 + let x1 = x0 - i1 + G2; // Offsets for middle corner in (x,y) unskewed coords + let y1 = y0 - j1 + G2; + let x2 = x0 - 1 + 2 * G2; // Offsets for last corner in (x,y) unskewed coords + let y2 = y0 - 1 + 2 * G2; + // Work out the hashed gradient indices of the three simplex corners + let ii = Object(util["mod"])(i, count); + let jj = Object(util["mod"])(j, count); + // Calculate the contribution from the three corners + let t0 = 0.5 - x0 * x0 - y0 * y0; + if (t0 >= 0) { + t0 *= t0; + gi = indexes[ii + perms[jj]]; + let grad = this._gradients[gi]; + n0 = t0 * t0 * (grad[0] * x0 + grad[1] * y0); + } + let t1 = 0.5 - x1 * x1 - y1 * y1; + if (t1 >= 0) { + t1 *= t1; + gi = indexes[ii + i1 + perms[jj + j1]]; + let grad = this._gradients[gi]; + n1 = t1 * t1 * (grad[0] * x1 + grad[1] * y1); + } + let t2 = 0.5 - x2 * x2 - y2 * y2; + if (t2 >= 0) { + t2 *= t2; + gi = indexes[ii + 1 + perms[jj + 1]]; + let grad = this._gradients[gi]; + n2 = t2 * t2 * (grad[0] * x2 + grad[1] * y2); + } + // Add contributions from each corner to get the final noise value. + // The result is scaled to return values in the interval [-1,1]. + return 70 * (n0 + n1 + n2); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/noise/index.js + +/* harmony default export */ var noise = ({ Simplex: simplex_Simplex }); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/path/path.js + +/** + * @class Abstract pathfinder + * @param {int} toX Target X coord + * @param {int} toY Target Y coord + * @param {function} passableCallback Callback to determine map passability + * @param {object} [options] + * @param {int} [options.topology=8] + */ +class path_Path { + constructor(toX, toY, passableCallback, options = {}) { + this._toX = toX; + this._toY = toY; + this._passableCallback = passableCallback; + this._options = Object.assign({ + topology: 8 + }, options); + this._dirs = DIRS[this._options.topology]; + if (this._options.topology == 8) { /* reorder dirs for more aesthetic result (vertical/horizontal first) */ + this._dirs = [ + this._dirs[0], + this._dirs[2], + this._dirs[4], + this._dirs[6], + this._dirs[1], + this._dirs[3], + this._dirs[5], + this._dirs[7] + ]; + } + } + _getNeighbors(cx, cy) { + let result = []; + for (let i = 0; i < this._dirs.length; i++) { + let dir = this._dirs[i]; + let x = cx + dir[0]; + let y = cy + dir[1]; + if (!this._passableCallback(x, y)) { + continue; + } + result.push([x, y]); + } + return result; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/path/dijkstra.js + +/** + * @class Simplified Dijkstra's algorithm: all edges have a value of 1 + * @augments ROT.Path + * @see ROT.Path + */ +class dijkstra_Dijkstra extends path_Path { + constructor(toX, toY, passableCallback, options) { + super(toX, toY, passableCallback, options); + this._computed = {}; + this._todo = []; + this._add(toX, toY, null); + } + /** + * Compute a path from a given point + * @see ROT.Path#compute + */ + compute(fromX, fromY, callback) { + let key = fromX + "," + fromY; + if (!(key in this._computed)) { + this._compute(fromX, fromY); + } + if (!(key in this._computed)) { + return; + } + let item = this._computed[key]; + while (item) { + callback(item.x, item.y); + item = item.prev; + } + } + /** + * Compute a non-cached value + */ + _compute(fromX, fromY) { + while (this._todo.length) { + let item = this._todo.shift(); + if (item.x == fromX && item.y == fromY) { + return; + } + let neighbors = this._getNeighbors(item.x, item.y); + for (let i = 0; i < neighbors.length; i++) { + let neighbor = neighbors[i]; + let x = neighbor[0]; + let y = neighbor[1]; + let id = x + "," + y; + if (id in this._computed) { + continue; + } /* already done */ + this._add(x, y, item); + } + } + } + _add(x, y, prev) { + let obj = { + x: x, + y: y, + prev: prev + }; + this._computed[x + "," + y] = obj; + this._todo.push(obj); + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/path/astar.js + +/** + * @class Simplified A* algorithm: all edges have a value of 1 + * @augments ROT.Path + * @see ROT.Path + */ +class astar_AStar extends path_Path { + constructor(toX, toY, passableCallback, options = {}) { + super(toX, toY, passableCallback, options); + this._todo = []; + this._done = {}; + } + /** + * Compute a path from a given point + * @see ROT.Path#compute + */ + compute(fromX, fromY, callback) { + this._todo = []; + this._done = {}; + this._fromX = fromX; + this._fromY = fromY; + this._add(this._toX, this._toY, null); + while (this._todo.length) { + let item = this._todo.shift(); + let id = item.x + "," + item.y; + if (id in this._done) { + continue; + } + this._done[id] = item; + if (item.x == fromX && item.y == fromY) { + break; + } + let neighbors = this._getNeighbors(item.x, item.y); + for (let i = 0; i < neighbors.length; i++) { + let neighbor = neighbors[i]; + let x = neighbor[0]; + let y = neighbor[1]; + let id = x + "," + y; + if (id in this._done) { + continue; + } + this._add(x, y, item); + } + } + let item = this._done[fromX + "," + fromY]; + if (!item) { + return; + } + while (item) { + callback(item.x, item.y); + item = item.prev; + } + } + _add(x, y, prev) { + let h = this._distance(x, y); + let obj = { + x: x, + y: y, + prev: prev, + g: (prev ? prev.g + 1 : 0), + h: h + }; + /* insert into priority queue */ + let f = obj.g + obj.h; + for (let i = 0; i < this._todo.length; i++) { + let item = this._todo[i]; + let itemF = item.g + item.h; + if (f < itemF || (f == itemF && h < item.h)) { + this._todo.splice(i, 0, obj); + return; + } + } + this._todo.push(obj); + } + _distance(x, y) { + switch (this._options.topology) { + case 4: + return (Math.abs(x - this._fromX) + Math.abs(y - this._fromY)); + break; + case 6: + let dx = Math.abs(x - this._fromX); + let dy = Math.abs(y - this._fromY); + return dy + Math.max(0, (dx - dy) / 2); + break; + case 8: + return Math.max(Math.abs(x - this._fromX), Math.abs(y - this._fromY)); + break; + } + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/path/index.js + + +/* harmony default export */ var path = ({ Dijkstra: dijkstra_Dijkstra, AStar: astar_AStar }); + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/engine.js +/** + * @class Asynchronous main loop + * @param {ROT.Scheduler} scheduler + */ +class Engine { + constructor(scheduler) { + this._scheduler = scheduler; + this._lock = 1; + } + /** + * Start the main loop. When this call returns, the loop is locked. + */ + start() { return this.unlock(); } + /** + * Interrupt the engine by an asynchronous action + */ + lock() { + this._lock++; + return this; + } + /** + * Resume execution (paused by a previous lock) + */ + unlock() { + if (!this._lock) { + throw new Error("Cannot unlock unlocked engine"); + } + this._lock--; + while (!this._lock) { + let actor = this._scheduler.next(); + if (!actor) { + return this.lock(); + } /* no actors */ + let result = actor.act(); + if (result && result.then) { /* actor returned a "thenable", looks like a Promise */ + this.lock(); + result.then(this.unlock.bind(this)); + } + } + return this; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/lighting.js + +; +; +; +; +/** + * Lighting computation, based on a traditional FOV for multiple light sources and multiple passes. + */ +class lighting_Lighting { + constructor(reflectivityCallback, options = {}) { + this._reflectivityCallback = reflectivityCallback; + this._options = {}; + options = Object.assign({ + passes: 1, + emissionThreshold: 100, + range: 10 + }, options); + this._lights = {}; + this._reflectivityCache = {}; + this._fovCache = {}; + this.setOptions(options); + } + /** + * Adjust options at runtime + */ + setOptions(options) { + Object.assign(this._options, options); + if (options && options.range) { + this.reset(); + } + return this; + } + /** + * Set the used Field-Of-View algo + */ + setFOV(fov) { + this._fov = fov; + this._fovCache = {}; + return this; + } + /** + * Set (or remove) a light source + */ + setLight(x, y, color) { + let key = x + "," + y; + if (color) { + this._lights[key] = (typeof (color) == "string" ? lib_color["fromString"](color) : color); + } + else { + delete this._lights[key]; + } + return this; + } + /** + * Remove all light sources + */ + clearLights() { this._lights = {}; } + /** + * Reset the pre-computed topology values. Call whenever the underlying map changes its light-passability. + */ + reset() { + this._reflectivityCache = {}; + this._fovCache = {}; + return this; + } + /** + * Compute the lighting + */ + compute(lightingCallback) { + let doneCells = {}; + let emittingCells = {}; + let litCells = {}; + for (let key in this._lights) { /* prepare emitters for first pass */ + let light = this._lights[key]; + emittingCells[key] = [0, 0, 0]; + lib_color["add_"](emittingCells[key], light); + } + for (let i = 0; i < this._options.passes; i++) { /* main loop */ + this._emitLight(emittingCells, litCells, doneCells); + if (i + 1 == this._options.passes) { + continue; + } /* not for the last pass */ + emittingCells = this._computeEmitters(litCells, doneCells); + } + for (let litKey in litCells) { /* let the user know what and how is lit */ + let parts = litKey.split(","); + let x = parseInt(parts[0]); + let y = parseInt(parts[1]); + lightingCallback(x, y, litCells[litKey]); + } + return this; + } + /** + * Compute one iteration from all emitting cells + * @param emittingCells These emit light + * @param litCells Add projected light to these + * @param doneCells These already emitted, forbid them from further calculations + */ + _emitLight(emittingCells, litCells, doneCells) { + for (let key in emittingCells) { + let parts = key.split(","); + let x = parseInt(parts[0]); + let y = parseInt(parts[1]); + this._emitLightFromCell(x, y, emittingCells[key], litCells); + doneCells[key] = 1; + } + return this; + } + /** + * Prepare a list of emitters for next pass + */ + _computeEmitters(litCells, doneCells) { + let result = {}; + for (let key in litCells) { + if (key in doneCells) { + continue; + } /* already emitted */ + let color = litCells[key]; + let reflectivity; + if (key in this._reflectivityCache) { + reflectivity = this._reflectivityCache[key]; + } + else { + let parts = key.split(","); + let x = parseInt(parts[0]); + let y = parseInt(parts[1]); + reflectivity = this._reflectivityCallback(x, y); + this._reflectivityCache[key] = reflectivity; + } + if (reflectivity == 0) { + continue; + } /* will not reflect at all */ + /* compute emission color */ + let emission = [0, 0, 0]; + let intensity = 0; + for (let i = 0; i < 3; i++) { + let part = Math.round(color[i] * reflectivity); + emission[i] = part; + intensity += part; + } + if (intensity > this._options.emissionThreshold) { + result[key] = emission; + } + } + return result; + } + /** + * Compute one iteration from one cell + */ + _emitLightFromCell(x, y, color, litCells) { + let key = x + "," + y; + let fov; + if (key in this._fovCache) { + fov = this._fovCache[key]; + } + else { + fov = this._updateFOV(x, y); + } + for (let fovKey in fov) { + let formFactor = fov[fovKey]; + let result; + if (fovKey in litCells) { /* already lit */ + result = litCells[fovKey]; + } + else { /* newly lit */ + result = [0, 0, 0]; + litCells[fovKey] = result; + } + for (let i = 0; i < 3; i++) { + result[i] += Math.round(color[i] * formFactor); + } /* add light color */ + } + return this; + } + /** + * Compute FOV ("form factor") for a potential light source at [x,y] + */ + _updateFOV(x, y) { + let key1 = x + "," + y; + let cache = {}; + this._fovCache[key1] = cache; + let range = this._options.range; + function cb(x, y, r, vis) { + let key2 = x + "," + y; + let formFactor = vis * (1 - r / range); + if (formFactor == 0) { + return; + } + cache[key2] = formFactor; + } + ; + this._fov.compute(x, y, range, cb.bind(this)); + return cache; + } +} + +// CONCATENATED MODULE: ./node_modules/rot-js/lib/index.js +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Util", function() { return Util; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Color", function() { return Color; }); +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Text", function() { return Text; }); +/* concated harmony reexport RNG */__webpack_require__.d(__webpack_exports__, "RNG", function() { return rng["a" /* default */]; }); +/* concated harmony reexport Display */__webpack_require__.d(__webpack_exports__, "Display", function() { return display_Display; }); +/* concated harmony reexport StringGenerator */__webpack_require__.d(__webpack_exports__, "StringGenerator", function() { return stringgenerator_StringGenerator; }); +/* concated harmony reexport EventQueue */__webpack_require__.d(__webpack_exports__, "EventQueue", function() { return EventQueue; }); +/* concated harmony reexport Scheduler */__webpack_require__.d(__webpack_exports__, "Scheduler", function() { return scheduler; }); +/* concated harmony reexport FOV */__webpack_require__.d(__webpack_exports__, "FOV", function() { return fov; }); +/* concated harmony reexport Map */__webpack_require__.d(__webpack_exports__, "Map", function() { return lib_map; }); +/* concated harmony reexport Noise */__webpack_require__.d(__webpack_exports__, "Noise", function() { return noise; }); +/* concated harmony reexport Path */__webpack_require__.d(__webpack_exports__, "Path", function() { return path; }); +/* concated harmony reexport Engine */__webpack_require__.d(__webpack_exports__, "Engine", function() { return Engine; }); +/* concated harmony reexport Lighting */__webpack_require__.d(__webpack_exports__, "Lighting", function() { return lighting_Lighting; }); +/* concated harmony reexport DEFAULT_WIDTH */__webpack_require__.d(__webpack_exports__, "DEFAULT_WIDTH", function() { return DEFAULT_WIDTH; }); +/* concated harmony reexport DEFAULT_HEIGHT */__webpack_require__.d(__webpack_exports__, "DEFAULT_HEIGHT", function() { return DEFAULT_HEIGHT; }); +/* concated harmony reexport DIRS */__webpack_require__.d(__webpack_exports__, "DIRS", function() { return DIRS; }); +/* concated harmony reexport KEYS */__webpack_require__.d(__webpack_exports__, "KEYS", function() { return KEYS; }); + + + + + + + + + + + + + +const Util = util; + +const Color = lib_color; + +const Text = text_namespaceObject; + + +/***/ }), +/* 4 */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "a", function() { return Backend; }); +/** + * @class Abstract display backend module + * @private + */ +class Backend { + getContainer() { return null; } + setOptions(options) { this._options = options; } +} + + +/***/ }), +/* 5 */ +/***/ (function(module, exports, __webpack_require__) { + +const ROT = __webpack_require__(3); +// See: http://ondras.github.io/rot.js/manual/#rng + +function setSeed(seed) { + if (typeof seed !== 'number') { + seed = makeSeed(); + } + ROT.RNG.setSeed(seed); +} + +function makeSeed() { + return Math.round(Math.random() * 10000); +} + +function roll(n, seed) { + if (typeof seed === 'number') { + setSeed(seed); + } + if (typeof n === 'number') { + return Math.floor(ROT.RNG.getUniform() * n); + } + if (typeof n === 'string') { + return rollDice(n, seed); + } +} + +function rollDice(str, seed) { // str like "1d4", "2d6", "3d8", "1d100" + // TODO: handle "+", "-", other? + const d = str.split('d'); + if (d.length !== 2) { + const n = Number(str); + // console.warn('Unexpected value:', str, '. Not valid dice notation. Using:', n); + return roll(n, seed); + } + const numberOfDice = d[0]; + const numberOfSides = d[1]; + let sum = 0; + for(let i = 0; i < numberOfDice; i++) { + sum += roll(numberOfSides, seed); + } + return sum; +} + +function getWeightedValue(obj, seed) { + if (typeof seed === 'number') { + setSeed(seed); + } + return ROT.RNG.getWeightedValue(obj); +} + +function shuffle(arr) { + return ROT.RNG.shuffle(arr); +} + +function pickOne(arr) { + return ROT.RNG.getItem(arr); +} + +module.exports = { + setSeed, + makeSeed, + roll, + getWeightedValue, + shuffle, + pickOne +}; + + +/***/ }), +/* 6 */ +/***/ (function(module, exports, __webpack_require__) { + +const Inventory = __webpack_require__(14); +const { DIRS_8 } = __webpack_require__(15); + +class Item { + constructor(options = {}) { + this.type = options.type || null; + this.name = options.name || 'nothing'; + this.x = options.x || 0; + this.y = options.y || 0; + this.character = options.character || '^'; + this.surrounding = options.surrounding || []; + this.color = options.color || '#05f'; + this.background = options.background || null; + this.inventory = new Inventory({ + size: options.inventorySize || 0 + }); + this.isWeapon = Boolean(options.weapon); + this.damage = parseInt(options.weapon, 10) || 0; + this.isDefense = Boolean(options.defense); + this.defense = parseInt(options.defense, 10) || 0; + this.illumination = options.illumination || 0; + this.portable = (typeof options.portable === 'boolean') ? options.portable : true; + this.containedIn = null; + this.actions = { ...options.on, ...options.actions }; // TODO: do we need the "on" alias? + if (options.use) { + this.actions.use = options.use; + } + this.states = options.states || {}; + this.teleport = null; // can this item move the character to another level, cell + } + + hasAction(verb) { + return Boolean(this.actions[verb]); + } + + action(actionName, who) { + const action = this.actions[actionName]; + let actionOutcome = {}; + if (typeof action === 'function') { + actionOutcome = action(this, who); + } else if (typeof action === 'object' && action !== null) { + actionOutcome = this.runAction(action, who); + } else { + console.warn('No action', actionName, 'for item', this); + } + return actionOutcome; + } + + runAction(action = {}, who) { // TODO: move to game/level? + let message = ''; + if (!this.requirementMet(action, who)) { + message = (action.missingMessage) ? action.missingMessage : `Some requirement is not met to use the ${this.name}`; + return { message }; + } + this.removeRequirements(action); + message = message + ((action.message) ? action.message : ''); + const effects = action.effects; + return { message, effects }; + } + + removeRequirements(action = {}) { + if (!action.requires) { return; } + action.requires.forEach((requirement) => { + const typeKey = requirement.item; + if (typeKey) { + this.inventory.removeType(typeKey); + } + }); + } + + requirementMet(action = {}, who) { + if (!action.requires) { + return true; + } + let met = 0; + action.requires.forEach((requirement) => { + if (requirement.item && this.inventory.containsType(requirement.item)) { + met += 1; + } + }); + return met === action.requires.length; + } + + draw(display, lighting = {}, inView = false) { + if (this.containedIn || !inView) { // Not visible if in a container + return false; + } + display.draw(this.x, this.y, this.character, this.color, this.background); + if (this.surrounding.length) { + this.surrounding.forEach((char, i) => { + let { x, y } = DIRS_8[i]; + display.draw(this.x + x, this.y + y, char, this.color, this.background); + }); + } + return true; + } + + addItem(item) { // mutates the item if successful + if (this.inventory.isFull()) { + return false; + } + const isAdded = this.inventory.add(item); + if (!isAdded) { + return false; + } + item.containedIn = this; + return true; + } + + //---- Inventory + + getContents(n) { + return this.inventory.get(n); + } + + hasContents() { + return this.inventory.hasContents(); + } + + contains(itemName) { + return this.inventory.contains(itemName); + } + + hasSpace() { + return this.inventory.hasSpace(); + } + + addToInventory(item) { + return this.inventory.add(item); + } + + //---- Sets + + setTeleport(options = {}) { + const { levelIndex, x, y, verb } = options; + this.teleport = { levelIndex, x, y }; + this.actions[verb] = 'teleport'; + } +} + +module.exports = Item; + + +/***/ }), +/* 7 */ +/***/ (function(module, exports) { + +function getDistance(x1, y1, x2, y2) { + return Math.sqrt( + Math.pow((x1 - x2), 2) + + Math.pow((y1 - y2), 2) + ); +} + +module.exports = { + getDistance +}; + + +/***/ }), +/* 8 */ +/***/ (function(module, exports, __webpack_require__) { + +const ROT = __webpack_require__(3); +const Inventory = __webpack_require__(14); +const geometer = __webpack_require__(7); +const random = __webpack_require__(5); + +class ActionVerbType { + static get MOVE () { + return 'move'; + } + + static get WAIT () { + return 'wait'; + } +} + +class FactionType { + + static get HUMAN () { + return 'human'; + } + + static get MONSTER () { + return 'monster'; + } + + static get NEUTRAL () { + return 'neutral'; + } +} + + +class Actor { + constructor(options = {}) { + this.type = options.type; + this.name = options.name || null; + this.faction = options.faction || FactionType.MONSTER; + this.isHero = Boolean(options.isHero); + this.x = options.x || 0; + this.y = options.y || 0; + this.character = options.character || 'M'; + this.originalCharacter = this.character; + this.color = options.color || '#df2'; + this.originalColor = this.color; + this.bloodColor = '#611'; + // this.game = options.game || console.error('must tie actor to game'); + this.inventory = new Inventory({ + size: options.inventorySize || 10 + }); + this.passable = false; + this.actionQueue = []; + this.maxMovement = this.isHero ? 1.42 : 1; + this.sightRange = (typeof options.sightRange === 'number') ? options.sightRange : 6; + this.target = null; + this.aggro = options.aggro || 0; // Level will set this to 100 for monsters + // stats + this.hp = (options.hp || typeof options.hp === 'number') ? parseInt(options.hp, 10) : 2; + this.hpMax = this.hp; + this.ap = options.ap || 0; // Attack/Arms + this.apMax = this.ap; + this.bp = options.bp || 0; // Balance + this.bpMax = this.bp; + this.ep = options.ep || 0; // Endurance + this.epMax = this.ep; + this.fp = options.fp || 0; // Focus + this.fpMax = this.fp; + this.wp = options.wp || 0; // Will(power) + this.wpMax = this.wp; + this.mp = options.mp || 0; // Mana + this.mpMax = this.mp; + // advancement + this.xp = options.xp || 0; + this.score = options.score || 0; + // abilities + this.maxAbilities = 9; + this.abilities = {}; + this.abilityList = []; + // temporary + this.initiativeBoost = 0; + + // all actors can carry currency + // everything is boiled down to copper pieces + // 100 copper = 1 silver + // 100 silver = 1 gold + this.currency = options.currency || 0; + } + + draw(display, lighting = {}, inView = false) { + if (!inView) { + return false; + } + // TODO: adjust colors based on lighting and inView + display.draw(this.x, this.y, this.character, this.color); + return true; + } + + queueAction(verb, params = {}) { + const actionParams = { ...params, verb }; + this.actionQueue.push(actionParams); + } + + clearQueue() { + this.actionQueue.length = 0; + } + + planAction(level, hero) { + if (this.isHero) { return; } + if (this.dead()) { + this.clearQueue(); + return; + } + const distanceToHero = geometer.getDistance(this.x, this.y, hero.x, hero.y); + const dangerouslyHurt = (this.hp <= 1); + this.act = function () { + console.log(`${this.name} acts`); + // if (g.getActiveLevel() !== level) { return; } + }; + if (this.aggro && distanceToHero <= this.getMaxSenseRange() && !hero.dead() && !dangerouslyHurt) { + const map = level.getMap(); + this.clearQueue(); + this.setTarget(hero); + this.setPathToTarget(map); + if (this.atEndOfPath()) { + this.queueAction('attack', { target: hero }); + } else if (this.actionQueue.length === 1) { + this.clearQueue(); + this.queueAction('attack', { target: hero }); + } + } else { + if (this.atEndOfPath()) { + this.setWanderPath(level); + } + } + // console.log(`${this.name} plans`, this.actionQueue); + } + + doAction() { + if (this.dead()) { return { verb: 'rot' }; } + const waitAction = { verb: ActionVerbType.WAIT }; + if (this.actionQueue.length === 0) { return waitAction; } + let action = this.actionQueue.shift(); + const moveAlreadyThere = (action.verb === ActionVerbType.MOVE && action.x === this.x && action.y === this.y); + const moveTooFar = (action.verb === ActionVerbType.MOVE && this.getDistanceToNextMove(action) > this.maxMovement); + // console.log(this.name, this.x, this.y, action.verb, action.x, action.y, this.getDistanceToNextMove(), this.maxMovement, moveTooFar, 'q', this.actionQueue.length); + if (moveAlreadyThere) { + return this.doAction(); + } + if (moveTooFar) { + action = this.doAction(); + } + if (!action) { + return waitAction; + } + return action; + } + + attack(who) { + console.log(`${this.name} attacks`, who); + // TODO + } + + setWanderPath(level) { + const map = level.getMap(); + const { x, y } = level.findRandomFreeCell(); + this.setPathTo(map, x, y); + } + + atEndOfPath() { + const nextAction = this.getNextAction(); + if (!nextAction) { return true; } + return (nextAction.verb === ActionVerbType.MOVE) ? false : true; + } + + wait() { + this.healPools(); + } + + //---- Movement + + move(x, y) { + this.x += parseInt(x, 10); + this.y += parseInt(y, 10); + } + + moveTo(x, y) { + this.setCoordinates(x, y); + } + + //---- Combat + + attackDamage(opponent) { + return 1; + } + + wound(n) { + return this.heal(n * -1); + } + + heal(n) { + const originalHp = this.hp; + this.hp += parseInt(n, 10); + this.hp = Math.min(this.hp, this.hpMax); + this.checkDeath(); + return this.hp - originalHp; + } + + dead() { + return (this.hp <= 0); + } + + checkDeath() { + if (this.dead()) { + this.character = 'X'; + this.color = this.bloodColor; + this.passable = true; + } + } + + //---- Healing + + healPools() { + this.healPool(this.getRandomPoolKey()); + } + + healPool(poolKey, amount = 1) { + const a = this.getAbilityReadiedAmounts(); + const max = this[poolKey + 'Max']; + if (a[poolKey] + this[poolKey] + amount <= max) { + this[poolKey] += amount; + } else { + if (this.isHero) { + console.log('No space to heal', poolKey, this); + } + } + } + + damagePool(poolKey, amount = 1) { + this[poolKey] -= amount; + this[poolKey] = Math.max(0, this[poolKey]); + } + + //---- Abilities + + hasAbility(abilityKey) { + return Boolean(this.abilities[abilityKey]); + } + + addAbility(abilityKey, abilityData) { + if (this.abilityList.length >= this.maxAbilities) { + return false; + } + if (this.hasAbility(abilityKey)) { + console.warn('Cannot add ability twice - would override'); + return; + } + // TODO: move to Activity class? + const ability = JSON.parse(JSON.stringify(abilityData)); + this.abilities[abilityKey] = ability; + ability.isReadied = false; + ability.key = abilityKey; + this.abilityList.push(abilityKey); + return ability; + } + + getAbilityByIndex(i) { + const key = this.abilityList[i]; + return this.abilities[key]; + } + + getAbilityReadiedAmounts() { + const a = { hp: 0, ap: 0, bp: 0, ep: 0, fp: 0, wp: 0 }; + this.abilityList.forEach((abilityKey) => { + const ability = this.abilities[abilityKey]; + Actor.loopOverAbilityCosts(ability, (costKey, val) => { + if (ability.isReadied) { + a[costKey] += val; + } + }); + }); + return a; + } + + canReadyAbility(ability) { + if (ability.isReadied) { return false; } + let canReady = true; + Actor.loopOverAbilityCosts(ability, (costKey, val) => { + const poolAmount = this[costKey]; + if (val > poolAmount) { + canReady = false; + } + }); + return canReady; + } + + readyAbilityByIndex(i) { + const ability = this.getAbilityByIndex(i); + if (!ability) { return false; } + if (!this.canReadyAbility(ability)) { return false; } + Actor.loopOverAbilityCosts(ability, (costKey, val) => { + this[costKey] -= val; + }); + ability.isReadied = true; + return ability; + } + + activateAbilities(eventName) { + const triggeredAbilities = this.getTriggeredAbilities(eventName); + let effects = []; + triggeredAbilities.forEach((ability) => { + ability.isReadied = false; + effects = effects.concat(ability.effects); + }); + return effects; + } + + getTriggeredAbilities(eventName) { + const triggeredAbilities = []; + this.abilityList.forEach((abilityKey) => { + const ability = this.abilities[abilityKey]; + if (ability.isReadied && ability.activateOn === eventName) { + triggeredAbilities.push(ability); + } + }); + return triggeredAbilities; + } + + static loopOverAbilityCosts(ability, fn) { + const costs = Object.keys(ability.readyCost); + costs.forEach((key) => { + fn(key, parseInt(ability.readyCost[key], 10)); + }); + } + + static getAbilityEffectsString(ability) { + let arr = []; + ability.effects.forEach((effect) => { + const words = (typeof effect === 'string') ? [effect] : Object.keys(effect); + arr = arr.concat(words); + }); + return arr.join(', '); + } + + static getAbilityDescriptionHtml(ability) { + let ready = 'Ready with'; + Actor.loopOverAbilityCosts(ability, (costKey, val) => { + ready += ' ' + val + ' ' + costKey.toUpperCase(); + }); + const effects = Actor.getAbilityEffectsString(ability); + return `
${ability.description}
+
${ready}
+
Activates on: ${ability.activateOn}
+
Causes: ${effects}
`; + } + + //---- Experience + + gainRandomPoolMax() { + const key = this.getRandomPoolKey() + 'Max'; + this[key] += 1; + } + + gainRandomAbility(abilitiesData) { + const abilityKeys = Object.keys(abilitiesData); + let abilityKey = random.pickOne(abilityKeys); + let attempts = 100; + while (this.hasAbility(abilityKey) && attempts--) { + abilityKey = random.pickOne(abilityKeys); + } + this.addAbility(abilityKey, abilitiesData[abilityKey]); + } + + //---- Gets + + getRandomPoolKey() { + return random.pickOne(['ap', 'bp', 'ep', 'wp']); + } + + getMaxSenseRange() { + return this.sightRange; + } + + getNextAction() { + return this.actionQueue[0]; + } + + getDistanceToNextMove(nextAction) { + if (!nextAction) { nextAction = this.getNextAction(); } + if (!nextAction) { return 0; } + const { x, y } = nextAction; + if (x !== undefined && y !== undefined) { + return geometer.getDistance(x, y, this.x, this.y); + } + return null; // ? + } + + getArmorDefense() { + if (!this.isHero) { + return 1; // TODO: change this so there is some kind of natural defense for monsters + } + let highestDefense = 0; + this.inventory.loopOverContents((item) => { + if (item.defense > highestDefense) { + highestDefense = item.defense; + } + }); + return highestDefense; + } + + getWeaponDamage() { + if (!this.isHero) { + return 1; // TODO: change this so there is some kind of natural damage for monsters + } + let highestDamage = 0; + this.inventory.loopOverContents((item) => { + if (item.damage > highestDamage) { + highestDamage = item.damage; + } + }); + return highestDamage; + } + + //---- Sets + + setCoordinates(x, y) { + this.x = parseInt(x, 10); + this.y = parseInt(y, 10); + } + + setPathTo(map, x = 0, y = 0) { + const passableCallback = function(x, y) { + return map.getCellPassability(x, y); + }; + const astar = new ROT.Path.AStar(x, y, passableCallback, { topology: 4 }); + const path = this.actionQueue; + const pathCallback = function(x, y) { + path.push({ x, y, verb: ActionVerbType.MOVE }); + }; + if (path[0] && path[0].x === this.x && path[0].y === this.y) { + console.alert('removing first'); + path.shift(); + } + astar.compute(this.x, this.y, pathCallback); + return true; + } + + setPathToTarget(map) { + return this.setPathTo(map, this.target.x, this.target.y); + } + + setTarget(target) { + if (typeof target.x !== 'number' || typeof target.y !== 'number') { + console.warn('Cannot set target to something without x,y'); + return; + } + this.target = target; + } +} + +module.exports = Actor; + + +/***/ }), +/* 9 */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +/* WEBPACK VAR INJECTION */(function(process) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "a", function() { return Term; }); +/* harmony import */ var _backend_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(4); +/* harmony import */ var _color_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(2); + + +function clearToAnsi(bg) { + return `\x1b[0;48;5;${termcolor(bg)}m\x1b[2J`; +} +function colorToAnsi(fg, bg) { + return `\x1b[0;38;5;${termcolor(fg)};48;5;${termcolor(bg)}m`; +} +function positionToAnsi(x, y) { + return `\x1b[${y + 1};${x + 1}H`; +} +function termcolor(color) { + const SRC_COLORS = 256.0; + const DST_COLORS = 6.0; + const COLOR_RATIO = DST_COLORS / SRC_COLORS; + let rgb = _color_js__WEBPACK_IMPORTED_MODULE_1__["fromString"](color); + let r = Math.floor(rgb[0] * COLOR_RATIO); + let g = Math.floor(rgb[1] * COLOR_RATIO); + let b = Math.floor(rgb[2] * COLOR_RATIO); + return r * 36 + g * 6 + b * 1 + 16; +} +class Term extends _backend_js__WEBPACK_IMPORTED_MODULE_0__[/* default */ "a"] { + constructor() { + super(); + this._offset = [0, 0]; + this._cursor = [-1, -1]; + this._lastColor = ""; + } + schedule(cb) { setTimeout(cb, 1000 / 60); } + setOptions(options) { + super.setOptions(options); + let size = [options.width, options.height]; + let avail = this.computeSize(); + this._offset = avail.map((val, index) => Math.floor((val - size[index]) / 2)); + } + clear() { + process.stdout.write(clearToAnsi(this._options.bg)); + } + draw(data, clearBefore) { + // determine where to draw what with what colors + let [x, y, ch, fg, bg] = data; + // determine if we need to move the terminal cursor + let dx = this._offset[0] + x; + let dy = this._offset[1] + y; + let size = this.computeSize(); + if (dx < 0 || dx >= size[0]) { + return; + } + if (dy < 0 || dy >= size[1]) { + return; + } + if (dx !== this._cursor[0] || dy !== this._cursor[1]) { + process.stdout.write(positionToAnsi(dx, dy)); + this._cursor[0] = dx; + this._cursor[1] = dy; + } + // terminals automatically clear, but if we're clearing when we're + // not otherwise provided with a character, just use a space instead + if (clearBefore) { + if (!ch) { + ch = " "; + } + } + // if we're not clearing and not provided with a character, do nothing + if (!ch) { + return; + } + // determine if we need to change colors + let newColor = colorToAnsi(fg, bg); + if (newColor !== this._lastColor) { + process.stdout.write(newColor); + this._lastColor = newColor; + } + // write the provided symbol to the display + let chars = [].concat(ch); + process.stdout.write(chars[0]); + // update our position, given that we wrote a character + this._cursor[0]++; + if (this._cursor[0] >= size[0]) { + this._cursor[0] = 0; + this._cursor[1]++; + } + } + computeFontSize() { throw new Error("Terminal backend has no notion of font size"); } + eventToPosition(x, y) { return [x, y]; } + computeSize() { return [process.stdout.columns, process.stdout.rows]; } +} + +/* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(18))) + +/***/ }), +/* 10 */ +/***/ (function(module, exports, __webpack_require__) { + +const FontFaceObserver = __webpack_require__(20); + +function domReady() { + return new Promise((resolve, reject) => { + if (document.readyState === "complete" || document.readyState === "loaded") { + resolve(); + } else { + document.addEventListener("DOMContentLoaded", () => { + resolve(); + }); + } + }); +} + +function ready(fn, fonts = []) { + if (fonts.length > 0) { + // TODO: allow multiple fonts ~ https://github.com/bramstein/fontfaceobserver + const font = new FontFaceObserver(fonts[0]); + font.load() + .then(() => { domReady().then(fn); }) + .catch(() => { console.warn('error loading font'); domReady().then(fn); }); + return; + } + domReady().then(fn); +} + +module.exports = ready; + + +/***/ }), +/* 11 */ +/***/ (function(module, exports, __webpack_require__) { + +const ROT = __webpack_require__(3); + +class Display { + constructor(options = {}) { + options = { width: 60, height: 30, ...options }; + this.width = null; + this.height = null; + this.center = {}; + this.setDimensions(options.width, options.height); + + this.id = options.id || 'display'; + this.rotDisplay = new ROT.Display(options); // , layout:"term"}); + this.displayContainer = null; + this.elt = null + this.cameraTarget = null; + this.setupElements(); + } + + setDimensions(x, y) { + this.width = x; + this.height = y; + this.center.x = Math.round(x/2); + this.center.y = Math.round(y/2); + } + + setupElements() { + this.displayContainer = document.getElementById(this.id); + this.elt = this.rotDisplay.getContainer(); // canvas + this.appendToElement(this.elt); + } + + appendToElement(elt) { + this.displayContainer.appendChild(elt); + } + + setCameraTarget(cameraTarget) { + if (!cameraTarget) { + console.warn("No target", cameraTarget); + return false; + } + if (typeof cameraTarget.x !== 'number' || typeof cameraTarget.y !== 'number') { + console.warn("Couldn't target", cameraTarget); + return false; + } + this.cameraTarget = cameraTarget; + return true; + } + + clear() { + this.rotDisplay.clear(); + } + + draw(x, y, character, fgColor, bgColor) { + if (this.cameraTarget) { + x += (this.center.x - this.cameraTarget.x); + y += (this.center.y - this.cameraTarget.y); + } + return this.rotDisplay.draw(x, y, character, fgColor, bgColor); + } + + drawLevel(game, level, hero) { + level.draw(this); + if (!hero) { return; } + hero.draw(this); + this.drawInterface(game, hero); + } + + drawHero(hero) { + if (!hero) { return; } + hero.draw(this); + } + + drawDamage(isDamaged = false, options = {}) { + // Override this + } + + drawInterface(game = {}, hero = {}, options = {}) { + // Override this + } + + static getPoolSquares(value, max, used) { + const maxLeft = max - value - used; + let str = ''; + let i; + for(i = 0; i < value; i++) { str += '■'; } + for(i = 0; i < used; i++) { str += '▣'; } + for(i = 0; i < maxLeft; i++) { str += '▢'; } + return str; + } + +} + +module.exports = Display; + +/***/ }), +/* 12 */ +/***/ (function(module, exports, __webpack_require__) { + +const Map = __webpack_require__(13); +const Actor = __webpack_require__(8); +const Item = __webpack_require__(6); +const Prop = __webpack_require__(16); +const geometer = __webpack_require__(7); +const random = __webpack_require__(5); +const { DIRS_4, DIRS_8_DIAGNOLS } = __webpack_require__(15); + +class Level { + constructor(options = {}, refData = {}) { + this.seed = options.seed || 1; + this.name = options.name || 'Unknown level'; + this.description = options.description || null; + this.levelIndex = options.levelIndex || 0; + this.color = options.color || '#777'; + this.background = options.background || '#222'; + const mapOptions = { + color: this.color, + background: this.background, + ...options.map, + seed: this.seed, + generators: options.generators || {} + }; + this.map = new Map(mapOptions); + this.actors = []; + this.items = []; + this.props = []; + this.actors = this.generateActors(options, refData); + this.items = this.generateItems(options, refData.items); + this.props = this.generateProps(options, refData.props); + this.eye = { x: 0, y: 0, sightRange: 7 }; + this.customEffects = { ...options.customEffects }; + } + + draw(display) { + display.clear(); + this.drawMap(display); + this.drawProps(display); + this.drawItems(display); + this.drawActors(display); + } + + drawMap(display) { + this.map.forEachCell((cell, x, y) => { + const inView = this.isInView(x, y); + // TODO: improve this + const fg = cell.getForegroundColor(inView); + const bg = cell.getBackgroundColor(inView); + display.draw(x, y, cell.character, fg, bg); + }); + } + + drawProps(display) { + this.props.forEach((prop) => { + const lighting = this.map.getLightingAt(this.eye.x, this.eye.y); + const inView = this.isInView(prop.x, prop.y); + prop.draw(display, lighting, inView); + }); + } + + drawItems(display) { + this.items.forEach((item) => { + const lighting = this.map.getLightingAt(this.eye.x, this.eye.y); + const inView = this.isInView(item.x, item.y); + item.draw(display, lighting, inView); + }); + } + + drawActors(display) { + // Draw dead first, then non-dead + this.actors.forEach((actor) => { + if (actor.dead()) { + this.drawActor(display, actor); + } + }); + this.actors.forEach((actor) => { + if (!actor.dead()) { + this.drawActor(display, actor); + } + }); + } + + drawActor(display, actor) { + const lighting = this.map.getLightingAt(this.eye.x, this.eye.y); + const inView = this.isInView(actor.x, actor.y); + actor.draw(display, lighting, inView); + } + + isInView(x, y) { // TODO: optimize + const r = geometer.getDistance(this.eye.x, this.eye.y, x, y); // TODO: allow more complicated POV + return (r <= this.eye.sightRange); + } + + addItem(item) { + this.items.push(item); + } + + removeItem(item) { + return this.removeThing('items', item); + } + + addActor(actor) { + this.actors.push(actor); + } + + removeActor(actor) { + return this.removeThing('actors', actor); + } + + removeThing(property, thing) { + const i = this[property].findIndex((a) => { return a === thing; }); + if (i <= -1) { + console.warn('nothing found in', property); + return; + } + const arr = this[property].splice(i, 1); + return arr[0]; + } + + findItem(x, y) { + const foundThings = this.findItems(x, y); + return (foundThings.length) ? foundThings[0] : null; + } + findItems(x, y) { + return this.items.filter((item) => { + return item.x === x && item.y === y && !item.containedIn; + }); + } + + findProp(x, y) { + const foundThings = this.findProps(x, y); + return (foundThings.length) ? foundThings[0] : null; + } + findProps(x, y) { + return this.props.filter((prop) => { return prop.x === x && prop.y === y; }); + } + findPropsByType(type) { + return this.props.filter((prop) => { return prop.type === type; }); + } + + findActor(x, y) { + const foundThings = this.findActors(x, y); + return (foundThings.length) ? foundThings[0] : null; + } + findActors(x, y) { + return this.actors.filter((actor) => { + return actor.x === x && actor.y === y && !actor.dead(); + }); + } + + findThingInView(what) { // 'actors', 'items', 'props' + return this[what].filter((a) => this.isInView(a.x, a.y)); + } + + findActorsInView(excludeHero) { + return this.actors.filter((a) => { + const inView = this.isInView(a.x, a.y); + if (excludeHero) { + return inView && !a.isHero; + } + return inView; + }); + } + + findEverythingInView(options = {}) { + return this.findActorsInView(options.excludeHero) + .concat(this.findThingInView('items')) + .concat(this.findThingInView('props')); + } + + findThings(x, y) { + const props = this.findProps(x, y); + const items = this.findItems(x, y); + const allThings = props.concat(items); + return allThings; + } + + findThingsCardinal(x, y) { + const props = this.findThingsByDirections('props', DIRS_4, x, y); + const items = this.findThingsByDirections('items', DIRS_4, x, y); + const allThings = props.concat(items); + return allThings; + } + + findThingsDiagnol(x, y) { + const props = this.findThingsByDirections('props', DIRS_8_DIAGNOLS, x, y); + const items = this.findThingsByDirections('items', DIRS_8_DIAGNOLS, x, y); + const allThings = props.concat(items); + return allThings; + } + + findThingsByDirections(thingName, dirs, x, y) { + const coords = []; + dirs.forEach((dir) => { coords.push({ x: x + dir.x, y: y + dir.y }); }); + return this[thingName].filter((thing) => { + const matches = coords.filter((xy) => { + return xy.x === thing.x && xy.y === thing.y && !thing.containedIn + }); + return matches.length > 0; + }); + } + + findThingSmart(x, y, perferredProperty) { + let things = this.findThings(x, y); + // console.log('find smart - on spot', things); + if (!things.length) { + things = this.findThingsCardinal(x, y); + // console.log('find smart - cardinal', things); + if (!things.length) { + things = this.findThingsDiagnol(x, y); + // console.log('find smart - diagnols', things); + } + } + if (perferredProperty) { + things.sort((a, b) => { + // console.log(a, b, a[perferredProperty]); + return b[perferredProperty]; + }); + // console.log("sorted", things); + } + return things[0]; + } + + findRandomFreeCell(seed, clearing, retries = 50) { + let cell = this.map.getRandomFreeCell(); + if (!retries) { + return cell; + } + const tryAgain = () => { + return this.findRandomFreeCell(seed, clearing, (retries - 1)); + }; + if (this.findActors(cell.x, cell.y).length > 0) { + return tryAgain(); + } + if (this.findThings(cell.x, cell.y).length > 0) { + return tryAgain(); + } + if (this.findMapClearing(cell.x, cell.y) >= clearing) { + return tryAgain(); + } + return cell; + } + + findMapClearing(x, y) { + // TODO: loop + return 0; + } + + discoverCircle(x, y, radius) { + return this.map.discoverCircle(x, y, radius); + } + + // Actions + + useThing(actor, actionName, thing) { + const outcome = thing.action(actionName, actor); + if (typeof outcome !== 'object') { + console.warn('action returns outcome that is not object', actionName, thing, outcome); + } + this.doEffects(outcome.effects, actor, actor); + return outcome; + } + + throw(actor, what, x, y) { + const item = actor.inventory.remove(what); + if (!item) { return false; } + item.x = (typeof x === 'number') ? x : actor.x; + item.y = (typeof y === 'number') ? y : actor.y; + const containers = this.findThings(x, y).filter((thing) => { + console.log(thing); + return thing.hasSpace(); + }); + if (containers.length) { + const container = containers[0]; + container.addToInventory(item); + return `${actor.name} puts ${what.name} into the ${container.name}.`; + } + this.addItem(item); + return `${actor.name} throws down a ${what.name}.`; + } + + doInitiative() { + const livingActors = this.actors.filter((actor) => !actor.dead()); + let orderedActors = random.shuffle(livingActors); + // TODO: Look for initiative boost, put at top of list + } + + static removeEffects(effects, key) { + let i = effects.indexOf(key); + while (i > -1) { + effects.splice(i, 1); + i = effects.indexOf(key); + } + } + + resolveRoundEffects() { + this.actors.forEach((actor) => { + const roundEffects = actor.activateAbilities('round'); + this.doEffects(roundEffects, actor, actor); + }); + } + + resolveCombatEffects(attacker, defender) { + let attackEffects = attacker.activateAbilities('attack'); + let defendEffects = defender.activateAbilities('attacked'); + let damageEffects; + let damagedEffects; + + attackEffects.push('attack'); + + console.log(attacker.name, JSON.stringify(attackEffects), 'vs', defender.name, JSON.stringify(defendEffects), defender); + + if (defendEffects.includes('cancelAttack')) { + Level.removeEffects(attackEffects, 'attack'); + } + + if (attackEffects.includes('attack')) { + attackEffects.push('weaponDamage'); + } + if (attackEffects.includes('damage') || attackEffects.includes('weaponDamage')) { + damageEffects = defender.activateAbilities('damage'); + attackEffects = attackEffects.concat(damageEffects); + damagedEffects = defender.activateAbilities('damaged'); + defendEffects = defendEffects.concat(damagedEffects); + } + if (defendEffects.includes('cancelDamage') || attackEffects.includes('cancelDamage')) { + Level.removeEffects(attackEffects, 'damage'); + Level.removeEffects(attackEffects, 'weaponDamage'); + } + + console.log(attacker.name, JSON.stringify(attackEffects), 'vs', JSON.stringify(defendEffects)); + + const outcomeAttack = this.doEffects(attackEffects, attacker, defender); + const outcomeDefend = this.doEffects(defendEffects, defender, attacker); + + // TODO: generate messages + + return { outcomeAttack, outcomeDefend, attackEffects, defendEffects }; + } + + doEffects(effects, actor, opponent) { + let damage = 0; + if (!effects) { return { damage }; } + effects.forEach((effect) => { + damage += this.doEffect(effect, actor, opponent); + }); + return { damage }; + } + + doEffect(effect, actor, opponent) { + console.log('doEffect', effect); + let damage = 0; + switch(effect) { + case 'damage': + damage += 1; + opponent.wound(1); + break; + case 'weaponDamage': + damage += actor.getWeaponDamage(); + opponent.wound(damage); + break; + case 'heal': + case 'hp': + actor.heal(1); + break; + case 'ap': + actor.healPool('ap', 1); + break; + case 'bp': + actor.healPool('bp', 1); + break; + case 'ep': + actor.healPool('ep', 1); + break; + case 'moveSwitch': { + const { x, y } = actor; + actor.setCoordinates(opponent.x, opponent.y); + opponent.setCoordinates(x, y); + } + break; + case 'apDamage': + actor.damagePool('ap', 1); + break; + case 'bpDamage': + actor.damagePool('bp', 1); + break; + case 'epDamage': + actor.damagePool('ep', 1); + break; + case 'push': + this.pushActor(actor, opponent); + break; + case 'pushAoe': + this.pushActor(actor, opponent); + // TODO: Handle aoe --> everyone (accept opponent) around hitX, hitY gets knocked back + break; + case 'moveBack': + this.pushActor(opponent, actor); + break; + case 'initiative': + actor.initiativeBoost = 1; + break; + case "fire": + // TODO: + break; + case "endGame": + // TODO + break; + case "score1000": + actor.score += 1000; + break; + default: + this.doCustomEffect(effect, actor, opponent); + break; + } + return damage; + } + + doCustomEffect(effect, actor, opponent) { + if (typeof this.customEffects[effect] === 'function') { + this.customEffects[effect](effect, actor, opponent); + } + } + + pushActor(pusher, pushee) { + const { x, y } = pusher; + let moveX = pushee.x - x; + let moveY = pushee.y - y; + moveX = moveX / (moveX === 0 ? 1 : Math.abs(moveX)); + moveY = moveY / (moveY === 0 ? 1 : Math.abs(moveY)); + // console.log('pushing', pushee.name, moveX, moveY); + pushee.move(moveX, moveY); + } + + getActorsInitiativeOrdered() { + const randomActors = random.shuffle(this.actors); + const firstActors = []; + let i = randomActors.length - 1; + while (i--) { + const actor = randomActors[i]; + if (actor.initiativeBoost > 0) { + firstActors.push(actor); + randomActors.splice(i, 1); + } + } + return firstActors.concat(randomActors); + } + + coolOffInitiativeBoosts() { + this.actors.forEach((actor) => { + actor.initiativeBoost = 0; + }); + } + + // Generation + + generateItem(levelItem = {}, Class, seed = 0, types = [], background = undefined) { + const { x, y } = this.findRandomFreeCell(seed, levelItem.clearing); + const itemTypeOptions = (levelItem.type && types[levelItem.type]) ? types[levelItem.type] : {}; + const itemOptions = { + x, y, + ...itemTypeOptions, + ...levelItem + }; + if (background) { itemOptions.background = background; } + const item = new Class(itemOptions); + return item; + } + + generateItems(options = {}, itemTypes = {}) { + let seed = this.seed + 200; + let { items = [] } = options; + + const arr = []; + items.forEach((levelItem) => { + const quantity = (typeof levelItem.quantity === 'number') ? levelItem.quantity : 1; + // TODO: handle weight, etc. + for (let i = 0; i < quantity; i++) { + const item = this.generateItem(levelItem, Item, ++seed, itemTypes); + arr.push(item); + } + }); + return arr; + } + + generateProps(options = {}, propTypes = {}) { + let seed = this.seed + 100; + let { props = [] } = options; + const background = this.background; + + const arr = []; + props.forEach((levelProp) => { + const quantity = (typeof levelProp.quantity === 'number') ? levelProp.quantity : 1; + // TODO: handle weight, etc. + for (let i = 0; i < quantity; i++) { + const prop = this.generateItem(levelProp, Prop, ++seed, propTypes, background); + arr.push(prop); + } + }); + return arr; + } + + generateActors(options = {}, refData = {}) { + console.log('generateActors', options); + let seed = this.seed + 999; // ? + const { monsterSpawn, monsters = [] } = options; + const monsterTypes = refData.monsters; + const depth = options.levelIndex; + const availableMonsters = monsters.filter((levelMonster) => { + return (!levelMonster.minDepth) || levelMonster.minDepth <= depth; + }); + const availableMonsterWeights = {}; + availableMonsters.forEach((levelMonster) => { + if (levelMonster.weight) { + availableMonsterWeights[levelMonster.type] = levelMonster.weight; + } + }); + const hasMonstersWithWeights = Object.keys(availableMonsterWeights).length > 0; + const monsterSpawnNumber = (monsterSpawn === undefined) ? 10 : random.roll(monsterSpawn); + const totalMonsterSpawnQuantity = monsterSpawnNumber; + const actors = []; + // Create monsters with fixed quantities + // Note: this could exceed the total quantity + const availableMonstersFixedQuantities = availableMonsters.filter((levelMonster) => { + return levelMonster.quantity; + }); + availableMonstersFixedQuantities.forEach((levelMonster) => { + const monsterTypeKey = levelMonster.type; + for (let i = 0; i < levelMonster.quantity; i++) { + const monster = this.createActor(monsterTypes, monsterTypeKey, ++seed); + actors.push(monster); + } + }); + // Create weighted monsters + if (hasMonstersWithWeights) { + let stopper = 0; + while (actors.length < totalMonsterSpawnQuantity && stopper < 9000) { + stopper++; + (() => { + const monsterTypeKey = random.getWeightedValue(availableMonsterWeights); + if (!monsterTypeKey) { return; } + const monster = this.createActor(monsterTypes, monsterTypeKey, ++seed); + actors.push(monster); + })(); + } + } + // console.log('Actors at depth', depth, actors); + return actors; + } + + createActor(monsterTypes, monsterTypeKey, seed) { + if (!monsterTypeKey) { return; } + const { x, y } = this.findRandomFreeCell(seed); + let monsterOptions = monsterTypes[monsterTypeKey]; + monsterOptions = { type: monsterTypeKey, aggro: 100, ...monsterOptions, x, y }; + // console.log(monsterTypes, monsterTypeKey, monsterOptions); + return new Actor(monsterOptions); + } + + // Gets + + getMap() { + return this.map; + } + + getCellPassability(x, y) { + const isMapPassable = this.map.getCellPassability(x, y); + if (!isMapPassable) { return false; } + const actorsHere = this.actors.filter((actor) => { + return actor.x === x && actor.y === y && !actor.passable; + }); + return (actorsHere.length === 0); + } + + // Sets + + setEye(actorThing) { + this.eye.x = actorThing.x; + this.eye.y = actorThing.y; + this.eye.sightRange = actorThing.sightRange; + } +} + +module.exports = Level; + + +/***/ }), +/* 13 */ +/***/ (function(module, exports, __webpack_require__) { + +const ROT = __webpack_require__(3); +const Cell = __webpack_require__(21); +const geometer = __webpack_require__(7); +const random = __webpack_require__(5); + +const DIGGER_TYPE = 'digger'; + +class Map { + constructor(options = {}) { + this.baseSeed = options.seed || 1; + this.seed = this.baseSeed; + this.color = options.color || '#777'; + this.background = options.background || '#222'; + this.type = options.type || DIGGER_TYPE; + this.rotMap = options.rotMap; + this.cells = {}; + this.freeCells = []; + this.walls = Boolean(options.walls) || Boolean(options.wallsCharacter); + this.wallsCharacter = options.wallsCharacter || '#'; // ▧ + this.floorCharacter = options.floorCharacter || '.'; + this.generate(options); + } + + generate(options = {}) { + const generators = options.generators || {}; + + if (typeof generators[this.type] === 'function') { + this.clearCells(); + generators[this.type](this.seed, this, options); + return; + } + + if (this.type === DIGGER_TYPE) { + this.generateDigger(); + return; + } + if (this.type === ARENA_TYPE) { + this.generateArena(options.x, options.y); + return; + } + // TODO: handle other rot-js types + + console.warn('Undefined map type:', this.type, generators); + this.generateArena(3, 3); + // TODO: Have default be a big empty room instead? + } + + generateArena(x, y) { + ROT.RNG.setSeed(this.seed); + this.rotMap = new ROT.Map.Arena(x, y); + this.setupRotMap(); + } + + generateDigger() { + ROT.RNG.setSeed(this.seed); + this.rotMap = new ROT.Map.Digger(); + this.setupRotMap(); + } + + setupRotMap() { + this.clearCells(); + this.rotMap.create((x, y, value) => { + if (value) { + return; + } + this.setFloorAt(x, y); + }); + + if (this.walls) { + this.addWalls(); + } + } + + addWalls() { + this.forEachCell((cell, x, y) => { + Map.forEachDirection((dir, dirX, dirY) => { + const newX = x + dirX; + const newY = y + dirY; + const wallCell = this.getCellAt(newX, newY); + if (!wallCell) { + this.setWallAt(newX, newY); + } + }); + }); + } + + discoverCircle(x, y, radius) { + this.forEachCellInCircle(x, y, radius, (cell) => { + cell.discover(); + }); + } + + static parseKeyCoordinates(key) { + const parts = key.split(","); + const x = parseInt(parts[0]); + const y = parseInt(parts[1]); + return { x, y }; + } + + static makeKey(x, y) { + return x + ',' + y; + } + + static forEachDirection(callback) { + const dirCoords = [ + {x: 0, y: -1}, // top + {x: 1, y: -1}, + {x: 1, y: 0}, // right + {x: 1, y: 1}, + {x: 0, y: 1}, // bottom + {x: -1, y: 1}, + {x: -1, y: 0}, // left + {x: -1, y: -1}, + ]; + for (let i = 0; i < 8; i++) { + callback(i, dirCoords[i].x, dirCoords[i].y); + } + } + + clearCells() { + this.cells = {}; + this.freeCells.length = 0; + } + + forEachCell(callback) { + for (let key in this.cells) { + const { x, y } = Map.parseKeyCoordinates(key); + callback(this.cells[key], x, y, key); + } + } + + forEachCellInCircle(centerX, centerY, radius, callback, includeEmptyCells = false) { + const maxX = centerX + radius; + const maxY = centerY + radius; + let x; + for (x = centerX - radius; x <= maxX; x++) { + let y; + for (y = centerY - radius; y <= maxY; y++) { + const r = Math.round(geometer.getDistance(centerX, centerY, x, y)); + if (r < radius) { + const cell = this.getCellAt(x, y); + if (cell || includeEmptyCells) { + callback(cell, x, y) + } + } + } + } + } + + getRandomFreeCell(seed) { + const i = random.roll(this.freeCells.length, seed); + + // TODO: TBD- Is it still a free cell? + // var key = freeCells.splice(index, 1)[0]; + // this.map[key] = "*"; + const key = this.freeCells[i]; + const cell = this.cells[key]; + + const { x, y } = Map.parseKeyCoordinates(key); + // console.log(seed, key, i, x, y); + return { x, y, cell }; + } + + getCellAt(x, y) { + const key = Map.makeKey(x, y); + return this.cells[key]; + } + + getCharacterAt(x, y) { + const cell = this.getCellAt(x, y); + return (cell) ? cell.getCharacter() : null; + } + + setFloorAt(x, y) { + const key = this.setCharacterAt(this.floorCharacter, x, y); + this.freeCells.push(key); + return key; + } + + setWallAt(x, y) { + return this.setCharacterAt(this.wallsCharacter, x, y); + } + + setCharacterAt(char, x, y) { + const key = Map.makeKey(x, y); + const cell = this.cells[key]; + if (cell) { + cell.setCharacter(char); + } else { + const { color, background } = this; + this.cells[key] = new Cell({ color, background, character: char }); + } + return key; + } + + getCellPassability(x, y) { + const cell = this.getCellAt(x, y); + return (cell) ? cell.getPassability() : false; + } + + getLightingAt(x, y) { + return {}; // TODO + } + + // _generateBoxes(freeCells) { + // for (var i=0;i<10;i++) { + // var index = Math.floor(ROT.RNG.getUniform() * freeCells.length); + // var key = freeCells.splice(index, 1)[0]; + // this.map[key] = "*"; + // } + // } +} + +module.exports = Map; + + +/***/ }), +/* 14 */ +/***/ (function(module, exports) { + +class Inventory { + constructor(options = {}) { + this.size = (typeof options.size === 'number') ? options.size : 10; + this.items = []; + } + + isFull() { + return (this.items.length >= this.size); + } + + hasSpace() { + return !this.isFull(); + } + + add(item) { + if (this.isFull()) { + return false; + } + if (!item.portable) { + return false; + } + this.items.push(item); + return true; + } + + remove(item) { + const i = this.items.indexOf(item); + if (i <= -1) { + console.warn('nothing found in', this.items, item); + return false; + } + const arr = this.items.splice(i, 1); + return arr[0]; + } + + removeType(typeKey) { + const itemsOfType = this.items.filter((item) => { return item.type === typeKey; }); + if (itemsOfType.length === 0) { + return false; + } + return this.remove(itemsOfType[0]); + } + + get(n) { + if (typeof n === 'number') { + return this.items[n]; + } else if (typeof n === 'string') { + return this.items.find((item) => { return item.name === n; }); + } + return this.items; + } + + getString() { + const arr = this.items.map((item, i) => { return `[${(i + 1)}] ${item.name}`; }); + return (arr.length) ? arr.join(', ') : 'nothing'; + } + + loopOverContents(fn) { + this.items.forEach((item, i) => { + fn(item, i); + }); + } + + hasContents() { + return (this.items.length > 0); + } + + contains(itemName) { + let foundItem = this.items.find((item) => { return (item.name === itemName); }); + return Boolean(foundItem); + } + + containsType(typeName) { + let foundItem = this.items.find((item) => { return (item.type === typeName); }); + return Boolean(foundItem); + } +} + +module.exports = Inventory; + + +/***/ }), +/* 15 */ +/***/ (function(module, exports) { + + +module.exports = { + DIRS_4: Object.freeze([ + { x: 0, y: -1 }, + { x: 1, y: 0 }, + { x: 0, y: 1 }, + { x: -1, y: 0 }, + ]), + DIRS_8: Object.freeze([ + { x: 0, y: -1 }, + { x: 1, y: -1 }, + { x: 1, y: 0 }, + { x: 1, y: 1 }, + { x: 0, y: 1 }, + { x: -1, y: 1 }, + { x: -1, y: 0 }, + { x: -1, y: -1 }, + ]), + DIRS_8_DIAGNOLS: Object.freeze([ + { x: 1, y: -1 }, + { x: 1, y: 1 }, + { x: -1, y: 1 }, + { x: -1, y: -1 }, + ]), +}; + + +/***/ }), +/* 16 */ +/***/ (function(module, exports, __webpack_require__) { + +const Item = __webpack_require__(6); + +class Prop extends Item { + constructor(options = {}) { + options = { portable: false, ...options }; + super(options); + } +} + +module.exports = Prop; + + +/***/ }), +/* 17 */ +/***/ (function(module, exports, __webpack_require__) { + +const ROT = __webpack_require__(3); +const Game = __webpack_require__(19); +const Item = __webpack_require__(6); +const Map = __webpack_require__(13); +const Actor = __webpack_require__(8); +const Prop = __webpack_require__(16); +const Level = __webpack_require__(12); +const Display = __webpack_require__(11); +const random = __webpack_require__(5); +const ready = __webpack_require__(10); + +const rote = { + ROT, + Game, Level, Map, Item, Prop, Actor, Display, + random, + ready +}; + +if (window) { + window.rote = rote; +} + +module.exports = rote; + + +/***/ }), +/* 18 */ +/***/ (function(module, exports) { + +// shim for using process in browser +var process = module.exports = {}; + +// cached from whatever global is present so that test runners that stub it +// don't break things. But we need to wrap it in a try catch in case it is +// wrapped in strict mode code which doesn't define any globals. It's inside a +// function because try/catches deoptimize in certain engines. + +var cachedSetTimeout; +var cachedClearTimeout; + +function defaultSetTimout() { + throw new Error('setTimeout has not been defined'); +} +function defaultClearTimeout () { + throw new Error('clearTimeout has not been defined'); +} +(function () { + try { + if (typeof setTimeout === 'function') { + cachedSetTimeout = setTimeout; + } else { + cachedSetTimeout = defaultSetTimout; + } + } catch (e) { + cachedSetTimeout = defaultSetTimout; + } + try { + if (typeof clearTimeout === 'function') { + cachedClearTimeout = clearTimeout; + } else { + cachedClearTimeout = defaultClearTimeout; + } + } catch (e) { + cachedClearTimeout = defaultClearTimeout; + } +} ()) +function runTimeout(fun) { + if (cachedSetTimeout === setTimeout) { + //normal enviroments in sane situations + return setTimeout(fun, 0); + } + // if setTimeout wasn't available but was latter defined + if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) { + cachedSetTimeout = setTimeout; + return setTimeout(fun, 0); + } + try { + // when when somebody has screwed with setTimeout but no I.E. maddness + return cachedSetTimeout(fun, 0); + } catch(e){ + try { + // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally + return cachedSetTimeout.call(null, fun, 0); + } catch(e){ + // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error + return cachedSetTimeout.call(this, fun, 0); + } + } + + +} +function runClearTimeout(marker) { + if (cachedClearTimeout === clearTimeout) { + //normal enviroments in sane situations + return clearTimeout(marker); + } + // if clearTimeout wasn't available but was latter defined + if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) { + cachedClearTimeout = clearTimeout; + return clearTimeout(marker); + } + try { + // when when somebody has screwed with setTimeout but no I.E. maddness + return cachedClearTimeout(marker); + } catch (e){ + try { + // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally + return cachedClearTimeout.call(null, marker); + } catch (e){ + // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error. + // Some versions of I.E. have different rules for clearTimeout vs setTimeout + return cachedClearTimeout.call(this, marker); + } + } + + + +} +var queue = []; +var draining = false; +var currentQueue; +var queueIndex = -1; + +function cleanUpNextTick() { + if (!draining || !currentQueue) { + return; + } + draining = false; + if (currentQueue.length) { + queue = currentQueue.concat(queue); + } else { + queueIndex = -1; + } + if (queue.length) { + drainQueue(); + } +} + +function drainQueue() { + if (draining) { + return; + } + var timeout = runTimeout(cleanUpNextTick); + draining = true; + + var len = queue.length; + while(len) { + currentQueue = queue; + queue = []; + while (++queueIndex < len) { + if (currentQueue) { + currentQueue[queueIndex].run(); + } + } + queueIndex = -1; + len = queue.length; + } + currentQueue = null; + draining = false; + runClearTimeout(timeout); +} + +process.nextTick = function (fun) { + var args = new Array(arguments.length - 1); + if (arguments.length > 1) { + for (var i = 1; i < arguments.length; i++) { + args[i - 1] = arguments[i]; + } + } + queue.push(new Item(fun, args)); + if (queue.length === 1 && !draining) { + runTimeout(drainQueue); + } +}; + +// v8 likes predictible objects +function Item(fun, array) { + this.fun = fun; + this.array = array; +} +Item.prototype.run = function () { + this.fun.apply(null, this.array); +}; +process.title = 'browser'; +process.browser = true; +process.env = {}; +process.argv = []; +process.version = ''; // empty string to avoid regexp issues +process.versions = {}; + +function noop() {} + +process.on = noop; +process.addListener = noop; +process.once = noop; +process.off = noop; +process.removeListener = noop; +process.removeAllListeners = noop; +process.emit = noop; +process.prependListener = noop; +process.prependOnceListener = noop; + +process.listeners = function (name) { return [] } + +process.binding = function (name) { + throw new Error('process.binding is not supported'); +}; + +process.cwd = function () { return '/' }; +process.chdir = function (dir) { + throw new Error('process.chdir is not supported'); +}; +process.umask = function() { return 0; }; + + +/***/ }), +/* 19 */ +/***/ (function(module, exports, __webpack_require__) { + +const ROT = __webpack_require__(3); +const ready = __webpack_require__(10); +const Display = __webpack_require__(11); +const Level = __webpack_require__(12); +const Actor = __webpack_require__(8); +const Item = __webpack_require__(6); +const Keyboard = __webpack_require__(22); +const MusicBox = __webpack_require__(23); +const Console = __webpack_require__(24); + +const INIT_STATE = 'INIT'; +const MAIN_GAME_STATE = 'GAME'; +const SPLASH_STATE = 'SPLASH'; +const OFF_STATE = 'OFF'; + +class Game { + constructor(options) { + const { id, consoleId, data, customEffects, haveSplash, fontFamilies } = options; + this.id = id; + this.displayContainer = document.getElementById(id || 'display'); + this.console = new Console({ id: consoleId }); + this.display = null; + this.haveSplash = Boolean(haveSplash); + this.fontFamilies = fontFamilies || []; + this.activeLevelIndex = 0; + // The generated levels + this.levels = []; + // Custom funcitons for generating things + this.generators = options.generators || {}; + // Reference data on prototypical "things" (monsters, items) + this.data = { + monsters: {}, + items: {}, + props: {}, + playlist: [], + }; + // The main actor + this.hero = null; // player character / player actor + // Guts + this.scheduler = new ROT.Scheduler.Simple(); + this.engine = null; + this.keyboard = null; + this.state = INIT_STATE; + this.states = new Set([INIT_STATE, SPLASH_STATE, MAIN_GAME_STATE, OFF_STATE]); + // this.setupEngine(); + this.loadingPromise = null; + this.console.setup(); + this.loadData(data); + this.hooks = {}; + this.customEffects = { ...customEffects }; + } + + setupEngine() { + this.engine = new ROT.Engine(this.scheduler); + this.engine.start(); + return this.engine; + } + + setupKeyboard() { + this.keyboard = new Keyboard({ state: MAIN_GAME_STATE, autoStart: true }); + // Splash state + this.keyboard.on(SPLASH_STATE, 'ENTER', () => { + this.setState(MAIN_GAME_STATE); + }); + // Main state + this.keyboard.on(MAIN_GAME_STATE, 'DIRECTION', (keyName, keyCode, direction) => { + // TODO: Lock and unlock the game? or do something else to determine if it's OK to move + this.hero.queueAction('move', { direction }); + this.advance(); + }); + this.keyboard.on(MAIN_GAME_STATE, 'ENTER', () => { + // this.actorDefaultAction(this.hero); // TODO: Remove me + this.actorAddDefaultAction(this.hero); + this.advance(); + }); + this.keyboard.on(MAIN_GAME_STATE, 'SPACE', () => { + this.hero.queueAction('wait'); + this.advance(); + }); + this.keyboard.on(MAIN_GAME_STATE, 't', () => { + this.showInventory(); + this.print('> Throw which item?'); + let n = prompt('Throw which item? \n\n' + this.hero.inventory.getString()); + if (!n || n.length === 0) { + this.print('None'); + return; + } + n = parseInt(n, 10); + const i = (isNaN(n)) ? -1 : n - 1; + const item = this.hero.inventory.get(i); + if (item) { + this.hero.queueAction('throw', { what: item, x: this.hero.x, y: this.hero.y }); + this.advance(); + } else { + this.print(`Invalid item [${n}]`); + } + }); + this.keyboard.on(MAIN_GAME_STATE, 'i', () => { this.showInventory(); }); + this.keyboard.on(MAIN_GAME_STATE, 'p', () => { + this.hero.queueAction('pickup'); + this.advance(); + }); + this.keyboard.on(MAIN_GAME_STATE, 'o', () => { + this.hero.queueAction('look'); + this.advance(); + }); + for (let i = 0; i < 9; i++) { + const key = String(i + 1); + this.keyboard.on(MAIN_GAME_STATE, key, () => { + this.hero.readyAbilityByIndex(i); + this.draw(); + }); + } + // this.keyboard.start(); + } + + setupMusic() { + this.music = new MusicBox(this.data.playlist); + } + + removeKeyboard() { + // TODO: this.keyboard.off() on all listeners + } + + createDisplay(options = {}) { + this.display = new Display(options); + this.display.setupElements(); + } + + //---- Draw / Render + + print(str, classes = '', wait = 0) { + if (wait) { + setTimeout(() => { this.print(str, classes); }, wait); + return; + } + this.console.print(str, classes); + } + + showInventory() { + const items = this.hero.inventory.getString(); + this.print('Inventory: ' + items); + } + + draw() { + this.display.drawLevel(this, this.getActiveLevel(), this.hero); + } + + + //---- Generation + + createLevel(options = {}, seed) { + options.seed = seed || options.seed; + const levelOptions = { + customEffects: this.customEffects, + ...options, + levelIndex: this.levels.length, + generators: this.generators, + }; + // console.warn(this.customEffects, levelOptions); + const level = new Level(levelOptions, this.data); + this.levels.push(level); + return level; + } + + createLevels(arr = [], baseSeed = 1) { + let seed = baseSeed; + arr.forEach((item, i) => { + seed += i; + if (typeof item === 'string') { // level type key + this.createLevel(this.getLevelType(item), seed); + } else if (typeof item === 'object' && item !== null) { + const n = (typeof item.repeat === 'number') ? item.repeat : 1; + for (let r = 0; r < n; r++) { + seed += r; + this.createLevel(this.getLevelType(item.levelTypeKey), seed); + } + } + }); + this.connectStairs(); + return this.levels; + } + + createActor(options = {}, level) { + const actor = new Actor(options); + this.scheduler.add(actor, true); + level = (level === true) ? this.getActiveLevel() : level; + if (level) { + level.addActor(actor); + } + return actor; + } + + createItem(options = {}, level) { + const item = new Item(options); + level = (level === true) ? this.getActiveLevel() : level; + if (level) { + level.addItem(item); + } + return item; + } + + createHero(options = {}) { + const heroOptions = { ...options, character: '@', isHero: true }; + this.hero = this.createActor(heroOptions, true); + + const g = this; + // Setup action stuff ... this needs to be refactored + this.hero.act = function () { + g.engine.lock(); + window.addEventListener('keydown', this); // pass the hero; the `handleEvent` will be used + }; + this.hero.handleEvent = function (e) { // Leftover from tutorial, part 2 + window.removeEventListener('keydown', this); + g.engine.unlock(); + }; + if (this.display) { + this.display.setCameraTarget(this.hero); + } + this.discoverAroundHero(); + return this.hero; + } + + connectStairs() { + const STAIR_LINK = 'stairLink'; + const propTypes = this.getDataPropArray(); + const stairsDownTypes = propTypes.filter((propType) => { return Boolean(propType[STAIR_LINK]); }); + this.levels.forEach((level, i) => { + // Handle each type of stairs + stairsDownTypes.forEach((stairsDownType) => { + const stairDownTypeKey = stairsDownType.key; + const stairUpTypeKey = stairsDownType[STAIR_LINK]; + const levelStairsDown = level.props.filter((prop) => { + return prop.type === stairDownTypeKey; + }); + levelStairsDown.forEach((stair) => { + const levelBelow = this.levels[i + 1]; + if (!levelBelow) { return; } + const possibleStairsUp = levelBelow.props.filter((prop) => { + return prop.type === stairUpTypeKey && !Boolean(prop.teleport); + }); + // TODO: Find stairs to connect to based on proximity + const levelBelowStairsUp = possibleStairsUp[0]; // TODO: remove this + this.connectTeleportProps(levelBelowStairsUp, stair, i, i + 1, 'ascend', 'descend'); + }); + }); + }); + } + + connectTeleportProps(prop1, prop2, levelIndex1, levelIndex2, verb1, verb2) { + if (!prop1 || !prop2) { return; } + prop1.setTeleport({ + levelIndex: levelIndex1, x: prop2.x, y: prop2.y, verb: verb1 + }); + prop2.setTeleport({ + levelIndex: levelIndex2, x: prop1.x, y: prop1.y, verb: verb2 + }); + } + + //---- Movement, Combat + + moveActor(actor, direction, bumpCombat = false) { + const diff = ROT.DIRS[8][direction]; + var newX = actor.x + diff[0]; + var newY = actor.y + diff[1]; + return this.moveActorTo(actor, newX, newY, bumpCombat); + } + + moveActorTo(actor, x, y, bumpCombat = false) { + const level = this.getActiveLevel(); + const canMoveToCell = level.getCellPassability(x, y); + // console.log('considering moving to', x, y, '... free?', canMoveToCell); + if (!canMoveToCell) { + const blocker = level.findActor(x, y); + if (blocker) { + return this.bump(actor, blocker, x, y, bumpCombat); + } + return { x: x, y: y, moved: false }; + } + actor.moveTo(x, y); + // TODO: just redraw the space that was under the actor and the actor in the new spot? + if (actor.isHero) { + this.discoverAroundHero(); + this.narrateAroundHero(); + } + this.draw(); + return { x, y, moved: true }; + } + + bump(actor, blocker, x, y, bumpCombat) { + if (bumpCombat && actor.faction !== blocker.faction) { + this.resolveCombat(actor, blocker, x, y); + return { x, y, moved: false }; + } else if (Game.canBumpSwitch(actor, blocker)) { + actor.moveTo(x, y); + return { x, y, moved: true }; + // TODO: allow pushes based on authority/size + } else { // just blocked + return { x, y, moved: false }; + } + } + + static canBumpSwitch(actor, blocker) { + if (actor.aggro || blocker.aggro) { // TOOD: make this more nuanced + return false; + } + const blockersNextAction = blocker.getNextAction(); + if (!blockersNextAction) { return true; } + return ( + blockersNextAction.verb === 'move' && + blockersNextAction.x === actor.x && + blockersNextAction.y === actor.y + ); + } + + teleportActor(actor, teleportParams = {}) { + const originalLevelIndex = this.activeLevelIndex; + // console.warn('teleporting', actor, teleportParams); + const { levelIndex, x, y } = teleportParams; + const currentLevel = this.getActiveLevel(); + currentLevel.removeActor(actor); + this.setActiveLevel(levelIndex); + const newLevel = this.getActiveLevel(); + newLevel.addActor(actor); + actor.setCoordinates(x, y); + console.log('New Level:', newLevel); + if (actor.isHero) { + this.discoverAroundHero(); + this.narrateAroundHero(); + } + if (originalLevelIndex !== levelIndex) { + this.hook('afterTeleportLevel', { levelIndex, x, y }); + } + // this.draw(); + } + + resolveCombat(actor, opponent, x, y) { + const level = this.getActiveLevel(); + if (!actor || !opponent || actor.faction === opponent.faction) { + return false; + } + const { outcomeAttack } = level.resolveCombatEffects(actor, opponent); + // TODO: get messages from resolve and effects methods + g.print(`${actor.name} attacks ${opponent.name} and does ${outcomeAttack.damage} damage!`); + if (opponent.dead()) { + g.print(`${opponent.name} has been killed.`); + actor.score += (this.activeLevelIndex + 1) * 10; + } + } + + actorAddDefaultAction(actor) { + const level = this.getActiveLevel(); + const thing = level.findThingSmart(actor.x, actor.y, 'portable'); + // TODO: Maybe get multiple things and check if they have actions? + console.log(thing, actor.x, actor.y); + if (!thing) { + return; + } + if (thing.portable) { + actor.queueAction('pickup', { target: thing }); + } else if (thing.hasAction('open')) { + actor.queueAction('open', { target: thing }); + } else if (thing.hasAction('use')) { + actor.queueAction('use', { target: thing }); + } else if (thing.hasAction('descend') || thing.hasAction('ascend')) { + actor.queueAction('teleport', { teleport: thing.teleport }); + console.log('Planning to teleport...', actor.actionQueue); + } + } + + advance() { + const startHp = this.hero.hp; + // TODO: advance time + // Do actions for all actors + const level = this.getActiveLevel(); + level.resolveRoundEffects(); + const actors = level.getActorsInitiativeOrdered(); + level.coolOffInitiativeBoosts(); + actors.forEach((actor) => { + actor.planAction(level, this.hero); + this.advanceActor(actor); + }); + // this.advanceActor(this.hero); + const isDamaged = (startHp > this.hero.hp); + this.display.drawDamage(isDamaged); + if (this.hero.dead()) { + this.hook('afterHeroDeath', {}); + } + this.draw(); + } + + advanceActor(actor) { + const level = this.getActiveLevel(); + const action = actor.doAction(); + if (!action) { return; } + const { verb, target, what, x, y } = action; + if (actor.isHero) { + if (verb === 'move') { console.log(actor.name + ' ' + verb); } + else { console.log(actor.name, verb, action); } + } + let outcome = {}; + let message = ''; + switch (verb) { + case 'move': + const bumpCombat = (actor.isHero || actor.aggro > 0); + if (action.direction === undefined) { + this.moveActorTo(actor, action.x, action.y, bumpCombat); + } else { + this.moveActor(actor, action.direction, bumpCombat); + } + break; + case 'use': + outcome = level.useThing(actor, 'use', target); + message = outcome.message; + break; + case 'open': + outcome = level.useThing(actor, 'open', target); + message = outcome.message; + break; + case 'teleport': + message = `${actor.name} now entering: `; + this.teleportActor(actor, action.teleport); + { + const newLevel = this.getActiveLevel(); + message += newLevel.name; + if (newLevel.description) { + message += ' - ' + newLevel.description; + } + } + break; + case 'pickup': + const pickedUp = this.pickupItem(actor, target); + if (pickedUp) { + message = `${actor.name} picks up the ${target.name}.`; + } else if (target) { + message = `${actor.name} could not pick up the ${target.name}.`; + } else { + message = `Nothing to pick up.`; + } + break; + case 'throw': + message = level.throw(actor, what, x, y); + break; + case 'look': + const things = this.getActiveLevel().findEverythingInView({ excludeHero: true }); + const names = things.map((thing) => thing.name || '?').join(', '); + message = `${actor.name} looks around and sees: ${names}`; + break; + case 'wait': + actor.wait(); + if (actor.isHero) { + message = `${actor.name} waits (random recovery of AP, BP, or EP points).`; + } + break; + } + if (typeof message !== 'string') { + console.error('Unknown message from doing action', verb); + message = 'ERROR'; + } + this.print(message); + } + + pickupItem(actor, thing) { + if (!thing) { return false; } + if (!thing.portable) { return false; } + const level = this.getActiveLevel(); + const item = level.removeItem(thing); + if (!item) { return false; } + const added = actor.inventory.add(thing); + if (!added) { + level.addItem(item); + } + return added; + } + + //---- Exploration + + discoverAroundHero() { + const level = this.getActiveLevel(); + const illumination = this.hero.inventory.items.reduce((n, item) => { + return n + item.illumination; + }, 0); + level.discoverCircle(this.hero.x, this.hero.y, this.hero.sightRange + illumination); // TODO: allow different POV + level.setEye(this.hero); + } + + narrateAroundHero() { + const allThingsOnHero = this.getThingsOnActor(this.hero); + if (allThingsOnHero.length === 0) { return; } + const namesOnHero = allThingsOnHero.map((thing) => thing.name); + const namesString = (namesOnHero.length > 1) ? namesOnHero.join(', ') : 'a ' + namesOnHero[0]; + this.console.print(`You are on ${namesString}.`); + } + + //---- System + + ready(callback, fonts = []) { // TODO: remove fonts param? + const fontFamiliesToLoad = [ ...fonts ].concat(this.fontFamilies); + console.log(fontFamiliesToLoad); + ready(() => { + if (this.loadingPromise instanceof Promise) { + this.loadingPromise + .then(() => { callback(); }); + // .catch((err) => { console.error('Error loading something', err) }); + } else { + callback(); + } + }, fontFamiliesToLoad); + // TODO: return a promise so can be used async + } + + start() { + this.setupEngine(); + this.setupKeyboard(); + this.setupMusic(); + const startState = (this.haveSplash) ? SPLASH_STATE : MAIN_GAME_STATE; + this.setStateDetect(startState); + // TODO: start graphics loop + this.draw(); + } + + stop() { + this.setState(OFF_STATE); + this.removeKeyboard(); + // TODO: stop graphics loop + } + + loadData(data) { + const promises = []; + function parseJson(response) { return response.json(); } + function fixInnerObject(obj, key) { + return (typeof obj[key] === 'object') ? obj[key] : obj; + } + for (let key in data) { + if (typeof data[key] === 'string') { + const p = fetch(data[key]) + .then(parseJson) + .then((obj) => fixInnerObject(obj, key)) + .then((obj) => { this.setData(key, obj); }); + //.catch((err) => { console.error(data, key, err); }); + promises.push(p); + } else { + this.setData(key, data[key]); + } + } + this.loadingPromise = Promise.all(promises); // .then((resp) => { console.log(resp); }); + return this.loadingPromise; + } + + //---- Hooks + + addHook(hookName, fn) { + if (!this.hooks[hookName]) { + this.hooks[hookName] = []; + } + this.hooks[hookName].push(fn); + } + + removeHook(hookName, fn) { + if (!this.hooks[hookName]) { return; } + const i = this.hooks[hookName].indexOf(fn); + this.hooks[hookName].splice(i, 1); + } + + hook(hookName, data = {}) { + const hook = this.hooks[hookName]; + if (!hook) { return; } + hook.forEach((fn) => { + fn(data, this, hookName); + }); + } + + //---- Gets + + getActiveLevel() { + return this.levels[this.activeLevelIndex]; + } + + getLevelType(key) { + const lt = this.data.levels[key]; + if (typeof lt !== 'object' || lt === null) { + console.error('Cannot find level type ', key); + } + return lt; + } + + getDataPropArray() { + const propKeys = Object.keys(this.data.props); + const arr = []; + propKeys.forEach((key) => { + const prop = { ...this.data.props[key], key }; + arr.push(prop); + }); + return arr; + } + + getThingsOnActor(actor) { + const { x, y } = actor; + return this.getActiveLevel().findThings(x, y); + } + + //---- Sets + + setActiveLevel(i) { + this.activeLevelIndex = i; + return; + } + + setData(key, obj) { + this.data[key] = Object.freeze(obj); + } + + setStateDetect(stateFallback) { + // TODO: improve... not sure i like how this works + // const hash = location.hash.substring(1).toUpperCase(); + // if (this.states.includes(hash)) { + // return this.setState(hash); + // } + return this.setState(stateFallback); + } + + setState(state) { + const isLegitState = this.states.has(state); + const consoleMethod = (isLegitState) ? 'log' : 'warn'; + console[consoleMethod]('Setting state:', state); + const prefix = 'rote-state-'; + this.state = state; + // const body = document.getElementsByClassName('rote-state')[0]; + const body = document.getElementsByTagName('body')[0]; + // body.className = 'rote-state'; // TODO: make this smarter so it only removes rote states + // body.classList.add(prefix + this.state.toLowerCase()); + body.className = 'rote-state ' + prefix + state.toLowerCase(); + location.hash = state.toLowerCase(); + this.keyboard.setState(state); + } + + setMainGameState() { + this.setState(MAIN_GAME_STATE); + } + +} + +module.exports = Game; + + +/***/ }), +/* 20 */ +/***/ (function(module, exports, __webpack_require__) { + +/* Font Face Observer v2.1.0 - © Bram Stein. License: BSD-3-Clause */(function(){function l(a,b){document.addEventListener?a.addEventListener("scroll",b,!1):a.attachEvent("scroll",b)}function m(a){document.body?a():document.addEventListener?document.addEventListener("DOMContentLoaded",function c(){document.removeEventListener("DOMContentLoaded",c);a()}):document.attachEvent("onreadystatechange",function k(){if("interactive"==document.readyState||"complete"==document.readyState)document.detachEvent("onreadystatechange",k),a()})};function t(a){this.a=document.createElement("div");this.a.setAttribute("aria-hidden","true");this.a.appendChild(document.createTextNode(a));this.b=document.createElement("span");this.c=document.createElement("span");this.h=document.createElement("span");this.f=document.createElement("span");this.g=-1;this.b.style.cssText="max-width:none;display:inline-block;position:absolute;height:100%;width:100%;overflow:scroll;font-size:16px;";this.c.style.cssText="max-width:none;display:inline-block;position:absolute;height:100%;width:100%;overflow:scroll;font-size:16px;"; +this.f.style.cssText="max-width:none;display:inline-block;position:absolute;height:100%;width:100%;overflow:scroll;font-size:16px;";this.h.style.cssText="display:inline-block;width:200%;height:200%;font-size:16px;max-width:none;";this.b.appendChild(this.h);this.c.appendChild(this.f);this.a.appendChild(this.b);this.a.appendChild(this.c)} +function u(a,b){a.a.style.cssText="max-width:none;min-width:20px;min-height:20px;display:inline-block;overflow:hidden;position:absolute;width:auto;margin:0;padding:0;top:-999px;white-space:nowrap;font-synthesis:none;font:"+b+";"}function z(a){var b=a.a.offsetWidth,c=b+100;a.f.style.width=c+"px";a.c.scrollLeft=c;a.b.scrollLeft=a.b.scrollWidth+100;return a.g!==b?(a.g=b,!0):!1}function A(a,b){function c(){var a=k;z(a)&&a.a.parentNode&&b(a.g)}var k=a;l(a.b,c);l(a.c,c);z(a)};function B(a,b){var c=b||{};this.family=a;this.style=c.style||"normal";this.weight=c.weight||"normal";this.stretch=c.stretch||"normal"}var C=null,D=null,E=null,F=null;function G(){if(null===D)if(J()&&/Apple/.test(window.navigator.vendor)){var a=/AppleWebKit\/([0-9]+)(?:\.([0-9]+))(?:\.([0-9]+))/.exec(window.navigator.userAgent);D=!!a&&603>parseInt(a[1],10)}else D=!1;return D}function J(){null===F&&(F=!!document.fonts);return F} +function K(){if(null===E){var a=document.createElement("div");try{a.style.font="condensed 100px sans-serif"}catch(b){}E=""!==a.style.font}return E}function L(a,b){return[a.style,a.weight,K()?a.stretch:"","100px",b].join(" ")} +B.prototype.load=function(a,b){var c=this,k=a||"BESbswy",r=0,n=b||3E3,H=(new Date).getTime();return new Promise(function(a,b){if(J()&&!G()){var M=new Promise(function(a,b){function e(){(new Date).getTime()-H>=n?b(Error(""+n+"ms timeout exceeded")):document.fonts.load(L(c,'"'+c.family+'"'),k).then(function(c){1<=c.length?a():setTimeout(e,25)},b)}e()}),N=new Promise(function(a,c){r=setTimeout(function(){c(Error(""+n+"ms timeout exceeded"))},n)});Promise.race([N,M]).then(function(){clearTimeout(r);a(c)}, +b)}else m(function(){function v(){var b;if(b=-1!=f&&-1!=g||-1!=f&&-1!=h||-1!=g&&-1!=h)(b=f!=g&&f!=h&&g!=h)||(null===C&&(b=/AppleWebKit\/([0-9]+)(?:\.([0-9]+))/.exec(window.navigator.userAgent),C=!!b&&(536>parseInt(b[1],10)||536===parseInt(b[1],10)&&11>=parseInt(b[2],10))),b=C&&(f==w&&g==w&&h==w||f==x&&g==x&&h==x||f==y&&g==y&&h==y)),b=!b;b&&(d.parentNode&&d.parentNode.removeChild(d),clearTimeout(r),a(c))}function I(){if((new Date).getTime()-H>=n)d.parentNode&&d.parentNode.removeChild(d),b(Error(""+ +n+"ms timeout exceeded"));else{var a=document.hidden;if(!0===a||void 0===a)f=e.a.offsetWidth,g=p.a.offsetWidth,h=q.a.offsetWidth,v();r=setTimeout(I,50)}}var e=new t(k),p=new t(k),q=new t(k),f=-1,g=-1,h=-1,w=-1,x=-1,y=-1,d=document.createElement("div");d.dir="ltr";u(e,L(c,"sans-serif"));u(p,L(c,"serif"));u(q,L(c,"monospace"));d.appendChild(e.a);d.appendChild(p.a);d.appendChild(q.a);document.body.appendChild(d);w=e.a.offsetWidth;x=p.a.offsetWidth;y=q.a.offsetWidth;I();A(e,function(a){f=a;v()});u(e, +L(c,'"'+c.family+'",sans-serif'));A(p,function(a){g=a;v()});u(p,L(c,'"'+c.family+'",serif'));A(q,function(a){h=a;v()});u(q,L(c,'"'+c.family+'",monospace'))})})}; true?module.exports=B:(undefined);}()); + + +/***/ }), +/* 21 */ +/***/ (function(module, exports) { + +class Cell { + constructor(options = {}) { + this.character = options.character || ' '; + this.discovered = false; + this.color = options.color || '#777'; + this.background = options.background || '#222'; + this.passability = false; // TODO: handle this different? + } + + // Gets + + getPassability() { // TODO: update this + return (this.character === '.'); + } + + getCharacter() { + return this.character; + } + + getForegroundColor(inView = true) { + if (!this.discovered) { + return '#000'; + } + return (inView) ? this.color : '#232120'; + } + + getBackgroundColor(inView = true) { + if (!this.discovered) { + return '#000'; + } + return (inView) ? this.background : '#111010'; + } + + // Sets + + setCharacter(char) { + this.character = char; + } + + discover() { + this.discovered = true; + } +} + +module.exports = Cell; + + +/***/ }), +/* 22 */ +/***/ (function(module, exports) { + +const DIRECTION8 = { + 'UP': 0, 'UP-RIGHT': 1, + 'RIGHT': 2, 'DOWN-RIGHT': 3, + 'DOWN': 4, 'DOWN-LEFT': 5, + 'LEFT': 6, 'UP-LEFT': 7 +}; +const DIRECTION4 = { 'UP': 0, 'RIGHT': 1, 'DOWN': 2, 'LEFT': 3 }; +const DIRECTION4_ARRAY = ['UP', 'RIGHT', 'DOWN', 'LEFT']; + +const USED_KEYS = ['i', 't', 'o', 'p', '1', '2', '3', '4', '5', '6', '7', '8', '9']; +const KEY_MAP = { + "9": "TAB", + "13": "ENTER", + "27": "ESC", + "32": "SPACE", +}; +KEY_MAP[38] = 'UP'; // up +KEY_MAP[33] = 'UP-RIGHT'; +KEY_MAP[39] = 'RIGHT'; // right +KEY_MAP[34] = 'DOWN-RIGHT'; +KEY_MAP[40] = 'DOWN'; // down +KEY_MAP[35] = 'DOWN-LEFT'; +KEY_MAP[37] = 'LEFT'; // left +KEY_MAP[36] = 'UP-LEFT'; + +const WASD_KEYMAP = { + 87: 'UP', // w + 65: 'LEFT', // a + 83: 'DOWN', // s + 68: 'RIGHT', // d +}; +const WASD_DIAGONAL = { + ...WASD_KEYMAP, + 81: 'UP-LEFT', // q + 69: 'UP-RIGHT', // e + 90: 'DOWN-LEFT', // z + 67: 'DOWN-RIGHT', // c +}; +const VI_KEYMAP = { + 72: 'LEFT', // h + 74: 'DOWN', // j + 75: 'UP', // k + 76: 'RIGHT', // l +}; +const VI_DIAGONAL = { + ...VI_KEYMAP, + 89: 'UP-LEFT', // y + 85: 'UP-RIGHT', // u + 66: 'DOWN-LEFT', // b + 78: 'DOWN-RIGHT', // n +}; + + +const UNSPECIFIED_STATE = 'UNSPECIFIED'; + +class KeyboardListener { + constructor(options = {}) { + this.callbacks = {}; + this.isListening = false; + this.state = options.state || UNSPECIFIED_STATE; + this.autoStart = (options.autoStart === undefined) ? false : Boolean(options.autoStart); + } + + setState(state = UNSPECIFIED_STATE) { + this.state = state.toString(); + } + + on(state, key, callback) { + // key can be a keyCode or a keyType like 'DIRECTION' + this.callbacks[state + '_' + key] = callback; + if (this.autoStart) { + this.start(); + } + } + + off(state, key, callback) { + // TODO: remove callback + // TODO: if no more callbacks then stop + } + + getKeyMap() { + let keyMap = { ...KEY_MAP }; + // TODO: variations based on options selected + keyMap = { ...keyMap, ...WASD_DIAGONAL, ...VI_DIAGONAL }; + return keyMap; + } + + handleEvent(e) { + const keyMap = this.getKeyMap(); + const { keyCode, key } = e; + const isKeyUsed = USED_KEYS.includes(key) || (keyCode in keyMap); + + if (!isKeyUsed) { + console.log('Keyboard handleEvent - unaccounted for key:', key, keyCode); + return; + } + e.preventDefault(); + + // Lookup key name and direction + const keyName = keyMap[keyCode] || key; + const direction = DIRECTION8[keyName]; + // console.log('handleEvent', e, keyName, keyCode, direction); + + // Callbacks + if (direction !== undefined) { + const typeCallback = this.callbacks[this.state + '_DIRECTION']; + if (typeCallback) { + typeCallback(keyName, keyCode, direction); + } + } + const callback = this.callbacks[this.state + '_' + keyName]; + // console.log(this.state + '_' + keyName, callback); + if (callback) { + callback(keyName, keyCode, direction); + } + } + + start() { + if (this.isListening) { + return; + } + window.addEventListener('keydown', this); // pass this; the `handleEvent` will be used + this.isListening = true; + } + + stop() { + // TODO: remove event listener + } +} + +module.exports = KeyboardListener; + + +/***/ }), +/* 23 */ +/***/ (function(module, exports) { + + +class MusicBox { + constructor(playlist) { + this.audio = null; + this.playlist = [ ...playlist ]; + } + + addToPlaylist(songPath) { + this.playlist.push(songPath); + } + + play(i = 0) { + this.audio = new Audio(this.playlist[i]); + this.audio.play(); + } +} + +module.exports = MusicBox; + + +/***/ }), +/* 24 */ +/***/ (function(module, exports) { + +class Console { + constructor(options = {}) { + this.id = options.id || 'console'; + this.container = null; + this.list = null; + this.messages = []; + this.writeToConsoleLog = false; + } + + setup() { + this.container = document.getElementById(this.id); + this.clear(); + } + + clear() { + this.messages.length = 0; + this.container.innerHTML = '
    '; + this.list = this.container.firstChild; + } + + print(str, classes = '') { + if (!str) { + return; + } + if (this.writeToConsoleLog) { + console.log('%c' + str, 'color: #559955'); + } + const safeStr = str.replace('<', '<'); + this.list.innerHTML += `
  • ${safeStr}
  • `; + this.container.scrollTop = this.container.scrollHeight; + this.trim(); + } + + // aliases + log(str) { return this.print(str); } + add(str) { return this.print(str); } + + trim() { + if (this.list.innerHTML.length > 5000) { + this.list.removeChild(this.list.firstChild); + } + } +} + +module.exports = Console; + + +/***/ }) +/******/ ]); +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./node_modules/rot-js/lib/rng.js","webpack:///./node_modules/rot-js/lib/util.js","webpack:///./node_modules/rot-js/lib/color.js","webpack:///./node_modules/rot-js/lib/display/canvas.js","webpack:///./node_modules/rot-js/lib/display/hex.js","webpack:///./node_modules/rot-js/lib/display/rect.js","webpack:///./node_modules/rot-js/lib/display/tile.js","webpack:///./node_modules/rot-js/lib/display/tile-gl.js","webpack:///./node_modules/rot-js/lib/text.js","webpack:///./node_modules/rot-js/lib/constants.js","webpack:///./node_modules/rot-js/lib/display/display.js","webpack:///./node_modules/rot-js/lib/stringgenerator.js","webpack:///./node_modules/rot-js/lib/eventqueue.js","webpack:///./node_modules/rot-js/lib/scheduler/scheduler.js","webpack:///./node_modules/rot-js/lib/scheduler/simple.js","webpack:///./node_modules/rot-js/lib/scheduler/speed.js","webpack:///./node_modules/rot-js/lib/scheduler/action.js","webpack:///./node_modules/rot-js/lib/scheduler/index.js","webpack:///./node_modules/rot-js/lib/fov/fov.js","webpack:///./node_modules/rot-js/lib/fov/discrete-shadowcasting.js","webpack:///./node_modules/rot-js/lib/fov/precise-shadowcasting.js","webpack:///./node_modules/rot-js/lib/fov/recursive-shadowcasting.js","webpack:///./node_modules/rot-js/lib/fov/index.js","webpack:///./node_modules/rot-js/lib/map/map.js","webpack:///./node_modules/rot-js/lib/map/arena.js","webpack:///./node_modules/rot-js/lib/map/dungeon.js","webpack:///./node_modules/rot-js/lib/map/features.js","webpack:///./node_modules/rot-js/lib/map/uniform.js","webpack:///./node_modules/rot-js/lib/map/cellular.js","webpack:///./node_modules/rot-js/lib/map/digger.js","webpack:///./node_modules/rot-js/lib/map/ellermaze.js","webpack:///./node_modules/rot-js/lib/map/dividedmaze.js","webpack:///./node_modules/rot-js/lib/map/iceymaze.js","webpack:///./node_modules/rot-js/lib/map/rogue.js","webpack:///./node_modules/rot-js/lib/map/index.js","webpack:///./node_modules/rot-js/lib/noise/noise.js","webpack:///./node_modules/rot-js/lib/noise/simplex.js","webpack:///./node_modules/rot-js/lib/noise/index.js","webpack:///./node_modules/rot-js/lib/path/path.js","webpack:///./node_modules/rot-js/lib/path/dijkstra.js","webpack:///./node_modules/rot-js/lib/path/astar.js","webpack:///./node_modules/rot-js/lib/path/index.js","webpack:///./node_modules/rot-js/lib/engine.js","webpack:///./node_modules/rot-js/lib/lighting.js","webpack:///./node_modules/rot-js/lib/index.js","webpack:///./node_modules/rot-js/lib/display/backend.js","webpack:///./src/random.js","webpack:///./src/Item.js","webpack:///./src/geometer.js","webpack:///./src/Actor.js","webpack:///./node_modules/rot-js/lib/display/term.js","webpack:///./src/ready.js","webpack:///./src/Display.js","webpack:///./src/Level.js","webpack:///./src/Map.js","webpack:///./src/Inventory.js","webpack:///./src/constants.js","webpack:///./src/Prop.js","webpack:///./src/index.js","webpack:///./node_modules/process/browser.js","webpack:///./src/Game.js","webpack:///./node_modules/fontfaceobserver/fontfaceobserver.standalone.js","webpack:///./src/Cell.js","webpack:///./src/KeyboardListener.js","webpack:///./src/MusicBox.js","webpack:///./src/Console.js"],"names":[],"mappings":";QAAA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;;QAEA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;;;QAGA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA,0CAA0C,gCAAgC;QAC1E;QACA;;QAEA;QACA;QACA;QACA,wDAAwD,kBAAkB;QAC1E;QACA,iDAAiD,cAAc;QAC/D;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,yCAAyC,iCAAiC;QAC1E,gHAAgH,mBAAmB,EAAE;QACrI;QACA;;QAEA;QACA;QACA;QACA,2BAA2B,0BAA0B,EAAE;QACvD,iCAAiC,eAAe;QAChD;QACA;QACA;;QAEA;QACA,sDAAsD,+DAA+D;;QAErH;QACA;;;QAGA;QACA;;;;;;;;AClFA;AACA,oDAAoD;AACpD;AACA;AACA,oCAAoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,mBAAmB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,gDAAgD;AAChE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACe,sFAA6B,EAAC;;;;;;;;ACvI7C;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA,WAAW,OAAO;AAClB,WAAW,IAAI;AACf;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8CAA8C,IAAI,IAAI;AACtD;AACA;AACA;AACA;;;;;;;;ACrDA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAkC;AACP;AACpB;AACP;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,OAAO;AACtC;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD;AACzD;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,mBAAmB,OAAO;AAC1B,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,mBAAmB,OAAO;AAC1B,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,mBAAmB,OAAO;AAC1B,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,mBAAmB,OAAO;AAC1B,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA,mBAAmB,OAAO;AAC1B;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACO;AACP;AACA;AACA,mBAAmB,OAAO;AAC1B;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACO;AACP;AACA,0BAA0B,uDAAG;AAC7B;AACA;AACA,mBAAmB,OAAO;AAC1B,yDAAyD,uDAAG;AAC5D;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP,iCAAiC,sDAAK;AACtC,kBAAkB,kBAAkB;AACpC;AACO;AACP,iCAAiC,sDAAK;AACtC,eAAe,iBAAiB;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;AC/UmC;AACpB,MAAM,aAAM,SAAS,0BAAO;AAC3C;AACA;AACA;AACA;AACA,kBAAkB,2BAA2B;AAC7C,oBAAoB,yBAAyB;AAC7C;AACA;AACA,2CAA2C,eAAe;AAC1D,wBAAwB,MAAM,GAAG,cAAc,KAAK,gBAAgB;AACpE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AClCiC;AACA;AACjC;AACA;AACA;AACA;AACe,MAAM,OAAG,SAAS,aAAM;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,mBAAG,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACvIiC;AACjC;AACA;AACA;AACA;AACe,MAAM,SAAI,SAAS,aAAM;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY,SAAI;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,kBAAkB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAI;;;ACxG6B;AACjC;AACA;AACA;AACA;AACe,MAAM,SAAI,SAAS,aAAM;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA,yCAAyC,SAAS;AAClD;AACA,6CAA6C;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AC7EmC;AACE;AACrC;AACA;AACA;AACA;AACe,MAAM,cAAM,SAAS,0BAAO;AAC3C;AACA,wEAAwE,8BAA8B;AACtG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,2BAA2B;AAC7C,oBAAoB,wBAAwB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA,yCAAyC,SAAS;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA,6BAA6B,eAAe;;AAE5C,qDAAqD;AACrD;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,qBAAqB,OAAO;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wEAAwE,8BAA8B;AACtG;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,CAAC;AACD;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,OAAO;AAClC;AACA;AACA;AACA;AACA,qBAAqB,uBAAgB;AACrC;AACA;AACA;AACA;AACA;AACA;;;;;;ACjRA;AACA;AACA;AACA;AACA,2BAA2B,IAAI,IAAI;AACnC;AACO;AACA;AACA;AACA;AACP;AACA;AACA;AACO;AACP,kBAAkB;AAClB;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B;AAC/B;AACA,yCAAyC;AACzC;AACA;AACA;AACA,sCAAsC;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD;AACzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B;AAC9B;AACA;AACA,gDAAgD;AAChD;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA,iBAAiB,qBAAqB,EAAE;AACxC;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,KAAK;AAChB,aAAa,OAAO;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACrLA;AACO;AACP;AACO;AACA;AACP;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA,8BAA8B;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC9T2B;AACE;AACA;AACK;AACL;AACM;AAC6B;AAChE;AACA,WAAW,OAAG;AACd,YAAY,SAAI;AAChB,YAAY,SAAI;AAChB,eAAe,cAAM;AACrB,YAAY,uBAAI;AAChB;AACA;AACA,WAAW,aAAa;AACxB,YAAY,cAAc;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf;AACA;AACA;AACA;AACA;AACA;AACe,MAAM,eAAO;AAC5B,4BAA4B;AAC5B;AACA,4BAA4B;AAC5B;AACA,kCAAkC;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,sBAAsB;AACxC;AACA;AACA;AACA,oBAAoB,qCAAqC;AACzD;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,iBAAiB,OAAO;AACxB;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,iBAAiB,IAAI;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,MAAM;AACrB,iBAAiB,OAAO;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,mBAAmB;AAClC,eAAe,OAAO;AACtB,eAAe,OAAO;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,EAAE,GAAG,EAAE;AAC5B;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,OAAO,yDAAyD,KAAK,IAAI,KAAK,qBAAqB,KAAK;AACvH,eAAe,IAAI;AACnB,iBAAiB,IAAI;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,QAAa;AAClC,+BAA+B;AAC/B;AACA;AACA,qBAAqB,SAAc;AACnC;AACA,mCAAmC,wBAAwB;AAC3D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA,qBAAqB,OAAY;AACjC;AACA;AACA,qBAAqB,OAAY;AACjC;AACA;AACA,qBAAqB,YAAiB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA;AACA,aAAa;AACb;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,OAAO;AACtB,eAAe,KAAK;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAO,QAAQ,SAAI;AACnB,eAAO,OAAO,OAAG;AACjB,eAAO,QAAQ,SAAI;AACnB,eAAO,UAAU,cAAM;AACvB,eAAO,QAAQ,uBAAI;;;ACxPQ;AAC3B;AACA;AACA;AACA;AACA;AACe,MAAM,+BAAe;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,yBAAyB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,OAAO;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,mBAAmB;AAC1C;AACA;AACA,kEAAkE;AAClE,yCAAyC,mBAAmB;AAC5D;AACA;AACA,2BAA2B,oBAAoB;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA,eAAe,SAAS;AACxB,eAAe,OAAO;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,sBAAG;AAClB;AACA;AACA,eAAe;AACf,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC5Ie;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,OAAO;AACxB;AACA,eAAe,mBAAmB;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,EAAE;AACjB,eAAe,OAAO;AACtB;AACA;AACA;AACA,uBAAuB,6BAA6B;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,UAAU;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB;AACvB;AACA,2BAA2B,6BAA6B;AACxD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,EAAE;AACjB,iBAAiB,OAAO;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,EAAE;AACjB,iBAAiB,KAAK;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxF0C;AAC3B,MAAM,mBAAS;AAC9B;AACA;AACA;AACA;AACA,0BAA0B,UAAU;AACpC;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,8BAA8B;AAC7C;AACA,eAAe,EAAE;AACjB,eAAe,KAAK;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,EAAE;AACjB,iBAAiB,OAAO;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,EAAE;AACjB,iBAAiB,KAAK;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;;;ACjEuC;AACvC;AACA;AACA;AACe,MAAM,aAAM,SAAS,mBAAS;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACfuC;AACvC;AACA;AACA;AACe,MAAM,WAAK,SAAS,mBAAS;AAC5C;AACA,eAAe,OAAO;AACtB,eAAe,KAAK;AACpB,eAAe,OAAO;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACxBuC;AACvC;AACA;AACA;AACA;AACe,MAAM,aAAM,SAAS,mBAAS;AAC7C;AACA;AACA,kCAAkC;AAClC,+CAA+C;AAC/C;AACA;AACA,eAAe,OAAO;AACtB,eAAe,KAAK;AACpB,eAAe,OAAO;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AClDiC;AACF;AACE;AAClB,+CAAC,CAAC,qBAAM,EAAE,kBAAK,EAAE,qBAAM,EAAE,EAAC;;;ACHF;AACvC;AACA;AACe,MAAM,OAAG;AACxB;AACA;AACA,eAAe,SAAS;AACxB,eAAe,OAAO;AACtB,eAAe,IAAI;AACnB;AACA,iDAAiD;AACjD;AACA,uCAAuC,cAAc;AACrD;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,IAAI;AACxB,oBAAoB,IAAI;AACxB,oBAAoB,IAAI;AACxB,oBAAoB,IAAI;AACxB;AACA;AACA;AACA,uBAAuB,IAAI;AAC3B;AACA;AACA;AACA;AACA,uBAAuB,IAAI;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,iBAAiB;AACxC,2BAA2B,qBAAqB;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC7D2B;AAC3B;AACA;AACA;AACA;AACe,MAAM,4CAAqB,SAAS,OAAG;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA,2BAA2B,sBAAsB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB,aAAa;AACb,SAAS;AACT;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,KAAK;AACpB,eAAe,QAAQ;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AChG2B;AAC3B;AACA;AACA;AACA;AACe,MAAM,0CAAoB,SAAS,OAAG;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA,2BAA2B,mBAAmB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB,aAAa;AACb,SAAS;AACT;AACA;AACA,eAAe,OAAO;AACtB,eAAe,OAAO;AACtB,eAAe,KAAK;AACpB,eAAe,QAAQ;AACvB;AACA;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mDAAmD;AACnD;AACA;AACA,0EAA0E;AAC1E;AACA;AACA,mDAAmD;AACnD;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;;;AC7H2B;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACe,MAAM,8CAAsB,SAAS,OAAG;AACvD;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA,uBAAuB,oBAAoB;AAC3C;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA,+CAA+C;AAC/C,mDAAmD;AACnD,2CAA2C;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA,+CAA+C;AAC/C;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,MAAM;AACrB,eAAe,MAAM;AACrB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA,yBAAyB,aAAa;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACpJgE;AACF;AACI;AACnD,yCAAC,CAAC,mEAAqB,EAAE,gEAAoB,EAAE,sEAAsB,EAAE,EAAC;;;ACHvB;AAChE;AACe,MAAM,OAAG;AACxB;AACA;AACA,eAAe,IAAI;AACnB,eAAe,IAAI;AACnB;AACA,wBAAwB,aAAa,WAAW,cAAc;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,iBAAiB;AACxC;AACA,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA;AACA;;;ACvB2B;AAC3B;AACA;AACA;AACA;AACe,MAAM,WAAK,SAAS,OAAG;AACtC;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;;;ACjB2B;AAC3B;AACA;AACA;AACA;AACe,MAAM,eAAO,SAAS,OAAG;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA,gBAAgB,oBAAoB;AACpC;AACA;AACA,iBAAiB;AACjB;AACA,oBAAoB,wBAAwB;AAC5C;;;ACrB4B;AAC5B;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,IAAI;AACf;AACO,MAAM,aAAI;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,sBAAG;AACvB;AACA;AACA,qBAAqB,sBAAG;AACxB,sBAAsB;AACtB,oCAAoC,sBAAG;AACvC;AACA;AACA,uBAAuB;AACvB,oCAAoC,sBAAG;AACvC;AACA;AACA,sBAAsB;AACtB,oCAAoC,sBAAG;AACvC;AACA;AACA,uBAAuB;AACvB,oCAAoC,sBAAG;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,sBAAG;AACvB;AACA;AACA,qBAAqB,sBAAG;AACxB,iCAAiC,sBAAG;AACpC,iCAAiC,sBAAG;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,sBAAG;AACvB;AACA;AACA,qBAAqB,sBAAG;AACxB;AACA;AACA,gCAAgC,sBAAG;AACnC,gCAAgC,sBAAG;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,YAAY;AACtC,6BAA6B,aAAa;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,YAAY;AACtC,6BAA6B,aAAa;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B,YAAY;AACtC,6BAA6B,aAAa;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,iBAAiB;AAChC,gBAAgB,iBAAiB;AACjC,cAAc,iBAAiB;AAC/B,iBAAiB,iBAAiB;AAClC;AACA;AACA;AACA;AACA,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,IAAI;AACf;AACO,MAAM,iBAAQ;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,sBAAG;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,YAAY;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,YAAY;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC7TmC;AACY;AACnB;AAC5B;AACA;AACA;AACA;AACA;AACe,MAAM,eAAO,SAAS,eAAO;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC;AAChC,oCAAoC;AACpC,6BAA6B;AAC7B,+BAA+B;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,iBAAiB;AAC5C,+BAA+B,kBAAkB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,aAAI;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,KAAK;AACtB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,wBAAwB;AACnD;AACA;AACA;AACA;AACA,gCAAgC,sBAAG;AACnC;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,gCAAgC,sBAAG;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA,iBAAiB,SAAS;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD;AAChD;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA,oDAAoD;AACpD;AACA;AACA;AACA,yDAAyD;AACzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oEAAoE;AACpE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,YAAY;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sCAAsC,QAAQ;AAC9C;AACA;AACA;AACA;AACA,+BAA+B,sBAAG;AAClC;AACA;AACA;AACA;AACA;AACA,uBAAuB,mBAAmB;AAC1C;AACA;AACA,+BAA+B,iBAAQ;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC3V2B;AACY;AACX;AAC5B;AACA;AACA;AACA;AACA,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,OAAO;AAClB,WAAW,MAAM;AACjB,WAAW,MAAM;AACjB,WAAW,IAAI;AACf;AACe,MAAM,iBAAQ,SAAS,OAAG;AACzC,2CAA2C;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,IAAI;AACzB;AACA;AACA;AACA;AACA,eAAe,MAAM,oDAAoD;AACzE;AACA;AACA,uBAAuB,iBAAiB;AACxC,2BAA2B,kBAAkB;AAC7C,mCAAmC,sBAAG;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB,uCAAuC;AAChE,sBAAsB,yBAAyB;AAC/C;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,iBAAiB;AACrD;AACA;AACA,2DAA2D;AAC3D;AACA;AACA,8DAA8D;AAC9D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,iBAAiB;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,uBAAuB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,SAAS;AACxB,eAAe,IAAI;AACnB,eAAe,SAAS;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,kBAAkB;AACzC,4CAA4C,iBAAiB;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,sBAAG;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B;AAC5B,0BAA0B;AAC1B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,OAAO;AAC9B;AACA;AACA,oCAAoC,sBAAG;AACvC;AACA;AACA;AACA;AACA,yCAAyC,sBAAG;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,YAAY;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,WAAW;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;;;AChUmC;AACY;AACnB;AACW;AACvC;AACA,YAAY,aAAI;AAChB,gBAAgB,iBAAQ;AACxB;AACA;AACA;AACA;AACA;AACA;AACe,MAAM,aAAM,SAAS,eAAO;AAC3C,2CAA2C;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,6DAA6D;AAC7D,6EAA6E,8BAA8B,EAAE;AAC7G;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,SAAS,yEAAyE;AAClF;AACA;AACA,2BAA2B,iBAAiB;AAC5C,+BAA+B,kBAAkB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC;AACvC;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,8BAA8B;AAC/D;AACA;AACA;AACA;AACA,mBAAmB,aAAI;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,iBAAiB,sBAAG,qBAAqB;AACzC;AACA;AACA;AACA;AACA;AACA,iBAAiB,KAAK;AACtB;AACA;AACA,0BAA0B,sBAAG;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,aAAI;AACnC;AACA;AACA,+BAA+B,iBAAQ;AACvC;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,IAAI;AACzB,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,IAAI;AACzB,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,wBAAwB;AAC/C;AACA;AACA;AACA;AACA;AACA;;;AChO2B;AACC;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACe,MAAM,mBAAS,SAAS,OAAG;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,OAAO;AAC9B;AACA;AACA;AACA,sBAAsB;AACtB;AACA,mBAAmB,sBAAsB;AACzC;AACA,2BAA2B,OAAO;AAClC;AACA;AACA;AACA;AACA;AACA,qCAAqC,sBAAG;AACxC;AACA;AACA;AACA;AACA,iCAAiC,sBAAG;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,OAAO;AAC9B;AACA;AACA;AACA;AACA;AACA,+CAA+C,sBAAG;AAClD;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,iBAAiB;AACxC,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA;AACA;;;ACjF2B;AACC;AAC5B;AACA;AACA;AACA;AACe,MAAM,uBAAW,SAAS,OAAG;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,OAAO;AAC9B;AACA,2BAA2B,OAAO;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,OAAO;AAC9B,2BAA2B,OAAO;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,aAAa;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,aAAa;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB,sBAAG;AACnB,gBAAgB,sBAAG;AACnB;AACA;AACA;AACA,sBAAsB;AACtB,6BAA6B,OAAO;AACpC;AACA;AACA;AACA;AACA,sBAAsB;AACtB,2BAA2B,cAAc;AACzC;AACA;AACA;AACA;AACA,sBAAsB;AACtB,6BAA6B,OAAO;AACpC;AACA;AACA;AACA;AACA,sBAAsB;AACtB,2BAA2B,cAAc;AACzC;AACA;AACA;AACA,oBAAoB,sBAAG;AACvB,uBAAuB,kBAAkB;AACzC;AACA;AACA;AACA;AACA,uBAAuB,sBAAG;AAC1B;AACA;AACA,2DAA2D;AAC3D,2DAA2D;AAC3D,2DAA2D;AAC3D,2DAA2D;AAC3D;AACA;;;ACvG2B;AACC;AAC5B;AACA;AACA;AACA;AACe,MAAM,iBAAQ,SAAS,OAAG;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,sBAAG;AACvC,oCAAoC,sBAAG;AACvC;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC,sBAAG;AACtC;AACA;AACA;AACA,mCAAmC,OAAO;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA,SAAS;AACT,uBAAuB,iBAAiB;AACxC,2BAA2B,kBAAkB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,OAAO;AAC9B;AACA;AACA;AACA,2BAA2B,sBAAG;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACzG2B;AACC;AACW;AACvC;AACA;AACA;AACA;AACe,MAAM,WAAK,SAAS,OAAG;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,iBAAiB;AAC5C,+BAA+B,kBAAkB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,6BAA6B;AACpD;AACA,2BAA2B,8BAA8B;AACzD,oCAAoC,qFAAqF;AACzH;AACA;AACA;AACA;AACA;AACA,kBAAkB,sBAAG;AACrB,kBAAkB,sBAAG;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB,sBAAG;AAC5B;AACA;AACA;AACA,6BAA6B,IAAI;AACjC,6BAA6B,IAAI;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B,sBAAG;AACjC;AACA;AACA;AACA,uBAAuB,6BAA6B;AACpD,2BAA2B,8BAA8B;AACzD;AACA;AACA;AACA,iCAAiC,sBAAG;AACpC;AACA;AACA;AACA,uCAAuC,IAAI;AAC3C,uCAAuC,IAAI;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC,qCAAqC;AAC5E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B,2BAA2B,QAAQ;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,sBAAG;AAC3B,wBAAwB,sBAAG;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C,sBAAG;AAC7C,0CAA0C,sBAAG;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iCAAiC,iBAAiB;AAClD,qCAAqC,iBAAiB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,sBAAG;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA,iBAAiB,sBAAG;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB,uBAAuB;AACvB;AACA;AACA,sBAAsB,sBAAG,cAAc;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,IAAI;AAC5B,wBAAwB,IAAI;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B,2BAA2B,QAAQ;AACnC;AACA,+BAA+B,gCAAgC;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC9V+B;AACI;AACE;AACJ;AACM;AACI;AACN;AACN;AAChB,6CAAC,CAAC,kBAAK,EAAE,wBAAO,EAAE,2BAAQ,EAAE,qBAAM,EAAE,8BAAS,EAAE,oCAAW,EAAE,2BAAQ,EAAE,kBAAK,EAAE,EAAC;;;ACR7F;AACA;AACA;AACe;AACf;;;ACJ+B;AACH;AACK;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACe,MAAM,eAAO,SAAS,KAAK;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,eAAe;AACtC;AACA;AACA,uBAAuB,sBAAG;AAC1B;AACA;AACA,uBAAuB,mBAAmB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC;AACvC;AACA,iCAAiC;AACjC;AACA;AACA;AACA,uBAAuB;AACvB;AACA,0BAA0B;AAC1B;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,8BAA8B;AAC9B;AACA,iCAAiC;AACjC;AACA;AACA,iBAAiB,mBAAG;AACpB,iBAAiB,mBAAG;AACpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACtGmC;AACpB,2CAAC,CAAC,wBAAO,EAAE,EAAC;;;ACDY;AACvC;AACA;AACA,WAAW,IAAI;AACf,WAAW,IAAI;AACf,WAAW,SAAS;AACpB,WAAW,OAAO;AAClB,WAAW,IAAI;AACf;AACe,MAAM,SAAI;AACzB,wDAAwD;AACxD;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,qBAAqB,IAAI;AACzB,0CAA0C;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,uBAAuB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC5C6B;AAC7B;AACA;AACA;AACA;AACA;AACe,MAAM,iBAAQ,SAAS,SAAI;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,sBAAsB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC9D6B;AAC7B;AACA;AACA;AACA;AACA;AACe,MAAM,WAAK,SAAS,SAAI;AACvC,wDAAwD;AACxD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,sBAAsB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,uBAAuB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACzFqC;AACN;AAChB,0CAAC,CAAC,2BAAQ,EAAE,kBAAK,EAAE,EAAC;;;ACFnC;AACA;AACA,WAAW,cAAc;AACzB;AACe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,sBAAsB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA,wCAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;;;ACzCoC;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACe,MAAM,iBAAQ;AAC7B,kDAAkD;AAClD;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8DAA8D,uBAAgB;AAC9E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,mBAAmB;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC;AACvC;AACA;AACA,YAAY,iBAAU;AACtB;AACA,uBAAuB,0BAA0B,OAAO;AACxD;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,sCAAsC;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA,2BAA2B,OAAO;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qCAAqC;AACrC;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA,2BAA2B,OAAO;AAClC;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AClMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAA0C;AACgB;AACQ;AACV;AACI;AACZ;AACA;AACI;AACF;AACF;AACI;AACuB;AACzC;AAC3B,aAAa,IAAI;AACY;AAC7B,cAAc,SAAK;AACQ;AAC3B,aAAa,oBAAI;;;;;;;;ACjBxB;AAAA;AACA;AACA;AACA;AACe;AACf,oBAAoB,aAAa;AACjC,yBAAyB,yBAAyB;AAClD;;;;;;;ACPA,YAAY,mBAAO,CAAC,CAAQ;AAC5B;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,8BAA8B;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,kBAAkB;AACjC;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;ACjEA,kBAAkB,mBAAO,CAAC,EAAa;AACvC,OAAO,SAAS,GAAG,mBAAO,CAAC,EAAa;;AAExC;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,qCAAqC;AACvD;AACA;AACA;AACA;AACA,uBAAuB;AACvB;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEA,sBAAsB,QAAQ;AAC9B;AACA;AACA,yGAAyG,UAAU;AACnH,WAAW;AACX;AACA;AACA;AACA;AACA,UAAU;AACV;;AAEA,+BAA+B;AAC/B,yBAAyB,QAAQ;AACjC;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA,2BAA2B;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA,4BAA4B;AAC5B,oCAAoC;AACpC;AACA;AACA;AACA;AACA;AACA,SAAS,OAAO;AAChB;AACA,IAAI;AACJ;AACA;AACA;;AAEA,gBAAgB;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA,yBAAyB;AACzB,SAAS,yBAAyB;AAClC,mBAAmB;AACnB;AACA;AACA;;AAEA;;;;;;;AC5IA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;ACTA,YAAY,mBAAO,CAAC,CAAQ;AAC5B,kBAAkB,mBAAO,CAAC,EAAa;AACvC,iBAAiB,mBAAO,CAAC,CAAY;AACrC,eAAe,mBAAO,CAAC,CAAU;;AAEjC;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;;AAGA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,kCAAkC;AAClC;AACA;AACA;AACA,4BAA4B;AAC5B;AACA,4BAA4B;AAC5B;AACA,4BAA4B;AAC5B;AACA,4BAA4B;AAC5B;AACA,4BAA4B;AAC5B;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA,4BAA4B;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,8BAA8B;AAC9B,wBAAwB;AACxB;AACA;;AAEA;AACA;AACA;;AAEA;AACA,oBAAoB,QAAQ;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,UAAU;AAC5B,yCAAyC,QAAQ;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,eAAe;AAC/C,IAAI;AACJ;AACA,gCAAgC,eAAe;AAC/C;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,oBAAoB,UAAU;AAC9B;;AAEA;AACA,oBAAoB,SAAS,eAAe;AAC5C,sBAAsB;AACtB,sCAAsC,mBAAmB;AACzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,iBAAiB,UAAU;AAC3B;AACA;;AAEA;AACA;AACA,SAAS,OAAO;AAChB;AACA;;AAEA;AACA;AACA,oBAAoB,aAAa;AACjC;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,GAAG;AACH;AACA;;AAEA;AACA,0BAA0B,cAAc;AACxC;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA,iBAAiB,cAAc;AAC/B,uCAAuC,cAAc;AACrD;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,6CAA6C,oBAAoB;AACjE,oCAAoC,MAAM;AAC1C,oDAAoD,mBAAmB;AACvE,yCAAyC,QAAQ;AACjD;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,oBAAoB,mCAAmC;AACvD,oBAAoB,UAAU;AAC9B,SAAS,OAAO;AAChB;AACA;AACA;AACA,cAAc;AACd;;AAEA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,4DAA4D,cAAc;AAC1E;AACA;AACA,cAAc,kCAAkC;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;;;;;;;ACzcA;AAAA;AAAA;AAAmC;AACE;AACrC;AACA,mBAAmB,GAAG,EAAE,EAAE,cAAc;AACxC;AACA;AACA,mBAAmB,GAAG,EAAE,EAAE,eAAe,GAAG,EAAE,EAAE,cAAc;AAC9D;AACA;AACA,mBAAmB,OAAO,EAAE,MAAM;AAClC;AACA;AACA;AACA;AACA;AACA,cAAc,oDAAgB;AAC9B;AACA;AACA;AACA;AACA;AACe,mBAAmB,2DAAO;AACzC;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,2BAA2B;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,gEAAgE;AACvF,2BAA2B,eAAe;AAC1C,mBAAmB,sDAAsD;AACzE;;;;;;;;ACtFA,yBAAyB,mBAAO,CAAC,EAAkB;;AAEnD;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,IAAI;AACJ;AACA,EAAE;AACF;;AAEA;AACA;AACA;AACA;AACA;AACA,gBAAgB,qBAAqB,EAAE;AACvC,iBAAiB,oCAAoC,qBAAqB,EAAE;AAC5E;AACA;AACA;AACA;;AAEA;;;;;;;AC1BA,YAAY,mBAAO,CAAC,CAAQ;;AAE5B;AACA,yBAAyB;AACzB,aAAa;AACb;AACA;AACA;AACA;;AAEA;AACA,6CAA6C,oBAAoB;AACjE;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,4CAA4C;AAC5C;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,cAAc,QAAQ;AACtB;AACA;AACA;;AAEA;AACA,cAAc,QAAQ;AACtB;AACA;;AAEA,2CAA2C;AAC3C;AACA;;AAEA,wBAAwB,WAAW,cAAc;AACjD;AACA;;AAEA;AACA;AACA;AACA;AACA,YAAY,WAAW,OAAO,YAAY;AAC1C,YAAY,UAAU,OAAO,YAAY;AACzC,YAAY,aAAa,OAAO,YAAY;AAC5C;AACA;;AAEA;;AAEA,yB;;;;;;AC5FA,YAAY,mBAAO,CAAC,EAAO;AAC3B,cAAc,mBAAO,CAAC,CAAS;AAC/B,aAAa,mBAAO,CAAC,CAAQ;AAC7B,aAAa,mBAAO,CAAC,EAAQ;AAC7B,iBAAiB,mBAAO,CAAC,CAAY;AACrC,eAAe,mBAAO,CAAC,CAAU;AACjC,OAAO,0BAA0B,GAAG,mBAAO,CAAC,EAAa;;AAEzD;AACA,yBAAyB,cAAc;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd,wBAAwB;AACxB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;;AAEA,iBAAiB;AACjB,+DAA+D;AAC/D,oC;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,6CAA6C,oBAAoB,EAAE;AACnE;AACA;AACA;AACA;AACA;AACA,gB;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA,sCAAsC,qCAAqC,EAAE;AAC7E;AACA;AACA,sCAAsC,2BAA2B,EAAE;AACnE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA,wBAAwB;AACxB;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA,kCAAkC;AAClC;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,yBAAyB,cAAc,6BAA6B,EAAE,EAAE;AACxE;AACA;AACA;AACA,IAAI;AACJ;AACA,GAAG,E;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,cAAc,cAAc;AAC5B;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,aAAa,WAAW,QAAQ,UAAU,YAAY,eAAe;AACrE;AACA;AACA,YAAY,WAAW,iBAAiB,UAAU;AAClD;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA,UAAU;AACV;;AAEA;AACA;AACA,iBAAiB,SAAS,UAAU;AACpC;AACA;AACA,GAAG;AACH,UAAU;AACV;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW,OAAO;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,SAAS,OAAO;AAChB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,GAAG;AACH;;AAEA;;AAEA,4BAA4B;AAC5B,SAAS,OAAO;AAChB;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,qCAAqC;AACxD;AACA;AACA;;AAEA,2BAA2B,gBAAgB;AAC3C;AACA,OAAO,aAAa;;AAEpB;AACA;AACA;AACA;AACA,kBAAkB,cAAc;AAChC;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA,2BAA2B,gBAAgB;AAC3C;AACA,OAAO,aAAa;AACpB;;AAEA;AACA;AACA;AACA;AACA,kBAAkB,cAAc;AAChC;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA,4BAA4B,cAAc;AAC1C;AACA,6BAA6B;AAC7B,SAAS,8BAA8B;AACvC;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,kBAAkB,2BAA2B;AAC7C;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,QAAQ;AACnC;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;;AAEA;AACA,wBAAwB,QAAQ;AAChC,SAAS,OAAO;AAChB;AACA,oBAAoB;AACpB;AACA,mC;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,uBAAuB,cAAc;AACrC;AACA;AACA,GAAG;AACH;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;;;;;;;ACvkBA,YAAY,mBAAO,CAAC,CAAQ;AAC5B,aAAa,mBAAO,CAAC,EAAQ;AAC7B,iBAAiB,mBAAO,CAAC,CAAY;AACrC,eAAe,mBAAO,CAAC,CAAU;;AAEjC;;AAEA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sDAAsD;AACtD;AACA;AACA;;AAEA,sBAAsB;AACtB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,qB;AACA;;AAEA;AACA;AACA;AACA,qB;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA,G;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,GAAG;AACH;;AAEA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;;AAEA;AACA;AACA;;AAEA;AACA;AACA,IAAI,YAAY;AAChB,IAAI,YAAY;AAChB,IAAI,WAAW;AACf,IAAI,WAAW;AACf,IAAI,WAAW;AACf,IAAI,YAAY;AAChB,IAAI,YAAY;AAChB,IAAI,aAAa;AACjB;AACA,iBAAiB,OAAO;AACxB;AACA;AACA;;AAEA;AACA;AACA,4B;AACA;;AAEA;AACA;AACA,UAAU,OAAO;AACjB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,4BAA4B,WAAW;AACvC;AACA,6BAA6B,WAAW;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,0B;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,SAAS,OAAO;AAChB;AACA,UAAU;AACV;;AAEA;AACA;AACA,yB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH,UAAU,oBAAoB;AAC9B,+BAA+B,qCAAqC;AACpE;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,YAAY;AACZ;;AAEA;AACA,kBAAkB,KAAK;AACvB;AACA;AACA;AACA;AACA;AACA;;AAEA;;;;;;;ACrNA;AACA,yBAAyB;AACzB;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,mDAAmD,8BAA8B,EAAE;AACnF;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,GAAG;AACH,qCAAqC,wBAAwB,EAAE;AAC/D;AACA;AACA;;AAEA;AACA,2CAA2C,YAAY,QAAQ,IAAI,UAAU,EAAE,EAAE;AACjF;AACA;;AAEA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;;AAEA;AACA,6CAA6C,iCAAiC,EAAE;AAChF;AACA;;AAEA;AACA,6CAA6C,iCAAiC,EAAE;AAChF;AACA;AACA;;AAEA;;;;;;;;AC7EA;AACA;AACA,GAAG,cAAc;AACjB,GAAG,aAAa;AAChB,GAAG,aAAa;AAChB,GAAG,cAAc;AACjB;AACA;AACA,GAAG,cAAc;AACjB,GAAG,cAAc;AACjB,GAAG,aAAa;AAChB,GAAG,aAAa;AAChB,GAAG,aAAa;AAChB,GAAG,cAAc;AACjB,GAAG,cAAc;AACjB,GAAG,eAAe;AAClB;AACA;AACA,GAAG,cAAc;AACjB,GAAG,aAAa;AAChB,GAAG,cAAc;AACjB,GAAG,eAAe;AAClB;AACA;;;;;;;ACxBA,aAAa,mBAAO,CAAC,CAAQ;;AAE7B;AACA,yBAAyB;AACzB,aAAa;AACb;AACA;AACA;;AAEA;;;;;;;ACTA,YAAY,mBAAO,CAAC,CAAQ;AAC5B,aAAa,mBAAO,CAAC,EAAQ;AAC7B,aAAa,mBAAO,CAAC,CAAQ;AAC7B,YAAY,mBAAO,CAAC,EAAO;AAC3B,cAAc,mBAAO,CAAC,CAAS;AAC/B,aAAa,mBAAO,CAAC,EAAQ;AAC7B,cAAc,mBAAO,CAAC,EAAS;AAC/B,gBAAgB,mBAAO,CAAC,EAAW;AACnC,eAAe,mBAAO,CAAC,CAAU;AACjC,cAAc,mBAAO,CAAC,EAAS;;AAE/B;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;;;;;;ACtBA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,KAAK;AACL;AACA;AACA,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,uBAAuB,sBAAsB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB;AACrB;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,qCAAqC;;AAErC;AACA;AACA;;AAEA,2BAA2B;AAC3B;AACA;AACA;AACA,4BAA4B,UAAU;;;;;;;ACvLtC,YAAY,mBAAO,CAAC,CAAQ;AAC5B,cAAc,mBAAO,CAAC,EAAS;AAC/B,gBAAgB,mBAAO,CAAC,EAAW;AACnC,cAAc,mBAAO,CAAC,EAAS;AAC/B,cAAc,mBAAO,CAAC,CAAS;AAC/B,aAAa,mBAAO,CAAC,CAAQ;AAC7B,iBAAiB,mBAAO,CAAC,EAAoB;AAC7C,iBAAiB,mBAAO,CAAC,EAAY;AACrC,gBAAgB,mBAAO,CAAC,EAAW;;AAEnC;AACA;AACA;AACA;;AAEA;AACA;AACA,SAAS,+DAA+D;AACxE;AACA;AACA,8BAA8B,gBAAgB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,YAAY;AACZ,YAAY;AACZ;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB;AACxB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,gCAAgC,0CAA0C;AAC1E;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,kCAAkC,YAAY;AAC9C;AACA,GAAG;AACH;AACA,yCAAyC;AACzC;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,6CAA6C;AACjF;AACA,IAAI;AACJ,gCAAgC,EAAE;AAClC;AACA,GAAG;AACH,gDAAgD,sBAAsB,EAAE;AACxE;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH,iBAAiB,OAAO;AACxB;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,2BAA2B;AAC3B;AACA;AACA;;AAEA;;AAEA;AACA;AACA,qBAAqB,0BAA0B,EAAE;AACjD;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;AAGA;;AAEA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kCAAkC;AAClC;AACA,IAAI;AACJ;AACA,mBAAmB,OAAO;AAC1B;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,wBAAwB;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,wBAAwB;AACxB,uBAAuB;AACvB;;AAEA;AACA;AACA;AACA;AACA,4CAA4C,kBAAkB;AAC9D;AACA,wCAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,0DAA0D,sCAAsC,EAAE;AAClG;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA,MAAM;AACN;AACA,oDAAoD;AACpD;AACA,KAAK;AACL,IAAI;AACJ,GAAG;AACH;;AAEA;AACA,yBAAyB,QAAQ;AACjC;AACA;AACA,GAAG;AACH;AACA;AACA,GAAG;AACH;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;;AAEA;AACA;AACA;AACA,WAAW;AACX,GAAG;AACH;AACA,WAAW;AACX;AACA,GAAG,OAAO;AACV,WAAW;AACX;AACA;;AAEA;AACA,qCAAqC;AACrC;AACA;AACA;AACA,4BAA4B,aAAa;AACzC;AACA;AACA;AACA;AACA;AACA;;AAEA,yCAAyC;AACzC;AACA;AACA,SAAS,mBAAmB;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,mBAAmB;AACvD;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS,gBAAgB;AACzB;AACA,aAAa,WAAW,WAAW,cAAc,YAAY,qBAAqB;AAClF;AACA,cAAc,cAAc;AAC5B;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,gBAAgB;AAChD,GAAG;AACH,8BAA8B,gBAAgB;AAC9C,GAAG;AACH,6BAA6B,gBAAgB;AAC7C,GAAG;AACH,kCAAkC,2BAA2B;AAC7D;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,iCAAiC;AACjC;AACA;AACA;;AAEA;AACA;AACA;AACA,gBAAgB,QAAQ;AACxB,SAAS,2BAA2B;AACpC;AACA,yBAAyB,sCAAsC;AAC/D,SAAS,uCAAuC;AAChD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,WAAW;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,WAAW,gBAAgB,YAAY;AACzD,KAAK;AACL,kBAAkB,WAAW,yBAAyB,YAAY;AAClE,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+DAA+D,oBAAoB;AACnF;AACA,iBAAiB,WAAW,0BAA0B,MAAM;AAC5D;AACA;AACA;AACA;AACA,kBAAkB,WAAW;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,eAAe,cAAc;AAC7B,wBAAwB,cAAc;AACtC;AACA;AACA,cAAc,cAAc;AAC5B;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA,GAAG;AACH,sFAAsF;AACtF;AACA;;AAEA;AACA;AACA,qCAAqC,QAAQ;AAC7C;AACA;AACA,mCAAmC,YAAY;AAC/C;;AAEA;;AAEA,8BAA8B;AAC9B;AACA;AACA;AACA;AACA;AACA,kBAAkB,YAAY,EAAE;AAChC,yBAAyB,gDAAgD;AACzE,IAAI;AACJ;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,gCAAgC,wBAAwB;AACxD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qBAAqB,wBAAwB,EAAE;AAC/C,wBAAwB,+BAA+B,EAAE;AACzD;AACA,IAAI;AACJ;AACA;AACA;AACA,8CAA8C,qBAAqB,mBAAmB,EAAE;AACxF;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,8BAA8B,QAAQ;AACtC;AACA;AACA;;AAEA,yBAAyB;AACzB;AACA,cAAc,QAAQ;AACtB;AACA;AACA,GAAG;AACH;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA,GAAG;AACH;AACA;;AAEA;AACA,SAAS,OAAO;AAChB;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;;;;;;ACloBA,iFAAiF,gBAAgB,sFAAsF,cAAc,sGAAsG,mDAAmD,IAAI,yDAAyD,wHAAwH,GAAG,cAAc,qCAAqC,0CAA0C,+CAA+C,sCAAsC,sCAAsC,sCAAsC,sCAAsC,UAAU,qCAAqC,qBAAqB,kBAAkB,YAAY,WAAW,gBAAgB,eAAe,EAAE,qCAAqC,qBAAqB,kBAAkB,YAAY,WAAW,gBAAgB,eAAe;AAC1kC,qCAAqC,qBAAqB,kBAAkB,YAAY,WAAW,gBAAgB,eAAe,EAAE,2CAA2C,WAAW,YAAY,eAAe,eAAe,EAAE,2BAA2B,2BAA2B,2BAA2B;AACvT,gBAAgB,kCAAkC,eAAe,gBAAgB,qBAAqB,gBAAgB,kBAAkB,WAAW,SAAS,UAAU,WAAW,mBAAmB,oBAAoB,WAAW,EAAE,cAAc,8BAA8B,uBAAuB,iBAAiB,mCAAmC,6BAA6B,gBAAgB,aAAa,QAAQ,6BAA6B,QAAQ,SAAS,SAAS,MAAM,gBAAgB,YAAY,cAAc,6BAA6B,+BAA+B,iCAAiC,gCAAgC,aAAa,2DAA2D,2FAA2F,6BAA6B,UAAU,SAAS,aAAa,+BAA+B;AACj4B,aAAa,aAAa,oCAAoC,IAAI,0CAA0C,UAAU,oBAAoB,SAAS,gBAAgB;AACnK,+BAA+B,8DAA8D,iCAAiC,cAAc,gCAAgC,aAAa,6HAA6H,iCAAiC,IAAI,IAAI,8BAA8B,wBAAwB,qCAAqC,IAAI,EAAE,oCAAoC,gBAAgB,KAAK;AACzf,GAAG,kBAAkB,aAAa,MAAM,2SAA2S,oEAAoE,aAAa;AACpa,0BAA0B,KAAK,sBAAsB,gFAAgF,oBAAoB,mGAAmG,YAAY,uBAAuB,kBAAkB,sBAAsB,mBAAmB,mBAAmB,mBAAmB,6BAA6B,kBAAkB,kBAAkB,kBAAkB,IAAI,gBAAgB,IAAI,IAAI,EAAE;AACjf,kCAAkC,gBAAgB,IAAI,IAAI,EAAE,iCAAiC,gBAAgB,IAAI,IAAI,EAAE,qCAAqC,EAAE,GAAG,KAAwB,mBAAmB,SAAiF,GAAG;;;;;;;ACPhS;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;;AAEA;;AAEA,mBAAmB;AACnB;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;;;;;;AC5CA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB;AACpB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA,sBAAsB;AACtB;AACA,qBAAqB;AACrB;AACA,qBAAqB;AACrB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;;AAEA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,gBAAgB;AAChB;AACA,YAAY;AACZ;AACA;;AAEA;AACA;AACA,SAAS,eAAe;AACxB;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,2CAA2C,eAAe;AAC1D;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;;;;;;;ACjIA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;;;;;;ACjBA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wCAAwC;AACxC,uCAAuC,QAAQ,IAAI,QAAQ;AAC3D;AACA;AACA;;AAEA;AACA,WAAW,wBAAwB;AACnC,WAAW,wBAAwB;;AAEnC;AACA;AACA;AACA;AACA;AACA;;AAEA","file":"rote-0.6.0.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 17);\n","/**\n * This code is an implementation of Alea algorithm; (C) 2010 Johannes Baagøe.\n * Alea is licensed according to the http://en.wikipedia.org/wiki/MIT_License.\n */\nconst FRAC = 2.3283064365386963e-10; /* 2^-32 */\nclass RNG {\n    constructor() {\n        this._seed = 0;\n        this._s0 = 0;\n        this._s1 = 0;\n        this._s2 = 0;\n        this._c = 0;\n    }\n    getSeed() { return this._seed; }\n    /**\n     * Seed the number generator\n     */\n    setSeed(seed) {\n        seed = (seed < 1 ? 1 / seed : seed);\n        this._seed = seed;\n        this._s0 = (seed >>> 0) * FRAC;\n        seed = (seed * 69069 + 1) >>> 0;\n        this._s1 = seed * FRAC;\n        seed = (seed * 69069 + 1) >>> 0;\n        this._s2 = seed * FRAC;\n        this._c = 1;\n        return this;\n    }\n    /**\n     * @returns Pseudorandom value [0,1), uniformly distributed\n     */\n    getUniform() {\n        let t = 2091639 * this._s0 + this._c * FRAC;\n        this._s0 = this._s1;\n        this._s1 = this._s2;\n        this._c = t | 0;\n        this._s2 = t - this._c;\n        return this._s2;\n    }\n    /**\n     * @param lowerBound The lower end of the range to return a value from, inclusive\n     * @param upperBound The upper end of the range to return a value from, inclusive\n     * @returns Pseudorandom value [lowerBound, upperBound], using ROT.RNG.getUniform() to distribute the value\n     */\n    getUniformInt(lowerBound, upperBound) {\n        let max = Math.max(lowerBound, upperBound);\n        let min = Math.min(lowerBound, upperBound);\n        return Math.floor(this.getUniform() * (max - min + 1)) + min;\n    }\n    /**\n     * @param mean Mean value\n     * @param stddev Standard deviation. ~95% of the absolute values will be lower than 2*stddev.\n     * @returns A normally distributed pseudorandom value\n     */\n    getNormal(mean = 0, stddev = 1) {\n        let u, v, r;\n        do {\n            u = 2 * this.getUniform() - 1;\n            v = 2 * this.getUniform() - 1;\n            r = u * u + v * v;\n        } while (r > 1 || r == 0);\n        let gauss = u * Math.sqrt(-2 * Math.log(r) / r);\n        return mean + gauss * stddev;\n    }\n    /**\n     * @returns Pseudorandom value [1,100] inclusive, uniformly distributed\n     */\n    getPercentage() {\n        return 1 + Math.floor(this.getUniform() * 100);\n    }\n    /**\n     * @returns Randomly picked item, null when length=0\n     */\n    getItem(array) {\n        if (!array.length) {\n            return null;\n        }\n        return array[Math.floor(this.getUniform() * array.length)];\n    }\n    /**\n     * @returns New array with randomized items\n     */\n    shuffle(array) {\n        let result = [];\n        let clone = array.slice();\n        while (clone.length) {\n            let index = clone.indexOf(this.getItem(clone));\n            result.push(clone.splice(index, 1)[0]);\n        }\n        return result;\n    }\n    /**\n     * @param data key=whatever, value=weight (relative probability)\n     * @returns whatever\n     */\n    getWeightedValue(data) {\n        let total = 0;\n        for (let id in data) {\n            total += data[id];\n        }\n        let random = this.getUniform() * total;\n        let id, part = 0;\n        for (id in data) {\n            part += data[id];\n            if (random < part) {\n                return id;\n            }\n        }\n        // If by some floating-point annoyance we have\n        // random >= total, just return the last id.\n        return id;\n    }\n    /**\n     * Get RNG state. Useful for storing the state and re-setting it via setState.\n     * @returns Internal state\n     */\n    getState() { return [this._s0, this._s1, this._s2, this._c]; }\n    /**\n     * Set a previously retrieved state.\n     */\n    setState(state) {\n        this._s0 = state[0];\n        this._s1 = state[1];\n        this._s2 = state[2];\n        this._c = state[3];\n        return this;\n    }\n    /**\n     * Returns a cloned RNG\n     */\n    clone() {\n        let clone = new RNG();\n        return clone.setState(this.getState());\n    }\n}\nexport default new RNG().setSeed(Date.now());\n","/**\n * Always positive modulus\n * @param x Operand\n * @param n Modulus\n * @returns x modulo n\n */\nexport function mod(x, n) {\n    return (x % n + n) % n;\n}\nexport function clamp(val, min = 0, max = 1) {\n    if (val < min)\n        return min;\n    if (val > max)\n        return max;\n    return val;\n}\nexport function capitalize(string) {\n    return string.charAt(0).toUpperCase() + string.substring(1);\n}\n/**\n * Format a string in a flexible way. Scans for %s strings and replaces them with arguments. List of patterns is modifiable via String.format.map.\n * @param {string} template\n * @param {any} [argv]\n */\nexport function format(template, ...args) {\n    let map = format.map;\n    let replacer = function (match, group1, group2, index) {\n        if (template.charAt(index - 1) == \"%\") {\n            return match.substring(1);\n        }\n        if (!args.length) {\n            return match;\n        }\n        let obj = args[0];\n        let group = group1 || group2;\n        let parts = group.split(\",\");\n        let name = parts.shift() || \"\";\n        let method = map[name.toLowerCase()];\n        if (!method) {\n            return match;\n        }\n        obj = args.shift();\n        let replaced = obj[method].apply(obj, parts);\n        let first = name.charAt(0);\n        if (first != first.toLowerCase()) {\n            replaced = capitalize(replaced);\n        }\n        return replaced;\n    };\n    return template.replace(/%(?:([a-z]+)|(?:{([^}]+)}))/gi, replacer);\n}\nformat.map = {\n    \"s\": \"toString\"\n};\n","import { clamp } from \"./util.js\";\nimport RNG from \"./rng.js\";\nexport function fromString(str) {\n    let cached, r;\n    if (str in CACHE) {\n        cached = CACHE[str];\n    }\n    else {\n        if (str.charAt(0) == \"#\") { // hex rgb\n            let matched = str.match(/[0-9a-f]/gi) || [];\n            let values = matched.map((x) => parseInt(x, 16));\n            if (values.length == 3) {\n                cached = values.map((x) => x * 17);\n            }\n            else {\n                for (let i = 0; i < 3; i++) {\n                    values[i + 1] += 16 * values[i];\n                    values.splice(i, 1);\n                }\n                cached = values;\n            }\n        }\n        else if ((r = str.match(/rgb\\(([0-9, ]+)\\)/i))) { // decimal rgb\n            cached = r[1].split(/\\s*,\\s*/).map((x) => parseInt(x));\n        }\n        else { // html name\n            cached = [0, 0, 0];\n        }\n        CACHE[str] = cached;\n    }\n    return cached.slice();\n}\n/**\n * Add two or more colors\n */\nexport function add(color1, ...colors) {\n    let result = color1.slice();\n    for (let i = 0; i < 3; i++) {\n        for (let j = 0; j < colors.length; j++) {\n            result[i] += colors[j][i];\n        }\n    }\n    return result;\n}\n/**\n * Add two or more colors, MODIFIES FIRST ARGUMENT\n */\nexport function add_(color1, ...colors) {\n    for (let i = 0; i < 3; i++) {\n        for (let j = 0; j < colors.length; j++) {\n            color1[i] += colors[j][i];\n        }\n    }\n    return color1;\n}\n/**\n * Multiply (mix) two or more colors\n */\nexport function multiply(color1, ...colors) {\n    let result = color1.slice();\n    for (let i = 0; i < 3; i++) {\n        for (let j = 0; j < colors.length; j++) {\n            result[i] *= colors[j][i] / 255;\n        }\n        result[i] = Math.round(result[i]);\n    }\n    return result;\n}\n/**\n * Multiply (mix) two or more colors, MODIFIES FIRST ARGUMENT\n */\nexport function multiply_(color1, ...colors) {\n    for (let i = 0; i < 3; i++) {\n        for (let j = 0; j < colors.length; j++) {\n            color1[i] *= colors[j][i] / 255;\n        }\n        color1[i] = Math.round(color1[i]);\n    }\n    return color1;\n}\n/**\n * Interpolate (blend) two colors with a given factor\n */\nexport function interpolate(color1, color2, factor = 0.5) {\n    let result = color1.slice();\n    for (let i = 0; i < 3; i++) {\n        result[i] = Math.round(result[i] + factor * (color2[i] - color1[i]));\n    }\n    return result;\n}\nexport const lerp = interpolate;\n/**\n * Interpolate (blend) two colors with a given factor in HSL mode\n */\nexport function interpolateHSL(color1, color2, factor = 0.5) {\n    let hsl1 = rgb2hsl(color1);\n    let hsl2 = rgb2hsl(color2);\n    for (let i = 0; i < 3; i++) {\n        hsl1[i] += factor * (hsl2[i] - hsl1[i]);\n    }\n    return hsl2rgb(hsl1);\n}\nexport const lerpHSL = interpolateHSL;\n/**\n * Create a new random color based on this one\n * @param color\n * @param diff Set of standard deviations\n */\nexport function randomize(color, diff) {\n    if (!(diff instanceof Array)) {\n        diff = Math.round(RNG.getNormal(0, diff));\n    }\n    let result = color.slice();\n    for (let i = 0; i < 3; i++) {\n        result[i] += (diff instanceof Array ? Math.round(RNG.getNormal(0, diff[i])) : diff);\n    }\n    return result;\n}\n/**\n * Converts an RGB color value to HSL. Expects 0..255 inputs, produces 0..1 outputs.\n */\nexport function rgb2hsl(color) {\n    let r = color[0] / 255;\n    let g = color[1] / 255;\n    let b = color[2] / 255;\n    let max = Math.max(r, g, b), min = Math.min(r, g, b);\n    let h = 0, s, l = (max + min) / 2;\n    if (max == min) {\n        s = 0; // achromatic\n    }\n    else {\n        let d = max - min;\n        s = (l > 0.5 ? d / (2 - max - min) : d / (max + min));\n        switch (max) {\n            case r:\n                h = (g - b) / d + (g < b ? 6 : 0);\n                break;\n            case g:\n                h = (b - r) / d + 2;\n                break;\n            case b:\n                h = (r - g) / d + 4;\n                break;\n        }\n        h /= 6;\n    }\n    return [h, s, l];\n}\nfunction hue2rgb(p, q, t) {\n    if (t < 0)\n        t += 1;\n    if (t > 1)\n        t -= 1;\n    if (t < 1 / 6)\n        return p + (q - p) * 6 * t;\n    if (t < 1 / 2)\n        return q;\n    if (t < 2 / 3)\n        return p + (q - p) * (2 / 3 - t) * 6;\n    return p;\n}\n/**\n * Converts an HSL color value to RGB. Expects 0..1 inputs, produces 0..255 outputs.\n */\nexport function hsl2rgb(color) {\n    let l = color[2];\n    if (color[1] == 0) {\n        l = Math.round(l * 255);\n        return [l, l, l];\n    }\n    else {\n        let s = color[1];\n        let q = (l < 0.5 ? l * (1 + s) : l + s - l * s);\n        let p = 2 * l - q;\n        let r = hue2rgb(p, q, color[0] + 1 / 3);\n        let g = hue2rgb(p, q, color[0]);\n        let b = hue2rgb(p, q, color[0] - 1 / 3);\n        return [Math.round(r * 255), Math.round(g * 255), Math.round(b * 255)];\n    }\n}\nexport function toRGB(color) {\n    let clamped = color.map(x => clamp(x, 0, 255));\n    return `rgb(${clamped.join(\",\")})`;\n}\nexport function toHex(color) {\n    let clamped = color.map(x => clamp(x, 0, 255).toString(16).padStart(2, \"0\"));\n    return `#${clamped.join(\"\")}`;\n}\nconst CACHE = {\n    \"black\": [0, 0, 0],\n    \"navy\": [0, 0, 128],\n    \"darkblue\": [0, 0, 139],\n    \"mediumblue\": [0, 0, 205],\n    \"blue\": [0, 0, 255],\n    \"darkgreen\": [0, 100, 0],\n    \"green\": [0, 128, 0],\n    \"teal\": [0, 128, 128],\n    \"darkcyan\": [0, 139, 139],\n    \"deepskyblue\": [0, 191, 255],\n    \"darkturquoise\": [0, 206, 209],\n    \"mediumspringgreen\": [0, 250, 154],\n    \"lime\": [0, 255, 0],\n    \"springgreen\": [0, 255, 127],\n    \"aqua\": [0, 255, 255],\n    \"cyan\": [0, 255, 255],\n    \"midnightblue\": [25, 25, 112],\n    \"dodgerblue\": [30, 144, 255],\n    \"forestgreen\": [34, 139, 34],\n    \"seagreen\": [46, 139, 87],\n    \"darkslategray\": [47, 79, 79],\n    \"darkslategrey\": [47, 79, 79],\n    \"limegreen\": [50, 205, 50],\n    \"mediumseagreen\": [60, 179, 113],\n    \"turquoise\": [64, 224, 208],\n    \"royalblue\": [65, 105, 225],\n    \"steelblue\": [70, 130, 180],\n    \"darkslateblue\": [72, 61, 139],\n    \"mediumturquoise\": [72, 209, 204],\n    \"indigo\": [75, 0, 130],\n    \"darkolivegreen\": [85, 107, 47],\n    \"cadetblue\": [95, 158, 160],\n    \"cornflowerblue\": [100, 149, 237],\n    \"mediumaquamarine\": [102, 205, 170],\n    \"dimgray\": [105, 105, 105],\n    \"dimgrey\": [105, 105, 105],\n    \"slateblue\": [106, 90, 205],\n    \"olivedrab\": [107, 142, 35],\n    \"slategray\": [112, 128, 144],\n    \"slategrey\": [112, 128, 144],\n    \"lightslategray\": [119, 136, 153],\n    \"lightslategrey\": [119, 136, 153],\n    \"mediumslateblue\": [123, 104, 238],\n    \"lawngreen\": [124, 252, 0],\n    \"chartreuse\": [127, 255, 0],\n    \"aquamarine\": [127, 255, 212],\n    \"maroon\": [128, 0, 0],\n    \"purple\": [128, 0, 128],\n    \"olive\": [128, 128, 0],\n    \"gray\": [128, 128, 128],\n    \"grey\": [128, 128, 128],\n    \"skyblue\": [135, 206, 235],\n    \"lightskyblue\": [135, 206, 250],\n    \"blueviolet\": [138, 43, 226],\n    \"darkred\": [139, 0, 0],\n    \"darkmagenta\": [139, 0, 139],\n    \"saddlebrown\": [139, 69, 19],\n    \"darkseagreen\": [143, 188, 143],\n    \"lightgreen\": [144, 238, 144],\n    \"mediumpurple\": [147, 112, 216],\n    \"darkviolet\": [148, 0, 211],\n    \"palegreen\": [152, 251, 152],\n    \"darkorchid\": [153, 50, 204],\n    \"yellowgreen\": [154, 205, 50],\n    \"sienna\": [160, 82, 45],\n    \"brown\": [165, 42, 42],\n    \"darkgray\": [169, 169, 169],\n    \"darkgrey\": [169, 169, 169],\n    \"lightblue\": [173, 216, 230],\n    \"greenyellow\": [173, 255, 47],\n    \"paleturquoise\": [175, 238, 238],\n    \"lightsteelblue\": [176, 196, 222],\n    \"powderblue\": [176, 224, 230],\n    \"firebrick\": [178, 34, 34],\n    \"darkgoldenrod\": [184, 134, 11],\n    \"mediumorchid\": [186, 85, 211],\n    \"rosybrown\": [188, 143, 143],\n    \"darkkhaki\": [189, 183, 107],\n    \"silver\": [192, 192, 192],\n    \"mediumvioletred\": [199, 21, 133],\n    \"indianred\": [205, 92, 92],\n    \"peru\": [205, 133, 63],\n    \"chocolate\": [210, 105, 30],\n    \"tan\": [210, 180, 140],\n    \"lightgray\": [211, 211, 211],\n    \"lightgrey\": [211, 211, 211],\n    \"palevioletred\": [216, 112, 147],\n    \"thistle\": [216, 191, 216],\n    \"orchid\": [218, 112, 214],\n    \"goldenrod\": [218, 165, 32],\n    \"crimson\": [220, 20, 60],\n    \"gainsboro\": [220, 220, 220],\n    \"plum\": [221, 160, 221],\n    \"burlywood\": [222, 184, 135],\n    \"lightcyan\": [224, 255, 255],\n    \"lavender\": [230, 230, 250],\n    \"darksalmon\": [233, 150, 122],\n    \"violet\": [238, 130, 238],\n    \"palegoldenrod\": [238, 232, 170],\n    \"lightcoral\": [240, 128, 128],\n    \"khaki\": [240, 230, 140],\n    \"aliceblue\": [240, 248, 255],\n    \"honeydew\": [240, 255, 240],\n    \"azure\": [240, 255, 255],\n    \"sandybrown\": [244, 164, 96],\n    \"wheat\": [245, 222, 179],\n    \"beige\": [245, 245, 220],\n    \"whitesmoke\": [245, 245, 245],\n    \"mintcream\": [245, 255, 250],\n    \"ghostwhite\": [248, 248, 255],\n    \"salmon\": [250, 128, 114],\n    \"antiquewhite\": [250, 235, 215],\n    \"linen\": [250, 240, 230],\n    \"lightgoldenrodyellow\": [250, 250, 210],\n    \"oldlace\": [253, 245, 230],\n    \"red\": [255, 0, 0],\n    \"fuchsia\": [255, 0, 255],\n    \"magenta\": [255, 0, 255],\n    \"deeppink\": [255, 20, 147],\n    \"orangered\": [255, 69, 0],\n    \"tomato\": [255, 99, 71],\n    \"hotpink\": [255, 105, 180],\n    \"coral\": [255, 127, 80],\n    \"darkorange\": [255, 140, 0],\n    \"lightsalmon\": [255, 160, 122],\n    \"orange\": [255, 165, 0],\n    \"lightpink\": [255, 182, 193],\n    \"pink\": [255, 192, 203],\n    \"gold\": [255, 215, 0],\n    \"peachpuff\": [255, 218, 185],\n    \"navajowhite\": [255, 222, 173],\n    \"moccasin\": [255, 228, 181],\n    \"bisque\": [255, 228, 196],\n    \"mistyrose\": [255, 228, 225],\n    \"blanchedalmond\": [255, 235, 205],\n    \"papayawhip\": [255, 239, 213],\n    \"lavenderblush\": [255, 240, 245],\n    \"seashell\": [255, 245, 238],\n    \"cornsilk\": [255, 248, 220],\n    \"lemonchiffon\": [255, 250, 205],\n    \"floralwhite\": [255, 250, 240],\n    \"snow\": [255, 250, 250],\n    \"yellow\": [255, 255, 0],\n    \"lightyellow\": [255, 255, 224],\n    \"ivory\": [255, 255, 240],\n    \"white\": [255, 255, 255]\n};\n","import Backend from \"./backend.js\";\nexport default class Canvas extends Backend {\n    constructor() {\n        super();\n        this._ctx = document.createElement(\"canvas\").getContext(\"2d\");\n    }\n    schedule(cb) { requestAnimationFrame(cb); }\n    getContainer() { return this._ctx.canvas; }\n    setOptions(opts) {\n        super.setOptions(opts);\n        const style = (opts.fontStyle ? `${opts.fontStyle} ` : ``);\n        const font = `${style} ${opts.fontSize}px ${opts.fontFamily}`;\n        this._ctx.font = font;\n        this._updateSize();\n        this._ctx.font = font;\n        this._ctx.textAlign = \"center\";\n        this._ctx.textBaseline = \"middle\";\n    }\n    clear() {\n        this._ctx.fillStyle = this._options.bg;\n        this._ctx.fillRect(0, 0, this._ctx.canvas.width, this._ctx.canvas.height);\n    }\n    eventToPosition(x, y) {\n        let canvas = this._ctx.canvas;\n        let rect = canvas.getBoundingClientRect();\n        x -= rect.left;\n        y -= rect.top;\n        x *= canvas.width / rect.width;\n        y *= canvas.height / rect.height;\n        if (x < 0 || y < 0 || x >= canvas.width || y >= canvas.height) {\n            return [-1, -1];\n        }\n        return this._normalizedEventToPosition(x, y);\n    }\n}\n","import Canvas from \"./canvas.js\";\nimport { mod } from \"../util.js\";\n/**\n * @class Hexagonal backend\n * @private\n */\nexport default class Hex extends Canvas {\n    constructor() {\n        super();\n        this._spacingX = 0;\n        this._spacingY = 0;\n        this._hexSize = 0;\n    }\n    draw(data, clearBefore) {\n        let [x, y, ch, fg, bg] = data;\n        let px = [\n            (x + 1) * this._spacingX,\n            y * this._spacingY + this._hexSize\n        ];\n        if (this._options.transpose) {\n            px.reverse();\n        }\n        if (clearBefore) {\n            this._ctx.fillStyle = bg;\n            this._fill(px[0], px[1]);\n        }\n        if (!ch) {\n            return;\n        }\n        this._ctx.fillStyle = fg;\n        let chars = [].concat(ch);\n        for (let i = 0; i < chars.length; i++) {\n            this._ctx.fillText(chars[i], px[0], Math.ceil(px[1]));\n        }\n    }\n    computeSize(availWidth, availHeight) {\n        if (this._options.transpose) {\n            availWidth += availHeight;\n            availHeight = availWidth - availHeight;\n            availWidth -= availHeight;\n        }\n        let width = Math.floor(availWidth / this._spacingX) - 1;\n        let height = Math.floor((availHeight - 2 * this._hexSize) / this._spacingY + 1);\n        return [width, height];\n    }\n    computeFontSize(availWidth, availHeight) {\n        if (this._options.transpose) {\n            availWidth += availHeight;\n            availHeight = availWidth - availHeight;\n            availWidth -= availHeight;\n        }\n        let hexSizeWidth = 2 * availWidth / ((this._options.width + 1) * Math.sqrt(3)) - 1;\n        let hexSizeHeight = availHeight / (2 + 1.5 * (this._options.height - 1));\n        let hexSize = Math.min(hexSizeWidth, hexSizeHeight);\n        // compute char ratio\n        let oldFont = this._ctx.font;\n        this._ctx.font = \"100px \" + this._options.fontFamily;\n        let width = Math.ceil(this._ctx.measureText(\"W\").width);\n        this._ctx.font = oldFont;\n        let ratio = width / 100;\n        hexSize = Math.floor(hexSize) + 1; // closest larger hexSize\n        // FIXME char size computation does not respect transposed hexes\n        let fontSize = 2 * hexSize / (this._options.spacing * (1 + ratio / Math.sqrt(3)));\n        // closest smaller fontSize\n        return Math.ceil(fontSize) - 1;\n    }\n    _normalizedEventToPosition(x, y) {\n        let nodeSize;\n        if (this._options.transpose) {\n            x += y;\n            y = x - y;\n            x -= y;\n            nodeSize = this._ctx.canvas.width;\n        }\n        else {\n            nodeSize = this._ctx.canvas.height;\n        }\n        let size = nodeSize / this._options.height;\n        y = Math.floor(y / size);\n        if (mod(y, 2)) { /* odd row */\n            x -= this._spacingX;\n            x = 1 + 2 * Math.floor(x / (2 * this._spacingX));\n        }\n        else {\n            x = 2 * Math.floor(x / (2 * this._spacingX));\n        }\n        return [x, y];\n    }\n    /**\n     * Arguments are pixel values. If \"transposed\" mode is enabled, then these two are already swapped.\n     */\n    _fill(cx, cy) {\n        let a = this._hexSize;\n        let b = this._options.border;\n        const ctx = this._ctx;\n        ctx.beginPath();\n        if (this._options.transpose) {\n            ctx.moveTo(cx - a + b, cy);\n            ctx.lineTo(cx - a / 2 + b, cy + this._spacingX - b);\n            ctx.lineTo(cx + a / 2 - b, cy + this._spacingX - b);\n            ctx.lineTo(cx + a - b, cy);\n            ctx.lineTo(cx + a / 2 - b, cy - this._spacingX + b);\n            ctx.lineTo(cx - a / 2 + b, cy - this._spacingX + b);\n            ctx.lineTo(cx - a + b, cy);\n        }\n        else {\n            ctx.moveTo(cx, cy - a + b);\n            ctx.lineTo(cx + this._spacingX - b, cy - a / 2 + b);\n            ctx.lineTo(cx + this._spacingX - b, cy + a / 2 - b);\n            ctx.lineTo(cx, cy + a - b);\n            ctx.lineTo(cx - this._spacingX + b, cy + a / 2 - b);\n            ctx.lineTo(cx - this._spacingX + b, cy - a / 2 + b);\n            ctx.lineTo(cx, cy - a + b);\n        }\n        ctx.fill();\n    }\n    _updateSize() {\n        const opts = this._options;\n        const charWidth = Math.ceil(this._ctx.measureText(\"W\").width);\n        this._hexSize = Math.floor(opts.spacing * (opts.fontSize + charWidth / Math.sqrt(3)) / 2);\n        this._spacingX = this._hexSize * Math.sqrt(3) / 2;\n        this._spacingY = this._hexSize * 1.5;\n        let xprop;\n        let yprop;\n        if (opts.transpose) {\n            xprop = \"height\";\n            yprop = \"width\";\n        }\n        else {\n            xprop = \"width\";\n            yprop = \"height\";\n        }\n        this._ctx.canvas[xprop] = Math.ceil((opts.width + 1) * this._spacingX);\n        this._ctx.canvas[yprop] = Math.ceil((opts.height - 1) * this._spacingY + 2 * this._hexSize);\n    }\n}\n","import Canvas from \"./canvas.js\";\n/**\n * @class Rectangular backend\n * @private\n */\nexport default class Rect extends Canvas {\n    constructor() {\n        super();\n        this._spacingX = 0;\n        this._spacingY = 0;\n        this._canvasCache = {};\n    }\n    setOptions(options) {\n        super.setOptions(options);\n        this._canvasCache = {};\n    }\n    draw(data, clearBefore) {\n        if (Rect.cache) {\n            this._drawWithCache(data);\n        }\n        else {\n            this._drawNoCache(data, clearBefore);\n        }\n    }\n    _drawWithCache(data) {\n        let [x, y, ch, fg, bg] = data;\n        let hash = \"\" + ch + fg + bg;\n        let canvas;\n        if (hash in this._canvasCache) {\n            canvas = this._canvasCache[hash];\n        }\n        else {\n            let b = this._options.border;\n            canvas = document.createElement(\"canvas\");\n            let ctx = canvas.getContext(\"2d\");\n            canvas.width = this._spacingX;\n            canvas.height = this._spacingY;\n            ctx.fillStyle = bg;\n            ctx.fillRect(b, b, canvas.width - b, canvas.height - b);\n            if (ch) {\n                ctx.fillStyle = fg;\n                ctx.font = this._ctx.font;\n                ctx.textAlign = \"center\";\n                ctx.textBaseline = \"middle\";\n                let chars = [].concat(ch);\n                for (let i = 0; i < chars.length; i++) {\n                    ctx.fillText(chars[i], this._spacingX / 2, Math.ceil(this._spacingY / 2));\n                }\n            }\n            this._canvasCache[hash] = canvas;\n        }\n        this._ctx.drawImage(canvas, x * this._spacingX, y * this._spacingY);\n    }\n    _drawNoCache(data, clearBefore) {\n        let [x, y, ch, fg, bg] = data;\n        if (clearBefore) {\n            let b = this._options.border;\n            this._ctx.fillStyle = bg;\n            this._ctx.fillRect(x * this._spacingX + b, y * this._spacingY + b, this._spacingX - b, this._spacingY - b);\n        }\n        if (!ch) {\n            return;\n        }\n        this._ctx.fillStyle = fg;\n        let chars = [].concat(ch);\n        for (let i = 0; i < chars.length; i++) {\n            this._ctx.fillText(chars[i], (x + 0.5) * this._spacingX, Math.ceil((y + 0.5) * this._spacingY));\n        }\n    }\n    computeSize(availWidth, availHeight) {\n        let width = Math.floor(availWidth / this._spacingX);\n        let height = Math.floor(availHeight / this._spacingY);\n        return [width, height];\n    }\n    computeFontSize(availWidth, availHeight) {\n        let boxWidth = Math.floor(availWidth / this._options.width);\n        let boxHeight = Math.floor(availHeight / this._options.height);\n        /* compute char ratio */\n        let oldFont = this._ctx.font;\n        this._ctx.font = \"100px \" + this._options.fontFamily;\n        let width = Math.ceil(this._ctx.measureText(\"W\").width);\n        this._ctx.font = oldFont;\n        let ratio = width / 100;\n        let widthFraction = ratio * boxHeight / boxWidth;\n        if (widthFraction > 1) { /* too wide with current aspect ratio */\n            boxHeight = Math.floor(boxHeight / widthFraction);\n        }\n        return Math.floor(boxHeight / this._options.spacing);\n    }\n    _normalizedEventToPosition(x, y) {\n        return [Math.floor(x / this._spacingX), Math.floor(y / this._spacingY)];\n    }\n    _updateSize() {\n        const opts = this._options;\n        const charWidth = Math.ceil(this._ctx.measureText(\"W\").width);\n        this._spacingX = Math.ceil(opts.spacing * charWidth);\n        this._spacingY = Math.ceil(opts.spacing * opts.fontSize);\n        if (opts.forceSquareRatio) {\n            this._spacingX = this._spacingY = Math.max(this._spacingX, this._spacingY);\n        }\n        this._ctx.canvas.width = opts.width * this._spacingX;\n        this._ctx.canvas.height = opts.height * this._spacingY;\n    }\n}\nRect.cache = false;\n","import Canvas from \"./canvas.js\";\n/**\n * @class Tile backend\n * @private\n */\nexport default class Tile extends Canvas {\n    constructor() {\n        super();\n        this._colorCanvas = document.createElement(\"canvas\");\n    }\n    draw(data, clearBefore) {\n        let [x, y, ch, fg, bg] = data;\n        let tileWidth = this._options.tileWidth;\n        let tileHeight = this._options.tileHeight;\n        if (clearBefore) {\n            if (this._options.tileColorize) {\n                this._ctx.clearRect(x * tileWidth, y * tileHeight, tileWidth, tileHeight);\n            }\n            else {\n                this._ctx.fillStyle = bg;\n                this._ctx.fillRect(x * tileWidth, y * tileHeight, tileWidth, tileHeight);\n            }\n        }\n        if (!ch) {\n            return;\n        }\n        let chars = [].concat(ch);\n        let fgs = [].concat(fg);\n        let bgs = [].concat(bg);\n        for (let i = 0; i < chars.length; i++) {\n            let tile = this._options.tileMap[chars[i]];\n            if (!tile) {\n                throw new Error(`Char \"${chars[i]}\" not found in tileMap`);\n            }\n            if (this._options.tileColorize) { // apply colorization\n                let canvas = this._colorCanvas;\n                let context = canvas.getContext(\"2d\");\n                context.globalCompositeOperation = \"source-over\";\n                context.clearRect(0, 0, tileWidth, tileHeight);\n                let fg = fgs[i];\n                let bg = bgs[i];\n                context.drawImage(this._options.tileSet, tile[0], tile[1], tileWidth, tileHeight, 0, 0, tileWidth, tileHeight);\n                if (fg != \"transparent\") {\n                    context.fillStyle = fg;\n                    context.globalCompositeOperation = \"source-atop\";\n                    context.fillRect(0, 0, tileWidth, tileHeight);\n                }\n                if (bg != \"transparent\") {\n                    context.fillStyle = bg;\n                    context.globalCompositeOperation = \"destination-over\";\n                    context.fillRect(0, 0, tileWidth, tileHeight);\n                }\n                this._ctx.drawImage(canvas, x * tileWidth, y * tileHeight, tileWidth, tileHeight);\n            }\n            else { // no colorizing, easy\n                this._ctx.drawImage(this._options.tileSet, tile[0], tile[1], tileWidth, tileHeight, x * tileWidth, y * tileHeight, tileWidth, tileHeight);\n            }\n        }\n    }\n    computeSize(availWidth, availHeight) {\n        let width = Math.floor(availWidth / this._options.tileWidth);\n        let height = Math.floor(availHeight / this._options.tileHeight);\n        return [width, height];\n    }\n    computeFontSize() {\n        throw new Error(\"Tile backend does not understand font size\");\n    }\n    _normalizedEventToPosition(x, y) {\n        return [Math.floor(x / this._options.tileWidth), Math.floor(y / this._options.tileHeight)];\n    }\n    _updateSize() {\n        const opts = this._options;\n        this._ctx.canvas.width = opts.width * opts.tileWidth;\n        this._ctx.canvas.height = opts.height * opts.tileHeight;\n        this._colorCanvas.width = opts.tileWidth;\n        this._colorCanvas.height = opts.tileHeight;\n    }\n}\n","import Backend from \"./backend.js\";\nimport * as Color from \"../color.js\";\n/**\n * @class Tile backend\n * @private\n */\nexport default class TileGL extends Backend {\n    static isSupported() {\n        return !!document.createElement(\"canvas\").getContext(\"webgl2\", { preserveDrawingBuffer: true });\n    }\n    constructor() {\n        super();\n        this._uniforms = {};\n        try {\n            this._gl = this._initWebGL();\n        }\n        catch (e) {\n            alert(e.message);\n        }\n    }\n    schedule(cb) { requestAnimationFrame(cb); }\n    getContainer() { return this._gl.canvas; }\n    setOptions(opts) {\n        super.setOptions(opts);\n        this._updateSize();\n        let tileSet = this._options.tileSet;\n        if (tileSet && \"complete\" in tileSet && !tileSet.complete) {\n            tileSet.addEventListener(\"load\", () => this._updateTexture(tileSet));\n        }\n        else {\n            this._updateTexture(tileSet);\n        }\n    }\n    draw(data, clearBefore) {\n        const gl = this._gl;\n        const opts = this._options;\n        let [x, y, ch, fg, bg] = data;\n        let scissorY = gl.canvas.height - (y + 1) * opts.tileHeight;\n        gl.scissor(x * opts.tileWidth, scissorY, opts.tileWidth, opts.tileHeight);\n        if (clearBefore) {\n            if (opts.tileColorize) {\n                gl.clearColor(0, 0, 0, 0);\n            }\n            else {\n                gl.clearColor(...parseColor(bg));\n            }\n            gl.clear(gl.COLOR_BUFFER_BIT);\n        }\n        if (!ch) {\n            return;\n        }\n        let chars = [].concat(ch);\n        let bgs = [].concat(bg);\n        let fgs = [].concat(fg);\n        gl.uniform2fv(this._uniforms[\"targetPosRel\"], [x, y]);\n        for (let i = 0; i < chars.length; i++) {\n            let tile = this._options.tileMap[chars[i]];\n            if (!tile) {\n                throw new Error(`Char \"${chars[i]}\" not found in tileMap`);\n            }\n            gl.uniform1f(this._uniforms[\"colorize\"], opts.tileColorize ? 1 : 0);\n            gl.uniform2fv(this._uniforms[\"tilesetPosAbs\"], tile);\n            if (opts.tileColorize) {\n                gl.uniform4fv(this._uniforms[\"tint\"], parseColor(fgs[i]));\n                gl.uniform4fv(this._uniforms[\"bg\"], parseColor(bgs[i]));\n            }\n            gl.drawArrays(gl.TRIANGLE_STRIP, 0, 4);\n        }\n        /*\n        \n        \n                for (let i=0;i<chars.length;i++) {\n                    \n                    if (this._options.tileColorize) { // apply colorization\n                        let canvas = this._colorCanvas;\n                        let context = canvas.getContext(\"2d\") as CanvasRenderingContext2D;\n                        context.globalCompositeOperation = \"source-over\";\n                        context.clearRect(0, 0, tileWidth, tileHeight);\n        \n                        let fg = fgs[i];\n                        let bg = bgs[i];\n        \n                        context.drawImage(\n                            this._options.tileSet!,\n                            tile[0], tile[1], tileWidth, tileHeight,\n                            0, 0, tileWidth, tileHeight\n                        );\n        \n                        if (fg != \"transparent\") {\n                            context.fillStyle = fg;\n                            context.globalCompositeOperation = \"source-atop\";\n                            context.fillRect(0, 0, tileWidth, tileHeight);\n                        }\n        \n                        if (bg != \"transparent\") {\n                            context.fillStyle = bg;\n                            context.globalCompositeOperation = \"destination-over\";\n                            context.fillRect(0, 0, tileWidth, tileHeight);\n                        }\n        \n                        this._ctx.drawImage(canvas, x*tileWidth, y*tileHeight, tileWidth, tileHeight);\n                    } else { // no colorizing, easy\n                        this._ctx.drawImage(\n                            this._options.tileSet!,\n                            tile[0], tile[1], tileWidth, tileHeight,\n                            x*tileWidth, y*tileHeight, tileWidth, tileHeight\n                        );\n                    }\n                }\n        \n        */\n    }\n    clear() {\n        const gl = this._gl;\n        gl.clearColor(...parseColor(this._options.bg));\n        gl.scissor(0, 0, gl.canvas.width, gl.canvas.height);\n        gl.clear(gl.COLOR_BUFFER_BIT);\n    }\n    computeSize(availWidth, availHeight) {\n        let width = Math.floor(availWidth / this._options.tileWidth);\n        let height = Math.floor(availHeight / this._options.tileHeight);\n        return [width, height];\n    }\n    computeFontSize() {\n        throw new Error(\"Tile backend does not understand font size\");\n    }\n    eventToPosition(x, y) {\n        let canvas = this._gl.canvas;\n        let rect = canvas.getBoundingClientRect();\n        x -= rect.left;\n        y -= rect.top;\n        x *= canvas.width / rect.width;\n        y *= canvas.height / rect.height;\n        if (x < 0 || y < 0 || x >= canvas.width || y >= canvas.height) {\n            return [-1, -1];\n        }\n        return this._normalizedEventToPosition(x, y);\n    }\n    _initWebGL() {\n        let gl = document.createElement(\"canvas\").getContext(\"webgl2\", { preserveDrawingBuffer: true });\n        window.gl = gl;\n        let program = createProgram(gl, VS, FS);\n        gl.useProgram(program);\n        createQuad(gl);\n        UNIFORMS.forEach(name => this._uniforms[name] = gl.getUniformLocation(program, name));\n        this._program = program;\n        gl.enable(gl.BLEND);\n        gl.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA);\n        gl.enable(gl.SCISSOR_TEST);\n        return gl;\n    }\n    _normalizedEventToPosition(x, y) {\n        return [Math.floor(x / this._options.tileWidth), Math.floor(y / this._options.tileHeight)];\n    }\n    _updateSize() {\n        const gl = this._gl;\n        const opts = this._options;\n        const canvasSize = [opts.width * opts.tileWidth, opts.height * opts.tileHeight];\n        gl.canvas.width = canvasSize[0];\n        gl.canvas.height = canvasSize[1];\n        gl.viewport(0, 0, canvasSize[0], canvasSize[1]);\n        gl.uniform2fv(this._uniforms[\"tileSize\"], [opts.tileWidth, opts.tileHeight]);\n        gl.uniform2fv(this._uniforms[\"targetSize\"], canvasSize);\n    }\n    _updateTexture(tileSet) {\n        createTexture(this._gl, tileSet);\n    }\n}\nconst UNIFORMS = [\"targetPosRel\", \"tilesetPosAbs\", \"tileSize\", \"targetSize\", \"colorize\", \"bg\", \"tint\"];\nconst VS = `\n#version 300 es\n\nin vec2 tilePosRel;\nout vec2 tilesetPosPx;\n\nuniform vec2 tilesetPosAbs;\nuniform vec2 tileSize;\nuniform vec2 targetSize;\nuniform vec2 targetPosRel;\n\nvoid main() {\n\tvec2 targetPosPx = (targetPosRel + tilePosRel) * tileSize;\n\tvec2 targetPosNdc = ((targetPosPx / targetSize)-0.5)*2.0;\n\ttargetPosNdc.y *= -1.0;\n\n\tgl_Position = vec4(targetPosNdc, 0.0, 1.0);\n\ttilesetPosPx = tilesetPosAbs + tilePosRel * tileSize;\n}`.trim();\nconst FS = `\n#version 300 es\nprecision highp float;\n\nin vec2 tilesetPosPx;\nout vec4 fragColor;\nuniform sampler2D image;\nuniform bool colorize;\nuniform vec4 bg;\nuniform vec4 tint;\n\nvoid main() {\n\tfragColor = vec4(0, 0, 0, 1);\n\n\tvec4 texel = texelFetch(image, ivec2(tilesetPosPx), 0);\n\n\tif (colorize) {\n\t\ttexel.rgb = tint.a * tint.rgb + (1.0-tint.a) * texel.rgb;\n\t\tfragColor.rgb = texel.a*texel.rgb + (1.0-texel.a)*bg.rgb;\n\t\tfragColor.a = texel.a + (1.0-texel.a)*bg.a;\n\t} else {\n\t\tfragColor = texel;\n\t}\n}`.trim();\nfunction createProgram(gl, vss, fss) {\n    const vs = gl.createShader(gl.VERTEX_SHADER);\n    gl.shaderSource(vs, vss);\n    gl.compileShader(vs);\n    if (!gl.getShaderParameter(vs, gl.COMPILE_STATUS)) {\n        throw new Error(gl.getShaderInfoLog(vs) || \"\");\n    }\n    const fs = gl.createShader(gl.FRAGMENT_SHADER);\n    gl.shaderSource(fs, fss);\n    gl.compileShader(fs);\n    if (!gl.getShaderParameter(fs, gl.COMPILE_STATUS)) {\n        throw new Error(gl.getShaderInfoLog(fs) || \"\");\n    }\n    const p = gl.createProgram();\n    gl.attachShader(p, vs);\n    gl.attachShader(p, fs);\n    gl.linkProgram(p);\n    if (!gl.getProgramParameter(p, gl.LINK_STATUS)) {\n        throw new Error(gl.getProgramInfoLog(p) || \"\");\n    }\n    return p;\n}\nfunction createQuad(gl) {\n    const pos = new Float32Array([0, 0, 1, 0, 0, 1, 1, 1]);\n    const buf = gl.createBuffer();\n    gl.bindBuffer(gl.ARRAY_BUFFER, buf);\n    gl.bufferData(gl.ARRAY_BUFFER, pos, gl.STATIC_DRAW);\n    gl.enableVertexAttribArray(0);\n    gl.vertexAttribPointer(0, 2, gl.FLOAT, false, 0, 0);\n}\nfunction createTexture(gl, data) {\n    let t = gl.createTexture();\n    gl.bindTexture(gl.TEXTURE_2D, t);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT);\n    gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, 0);\n    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, data);\n    return t;\n}\nlet colorCache = {};\nfunction parseColor(color) {\n    if (!(color in colorCache)) {\n        let parsed;\n        if (color == \"transparent\") {\n            parsed = [0, 0, 0, 0];\n        }\n        else if (color.indexOf(\"rgba\") > -1) {\n            parsed = (color.match(/[\\d.]+/g) || []).map(Number);\n            for (let i = 0; i < 3; i++) {\n                parsed[i] = parsed[i] / 255;\n            }\n        }\n        else {\n            parsed = Color.fromString(color).map($ => $ / 255);\n            parsed.push(1);\n        }\n        colorCache[color] = parsed;\n    }\n    return colorCache[color];\n}\n","/**\n * @namespace\n * Contains text tokenization and breaking routines\n */\nconst RE_COLORS = /%([bc]){([^}]*)}/g;\n// token types\nexport const TYPE_TEXT = 0;\nexport const TYPE_NEWLINE = 1;\nexport const TYPE_FG = 2;\nexport const TYPE_BG = 3;\n/**\n * Measure size of a resulting text block\n */\nexport function measure(str, maxWidth) {\n    let result = { width: 0, height: 1 };\n    let tokens = tokenize(str, maxWidth);\n    let lineWidth = 0;\n    for (let i = 0; i < tokens.length; i++) {\n        let token = tokens[i];\n        switch (token.type) {\n            case TYPE_TEXT:\n                lineWidth += token.value.length;\n                break;\n            case TYPE_NEWLINE:\n                result.height++;\n                result.width = Math.max(result.width, lineWidth);\n                lineWidth = 0;\n                break;\n        }\n    }\n    result.width = Math.max(result.width, lineWidth);\n    return result;\n}\n/**\n * Convert string to a series of a formatting commands\n */\nexport function tokenize(str, maxWidth) {\n    let result = [];\n    /* first tokenization pass - split texts and color formatting commands */\n    let offset = 0;\n    str.replace(RE_COLORS, function (match, type, name, index) {\n        /* string before */\n        let part = str.substring(offset, index);\n        if (part.length) {\n            result.push({\n                type: TYPE_TEXT,\n                value: part\n            });\n        }\n        /* color command */\n        result.push({\n            type: (type == \"c\" ? TYPE_FG : TYPE_BG),\n            value: name.trim()\n        });\n        offset = index + match.length;\n        return \"\";\n    });\n    /* last remaining part */\n    let part = str.substring(offset);\n    if (part.length) {\n        result.push({\n            type: TYPE_TEXT,\n            value: part\n        });\n    }\n    return breakLines(result, maxWidth);\n}\n/* insert line breaks into first-pass tokenized data */\nfunction breakLines(tokens, maxWidth) {\n    if (!maxWidth) {\n        maxWidth = Infinity;\n    }\n    let i = 0;\n    let lineLength = 0;\n    let lastTokenWithSpace = -1;\n    while (i < tokens.length) { /* take all text tokens, remove space, apply linebreaks */\n        let token = tokens[i];\n        if (token.type == TYPE_NEWLINE) { /* reset */\n            lineLength = 0;\n            lastTokenWithSpace = -1;\n        }\n        if (token.type != TYPE_TEXT) { /* skip non-text tokens */\n            i++;\n            continue;\n        }\n        /* remove spaces at the beginning of line */\n        while (lineLength == 0 && token.value.charAt(0) == \" \") {\n            token.value = token.value.substring(1);\n        }\n        /* forced newline? insert two new tokens after this one */\n        let index = token.value.indexOf(\"\\n\");\n        if (index != -1) {\n            token.value = breakInsideToken(tokens, i, index, true);\n            /* if there are spaces at the end, we must remove them (we do not want the line too long) */\n            let arr = token.value.split(\"\");\n            while (arr.length && arr[arr.length - 1] == \" \") {\n                arr.pop();\n            }\n            token.value = arr.join(\"\");\n        }\n        /* token degenerated? */\n        if (!token.value.length) {\n            tokens.splice(i, 1);\n            continue;\n        }\n        if (lineLength + token.value.length > maxWidth) { /* line too long, find a suitable breaking spot */\n            /* is it possible to break within this token? */\n            let index = -1;\n            while (1) {\n                let nextIndex = token.value.indexOf(\" \", index + 1);\n                if (nextIndex == -1) {\n                    break;\n                }\n                if (lineLength + nextIndex > maxWidth) {\n                    break;\n                }\n                index = nextIndex;\n            }\n            if (index != -1) { /* break at space within this one */\n                token.value = breakInsideToken(tokens, i, index, true);\n            }\n            else if (lastTokenWithSpace != -1) { /* is there a previous token where a break can occur? */\n                let token = tokens[lastTokenWithSpace];\n                let breakIndex = token.value.lastIndexOf(\" \");\n                token.value = breakInsideToken(tokens, lastTokenWithSpace, breakIndex, true);\n                i = lastTokenWithSpace;\n            }\n            else { /* force break in this token */\n                token.value = breakInsideToken(tokens, i, maxWidth - lineLength, false);\n            }\n        }\n        else { /* line not long, continue */\n            lineLength += token.value.length;\n            if (token.value.indexOf(\" \") != -1) {\n                lastTokenWithSpace = i;\n            }\n        }\n        i++; /* advance to next token */\n    }\n    tokens.push({ type: TYPE_NEWLINE }); /* insert fake newline to fix the last text line */\n    /* remove trailing space from text tokens before newlines */\n    let lastTextToken = null;\n    for (let i = 0; i < tokens.length; i++) {\n        let token = tokens[i];\n        switch (token.type) {\n            case TYPE_TEXT:\n                lastTextToken = token;\n                break;\n            case TYPE_NEWLINE:\n                if (lastTextToken) { /* remove trailing space */\n                    let arr = lastTextToken.value.split(\"\");\n                    while (arr.length && arr[arr.length - 1] == \" \") {\n                        arr.pop();\n                    }\n                    lastTextToken.value = arr.join(\"\");\n                }\n                lastTextToken = null;\n                break;\n        }\n    }\n    tokens.pop(); /* remove fake token */\n    return tokens;\n}\n/**\n * Create new tokens and insert them into the stream\n * @param {object[]} tokens\n * @param {int} tokenIndex Token being processed\n * @param {int} breakIndex Index within current token's value\n * @param {bool} removeBreakChar Do we want to remove the breaking character?\n * @returns {string} remaining unbroken token value\n */\nfunction breakInsideToken(tokens, tokenIndex, breakIndex, removeBreakChar) {\n    let newBreakToken = {\n        type: TYPE_NEWLINE\n    };\n    let newTextToken = {\n        type: TYPE_TEXT,\n        value: tokens[tokenIndex].value.substring(breakIndex + (removeBreakChar ? 1 : 0))\n    };\n    tokens.splice(tokenIndex + 1, 0, newBreakToken, newTextToken);\n    return tokens[tokenIndex].value.substring(0, breakIndex);\n}\n","/** Default with for display and map generators */\nexport let DEFAULT_WIDTH = 80;\n/** Default height for display and map generators */\nexport let DEFAULT_HEIGHT = 25;\nexport const DIRS = {\n    4: [[0, -1], [1, 0], [0, 1], [-1, 0]],\n    8: [[0, -1], [1, -1], [1, 0], [1, 1], [0, 1], [-1, 1], [-1, 0], [-1, -1]],\n    6: [[-1, -1], [1, -1], [2, 0], [1, 1], [-1, 1], [-2, 0]]\n};\nexport const KEYS = {\n    /** Cancel key. */\n    VK_CANCEL: 3,\n    /** Help key. */\n    VK_HELP: 6,\n    /** Backspace key. */\n    VK_BACK_SPACE: 8,\n    /** Tab key. */\n    VK_TAB: 9,\n    /** 5 key on Numpad when NumLock is unlocked. Or on Mac, clear key which is positioned at NumLock key. */\n    VK_CLEAR: 12,\n    /** Return/enter key on the main keyboard. */\n    VK_RETURN: 13,\n    /** Reserved, but not used. */\n    VK_ENTER: 14,\n    /** Shift key. */\n    VK_SHIFT: 16,\n    /** Control key. */\n    VK_CONTROL: 17,\n    /** Alt (Option on Mac) key. */\n    VK_ALT: 18,\n    /** Pause key. */\n    VK_PAUSE: 19,\n    /** Caps lock. */\n    VK_CAPS_LOCK: 20,\n    /** Escape key. */\n    VK_ESCAPE: 27,\n    /** Space bar. */\n    VK_SPACE: 32,\n    /** Page Up key. */\n    VK_PAGE_UP: 33,\n    /** Page Down key. */\n    VK_PAGE_DOWN: 34,\n    /** End key. */\n    VK_END: 35,\n    /** Home key. */\n    VK_HOME: 36,\n    /** Left arrow. */\n    VK_LEFT: 37,\n    /** Up arrow. */\n    VK_UP: 38,\n    /** Right arrow. */\n    VK_RIGHT: 39,\n    /** Down arrow. */\n    VK_DOWN: 40,\n    /** Print Screen key. */\n    VK_PRINTSCREEN: 44,\n    /** Ins(ert) key. */\n    VK_INSERT: 45,\n    /** Del(ete) key. */\n    VK_DELETE: 46,\n    /***/\n    VK_0: 48,\n    /***/\n    VK_1: 49,\n    /***/\n    VK_2: 50,\n    /***/\n    VK_3: 51,\n    /***/\n    VK_4: 52,\n    /***/\n    VK_5: 53,\n    /***/\n    VK_6: 54,\n    /***/\n    VK_7: 55,\n    /***/\n    VK_8: 56,\n    /***/\n    VK_9: 57,\n    /** Colon (:) key. Requires Gecko 15.0 */\n    VK_COLON: 58,\n    /** Semicolon (;) key. */\n    VK_SEMICOLON: 59,\n    /** Less-than (<) key. Requires Gecko 15.0 */\n    VK_LESS_THAN: 60,\n    /** Equals (=) key. */\n    VK_EQUALS: 61,\n    /** Greater-than (>) key. Requires Gecko 15.0 */\n    VK_GREATER_THAN: 62,\n    /** Question mark (?) key. Requires Gecko 15.0 */\n    VK_QUESTION_MARK: 63,\n    /** Atmark (@) key. Requires Gecko 15.0 */\n    VK_AT: 64,\n    /***/\n    VK_A: 65,\n    /***/\n    VK_B: 66,\n    /***/\n    VK_C: 67,\n    /***/\n    VK_D: 68,\n    /***/\n    VK_E: 69,\n    /***/\n    VK_F: 70,\n    /***/\n    VK_G: 71,\n    /***/\n    VK_H: 72,\n    /***/\n    VK_I: 73,\n    /***/\n    VK_J: 74,\n    /***/\n    VK_K: 75,\n    /***/\n    VK_L: 76,\n    /***/\n    VK_M: 77,\n    /***/\n    VK_N: 78,\n    /***/\n    VK_O: 79,\n    /***/\n    VK_P: 80,\n    /***/\n    VK_Q: 81,\n    /***/\n    VK_R: 82,\n    /***/\n    VK_S: 83,\n    /***/\n    VK_T: 84,\n    /***/\n    VK_U: 85,\n    /***/\n    VK_V: 86,\n    /***/\n    VK_W: 87,\n    /***/\n    VK_X: 88,\n    /***/\n    VK_Y: 89,\n    /***/\n    VK_Z: 90,\n    /***/\n    VK_CONTEXT_MENU: 93,\n    /** 0 on the numeric keypad. */\n    VK_NUMPAD0: 96,\n    /** 1 on the numeric keypad. */\n    VK_NUMPAD1: 97,\n    /** 2 on the numeric keypad. */\n    VK_NUMPAD2: 98,\n    /** 3 on the numeric keypad. */\n    VK_NUMPAD3: 99,\n    /** 4 on the numeric keypad. */\n    VK_NUMPAD4: 100,\n    /** 5 on the numeric keypad. */\n    VK_NUMPAD5: 101,\n    /** 6 on the numeric keypad. */\n    VK_NUMPAD6: 102,\n    /** 7 on the numeric keypad. */\n    VK_NUMPAD7: 103,\n    /** 8 on the numeric keypad. */\n    VK_NUMPAD8: 104,\n    /** 9 on the numeric keypad. */\n    VK_NUMPAD9: 105,\n    /** * on the numeric keypad. */\n    VK_MULTIPLY: 106,\n    /** + on the numeric keypad. */\n    VK_ADD: 107,\n    /***/\n    VK_SEPARATOR: 108,\n    /** - on the numeric keypad. */\n    VK_SUBTRACT: 109,\n    /** Decimal point on the numeric keypad. */\n    VK_DECIMAL: 110,\n    /** / on the numeric keypad. */\n    VK_DIVIDE: 111,\n    /** F1 key. */\n    VK_F1: 112,\n    /** F2 key. */\n    VK_F2: 113,\n    /** F3 key. */\n    VK_F3: 114,\n    /** F4 key. */\n    VK_F4: 115,\n    /** F5 key. */\n    VK_F5: 116,\n    /** F6 key. */\n    VK_F6: 117,\n    /** F7 key. */\n    VK_F7: 118,\n    /** F8 key. */\n    VK_F8: 119,\n    /** F9 key. */\n    VK_F9: 120,\n    /** F10 key. */\n    VK_F10: 121,\n    /** F11 key. */\n    VK_F11: 122,\n    /** F12 key. */\n    VK_F12: 123,\n    /** F13 key. */\n    VK_F13: 124,\n    /** F14 key. */\n    VK_F14: 125,\n    /** F15 key. */\n    VK_F15: 126,\n    /** F16 key. */\n    VK_F16: 127,\n    /** F17 key. */\n    VK_F17: 128,\n    /** F18 key. */\n    VK_F18: 129,\n    /** F19 key. */\n    VK_F19: 130,\n    /** F20 key. */\n    VK_F20: 131,\n    /** F21 key. */\n    VK_F21: 132,\n    /** F22 key. */\n    VK_F22: 133,\n    /** F23 key. */\n    VK_F23: 134,\n    /** F24 key. */\n    VK_F24: 135,\n    /** Num Lock key. */\n    VK_NUM_LOCK: 144,\n    /** Scroll Lock key. */\n    VK_SCROLL_LOCK: 145,\n    /** Circumflex (^) key. Requires Gecko 15.0 */\n    VK_CIRCUMFLEX: 160,\n    /** Exclamation (!) key. Requires Gecko 15.0 */\n    VK_EXCLAMATION: 161,\n    /** Double quote () key. Requires Gecko 15.0 */\n    VK_DOUBLE_QUOTE: 162,\n    /** Hash (#) key. Requires Gecko 15.0 */\n    VK_HASH: 163,\n    /** Dollar sign ($) key. Requires Gecko 15.0 */\n    VK_DOLLAR: 164,\n    /** Percent (%) key. Requires Gecko 15.0 */\n    VK_PERCENT: 165,\n    /** Ampersand (&) key. Requires Gecko 15.0 */\n    VK_AMPERSAND: 166,\n    /** Underscore (_) key. Requires Gecko 15.0 */\n    VK_UNDERSCORE: 167,\n    /** Open parenthesis (() key. Requires Gecko 15.0 */\n    VK_OPEN_PAREN: 168,\n    /** Close parenthesis ()) key. Requires Gecko 15.0 */\n    VK_CLOSE_PAREN: 169,\n    /* Asterisk (*) key. Requires Gecko 15.0 */\n    VK_ASTERISK: 170,\n    /** Plus (+) key. Requires Gecko 15.0 */\n    VK_PLUS: 171,\n    /** Pipe (|) key. Requires Gecko 15.0 */\n    VK_PIPE: 172,\n    /** Hyphen-US/docs/Minus (-) key. Requires Gecko 15.0 */\n    VK_HYPHEN_MINUS: 173,\n    /** Open curly bracket ({) key. Requires Gecko 15.0 */\n    VK_OPEN_CURLY_BRACKET: 174,\n    /** Close curly bracket (}) key. Requires Gecko 15.0 */\n    VK_CLOSE_CURLY_BRACKET: 175,\n    /** Tilde (~) key. Requires Gecko 15.0 */\n    VK_TILDE: 176,\n    /** Comma (,) key. */\n    VK_COMMA: 188,\n    /** Period (.) key. */\n    VK_PERIOD: 190,\n    /** Slash (/) key. */\n    VK_SLASH: 191,\n    /** Back tick (`) key. */\n    VK_BACK_QUOTE: 192,\n    /** Open square bracket ([) key. */\n    VK_OPEN_BRACKET: 219,\n    /** Back slash (\\) key. */\n    VK_BACK_SLASH: 220,\n    /** Close square bracket (]) key. */\n    VK_CLOSE_BRACKET: 221,\n    /** Quote (''') key. */\n    VK_QUOTE: 222,\n    /** Meta key on Linux, Command key on Mac. */\n    VK_META: 224,\n    /** AltGr key on Linux. Requires Gecko 15.0 */\n    VK_ALTGR: 225,\n    /** Windows logo key on Windows. Or Super or Hyper key on Linux. Requires Gecko 15.0 */\n    VK_WIN: 91,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_KANA: 21,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_HANGUL: 21,\n    /** 英数 key on Japanese Mac keyboard. Requires Gecko 15.0 */\n    VK_EISU: 22,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_JUNJA: 23,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_FINAL: 24,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_HANJA: 25,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_KANJI: 25,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_CONVERT: 28,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_NONCONVERT: 29,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_ACCEPT: 30,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_MODECHANGE: 31,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_SELECT: 41,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_PRINT: 42,\n    /** Linux support for this keycode was added in Gecko 4.0. */\n    VK_EXECUTE: 43,\n    /** Linux support for this keycode was added in Gecko 4.0.\t */\n    VK_SLEEP: 95\n};\n","import Hex from \"./hex.js\";\nimport Rect from \"./rect.js\";\nimport Tile from \"./tile.js\";\nimport TileGL from \"./tile-gl.js\";\nimport Term from \"./term.js\";\nimport * as Text from \"../text.js\";\nimport { DEFAULT_WIDTH, DEFAULT_HEIGHT } from \"../constants.js\";\nconst BACKENDS = {\n    \"hex\": Hex,\n    \"rect\": Rect,\n    \"tile\": Tile,\n    \"tile-gl\": TileGL,\n    \"term\": Term\n};\nconst DEFAULT_OPTIONS = {\n    width: DEFAULT_WIDTH,\n    height: DEFAULT_HEIGHT,\n    transpose: false,\n    layout: \"rect\",\n    fontSize: 15,\n    spacing: 1,\n    border: 0,\n    forceSquareRatio: false,\n    fontFamily: \"monospace\",\n    fontStyle: \"\",\n    fg: \"#ccc\",\n    bg: \"#000\",\n    tileWidth: 32,\n    tileHeight: 32,\n    tileMap: {},\n    tileSet: null,\n    tileColorize: false\n};\n/**\n * @class Visual map display\n */\nexport default class Display {\n    constructor(options = {}) {\n        this._data = {};\n        this._dirty = false; // false = nothing, true = all, object = dirty cells\n        this._options = {};\n        options = Object.assign({}, DEFAULT_OPTIONS, options);\n        this.setOptions(options);\n        this.DEBUG = this.DEBUG.bind(this);\n        this._tick = this._tick.bind(this);\n        this._backend.schedule(this._tick);\n    }\n    /**\n     * Debug helper, ideal as a map generator callback. Always bound to this.\n     * @param {int} x\n     * @param {int} y\n     * @param {int} what\n     */\n    DEBUG(x, y, what) {\n        let colors = [this._options.bg, this._options.fg];\n        this.draw(x, y, null, null, colors[what % colors.length]);\n    }\n    /**\n     * Clear the whole display (cover it with background color)\n     */\n    clear() {\n        this._data = {};\n        this._dirty = true;\n    }\n    /**\n     * @see ROT.Display\n     */\n    setOptions(options) {\n        Object.assign(this._options, options);\n        if (options.width || options.height || options.fontSize || options.fontFamily || options.spacing || options.layout) {\n            if (options.layout) {\n                let ctor = BACKENDS[options.layout];\n                this._backend = new ctor();\n            }\n            this._backend.setOptions(this._options);\n            this._dirty = true;\n        }\n        return this;\n    }\n    /**\n     * Returns currently set options\n     */\n    getOptions() { return this._options; }\n    /**\n     * Returns the DOM node of this display\n     */\n    getContainer() { return this._backend.getContainer(); }\n    /**\n     * Compute the maximum width/height to fit into a set of given constraints\n     * @param {int} availWidth Maximum allowed pixel width\n     * @param {int} availHeight Maximum allowed pixel height\n     * @returns {int[2]} cellWidth,cellHeight\n     */\n    computeSize(availWidth, availHeight) {\n        return this._backend.computeSize(availWidth, availHeight);\n    }\n    /**\n     * Compute the maximum font size to fit into a set of given constraints\n     * @param {int} availWidth Maximum allowed pixel width\n     * @param {int} availHeight Maximum allowed pixel height\n     * @returns {int} fontSize\n     */\n    computeFontSize(availWidth, availHeight) {\n        return this._backend.computeFontSize(availWidth, availHeight);\n    }\n    computeTileSize(availWidth, availHeight) {\n        let width = Math.floor(availWidth / this._options.width);\n        let height = Math.floor(availHeight / this._options.height);\n        return [width, height];\n    }\n    /**\n     * Convert a DOM event (mouse or touch) to map coordinates. Uses first touch for multi-touch.\n     * @param {Event} e event\n     * @returns {int[2]} -1 for values outside of the canvas\n     */\n    eventToPosition(e) {\n        let x, y;\n        if (\"touches\" in e) {\n            x = e.touches[0].clientX;\n            y = e.touches[0].clientY;\n        }\n        else {\n            x = e.clientX;\n            y = e.clientY;\n        }\n        return this._backend.eventToPosition(x, y);\n    }\n    /**\n     * @param {int} x\n     * @param {int} y\n     * @param {string || string[]} ch One or more chars (will be overlapping themselves)\n     * @param {string} [fg] foreground color\n     * @param {string} [bg] background color\n     */\n    draw(x, y, ch, fg, bg) {\n        if (!fg) {\n            fg = this._options.fg;\n        }\n        if (!bg) {\n            bg = this._options.bg;\n        }\n        let key = `${x},${y}`;\n        this._data[key] = [x, y, ch, fg, bg];\n        if (this._dirty === true) {\n            return;\n        } // will already redraw everything \n        if (!this._dirty) {\n            this._dirty = {};\n        } // first!\n        this._dirty[key] = true;\n    }\n    /**\n     * Draws a text at given position. Optionally wraps at a maximum length. Currently does not work with hex layout.\n     * @param {int} x\n     * @param {int} y\n     * @param {string} text May contain color/background format specifiers, %c{name}/%b{name}, both optional. %c{}/%b{} resets to default.\n     * @param {int} [maxWidth] wrap at what width?\n     * @returns {int} lines drawn\n     */\n    drawText(x, y, text, maxWidth) {\n        let fg = null;\n        let bg = null;\n        let cx = x;\n        let cy = y;\n        let lines = 1;\n        if (!maxWidth) {\n            maxWidth = this._options.width - x;\n        }\n        let tokens = Text.tokenize(text, maxWidth);\n        while (tokens.length) { // interpret tokenized opcode stream\n            let token = tokens.shift();\n            switch (token.type) {\n                case Text.TYPE_TEXT:\n                    let isSpace = false, isPrevSpace = false, isFullWidth = false, isPrevFullWidth = false;\n                    for (let i = 0; i < token.value.length; i++) {\n                        let cc = token.value.charCodeAt(i);\n                        let c = token.value.charAt(i);\n                        // Assign to `true` when the current char is full-width.\n                        isFullWidth = (cc > 0xff00 && cc < 0xff61) || (cc > 0xffdc && cc < 0xffe8) || cc > 0xffee;\n                        // Current char is space, whatever full-width or half-width both are OK.\n                        isSpace = (c.charCodeAt(0) == 0x20 || c.charCodeAt(0) == 0x3000);\n                        // The previous char is full-width and\n                        // current char is nether half-width nor a space.\n                        if (isPrevFullWidth && !isFullWidth && !isSpace) {\n                            cx++;\n                        } // add an extra position\n                        // The current char is full-width and\n                        // the previous char is not a space.\n                        if (isFullWidth && !isPrevSpace) {\n                            cx++;\n                        } // add an extra position\n                        this.draw(cx++, cy, c, fg, bg);\n                        isPrevSpace = isSpace;\n                        isPrevFullWidth = isFullWidth;\n                    }\n                    break;\n                case Text.TYPE_FG:\n                    fg = token.value || null;\n                    break;\n                case Text.TYPE_BG:\n                    bg = token.value || null;\n                    break;\n                case Text.TYPE_NEWLINE:\n                    cx = x;\n                    cy++;\n                    lines++;\n                    break;\n            }\n        }\n        return lines;\n    }\n    /**\n     * Timer tick: update dirty parts\n     */\n    _tick() {\n        this._backend.schedule(this._tick);\n        if (!this._dirty) {\n            return;\n        }\n        if (this._dirty === true) { // draw all\n            this._backend.clear();\n            for (let id in this._data) {\n                this._draw(id, false);\n            } // redraw cached data \n        }\n        else { // draw only dirty \n            for (let key in this._dirty) {\n                this._draw(key, true);\n            }\n        }\n        this._dirty = false;\n    }\n    /**\n     * @param {string} key What to draw\n     * @param {bool} clearBefore Is it necessary to clean before?\n     */\n    _draw(key, clearBefore) {\n        let data = this._data[key];\n        if (data[4] != this._options.bg) {\n            clearBefore = true;\n        }\n        this._backend.draw(data, clearBefore);\n    }\n}\nDisplay.Rect = Rect;\nDisplay.Hex = Hex;\nDisplay.Tile = Tile;\nDisplay.TileGL = TileGL;\nDisplay.Term = Term;\n","import RNG from \"./rng.js\";\n/**\n * @class (Markov process)-based string generator.\n * Copied from a <a href=\"http://www.roguebasin.roguelikedevelopment.org/index.php?title=Names_from_a_high_order_Markov_Process_and_a_simplified_Katz_back-off_scheme\">RogueBasin article</a>.\n * Offers configurable order and prior.\n */\nexport default class StringGenerator {\n    constructor(options) {\n        this._options = {\n            words: false,\n            order: 3,\n            prior: 0.001\n        };\n        Object.assign(this._options, options);\n        this._boundary = String.fromCharCode(0);\n        this._suffix = this._boundary;\n        this._prefix = [];\n        for (let i = 0; i < this._options.order; i++) {\n            this._prefix.push(this._boundary);\n        }\n        this._priorValues = {};\n        this._priorValues[this._boundary] = this._options.prior;\n        this._data = {};\n    }\n    /**\n     * Remove all learning data\n     */\n    clear() {\n        this._data = {};\n        this._priorValues = {};\n    }\n    /**\n     * @returns {string} Generated string\n     */\n    generate() {\n        let result = [this._sample(this._prefix)];\n        while (result[result.length - 1] != this._boundary) {\n            result.push(this._sample(result));\n        }\n        return this._join(result.slice(0, -1));\n    }\n    /**\n     * Observe (learn) a string from a training set\n     */\n    observe(string) {\n        let tokens = this._split(string);\n        for (let i = 0; i < tokens.length; i++) {\n            this._priorValues[tokens[i]] = this._options.prior;\n        }\n        tokens = this._prefix.concat(tokens).concat(this._suffix); /* add boundary symbols */\n        for (let i = this._options.order; i < tokens.length; i++) {\n            let context = tokens.slice(i - this._options.order, i);\n            let event = tokens[i];\n            for (let j = 0; j < context.length; j++) {\n                let subcontext = context.slice(j);\n                this._observeEvent(subcontext, event);\n            }\n        }\n    }\n    getStats() {\n        let parts = [];\n        let priorCount = Object.keys(this._priorValues).length;\n        priorCount--; // boundary\n        parts.push(\"distinct samples: \" + priorCount);\n        let dataCount = Object.keys(this._data).length;\n        let eventCount = 0;\n        for (let p in this._data) {\n            eventCount += Object.keys(this._data[p]).length;\n        }\n        parts.push(\"dictionary size (contexts): \" + dataCount);\n        parts.push(\"dictionary size (events): \" + eventCount);\n        return parts.join(\", \");\n    }\n    /**\n     * @param {string}\n     * @returns {string[]}\n     */\n    _split(str) {\n        return str.split(this._options.words ? /\\s+/ : \"\");\n    }\n    /**\n     * @param {string[]}\n     * @returns {string}\n     */\n    _join(arr) {\n        return arr.join(this._options.words ? \" \" : \"\");\n    }\n    /**\n     * @param {string[]} context\n     * @param {string} event\n     */\n    _observeEvent(context, event) {\n        let key = this._join(context);\n        if (!(key in this._data)) {\n            this._data[key] = {};\n        }\n        let data = this._data[key];\n        if (!(event in data)) {\n            data[event] = 0;\n        }\n        data[event]++;\n    }\n    /**\n     * @param {string[]}\n     * @returns {string}\n     */\n    _sample(context) {\n        context = this._backoff(context);\n        let key = this._join(context);\n        let data = this._data[key];\n        let available = {};\n        if (this._options.prior) {\n            for (let event in this._priorValues) {\n                available[event] = this._priorValues[event];\n            }\n            for (let event in data) {\n                available[event] += data[event];\n            }\n        }\n        else {\n            available = data;\n        }\n        return RNG.getWeightedValue(available);\n    }\n    /**\n     * @param {string[]}\n     * @returns {string[]}\n     */\n    _backoff(context) {\n        if (context.length > this._options.order) {\n            context = context.slice(-this._options.order);\n        }\n        else if (context.length < this._options.order) {\n            context = this._prefix.slice(0, this._options.order - context.length).concat(context);\n        }\n        while (!(this._join(context) in this._data) && context.length > 0) {\n            context = context.slice(1);\n        }\n        return context;\n    }\n}\n","export default class EventQueue {\n    /**\n     * @class Generic event queue: stores events and retrieves them based on their time\n     */\n    constructor() {\n        this._time = 0;\n        this._events = [];\n        this._eventTimes = [];\n    }\n    /**\n     * @returns {number} Elapsed time\n     */\n    getTime() { return this._time; }\n    /**\n     * Clear all scheduled events\n     */\n    clear() {\n        this._events = [];\n        this._eventTimes = [];\n        return this;\n    }\n    /**\n     * @param {?} event\n     * @param {number} time\n     */\n    add(event, time) {\n        let index = this._events.length;\n        for (let i = 0; i < this._eventTimes.length; i++) {\n            if (this._eventTimes[i] > time) {\n                index = i;\n                break;\n            }\n        }\n        this._events.splice(index, 0, event);\n        this._eventTimes.splice(index, 0, time);\n    }\n    /**\n     * Locates the nearest event, advances time if necessary. Returns that event and removes it from the queue.\n     * @returns {? || null} The event previously added by addEvent, null if no event available\n     */\n    get() {\n        if (!this._events.length) {\n            return null;\n        }\n        let time = this._eventTimes.splice(0, 1)[0];\n        if (time > 0) { /* advance */\n            this._time += time;\n            for (let i = 0; i < this._eventTimes.length; i++) {\n                this._eventTimes[i] -= time;\n            }\n        }\n        return this._events.splice(0, 1)[0];\n    }\n    /**\n     * Get the time associated with the given event\n     * @param {?} event\n     * @returns {number} time\n     */\n    getEventTime(event) {\n        let index = this._events.indexOf(event);\n        if (index == -1) {\n            return undefined;\n        }\n        return this._eventTimes[index];\n    }\n    /**\n     * Remove an event from the queue\n     * @param {?} event\n     * @returns {bool} success?\n     */\n    remove(event) {\n        let index = this._events.indexOf(event);\n        if (index == -1) {\n            return false;\n        }\n        this._remove(index);\n        return true;\n    }\n    ;\n    /**\n     * Remove an event from the queue\n     * @param {int} index\n     */\n    _remove(index) {\n        this._events.splice(index, 1);\n        this._eventTimes.splice(index, 1);\n    }\n    ;\n}\n","import EventQueue from \"../eventqueue.js\";\nexport default class Scheduler {\n    /**\n     * @class Abstract scheduler\n     */\n    constructor() {\n        this._queue = new EventQueue();\n        this._repeat = [];\n        this._current = null;\n    }\n    /**\n     * @see ROT.EventQueue#getTime\n     */\n    getTime() { return this._queue.getTime(); }\n    /**\n     * @param {?} item\n     * @param {bool} repeat\n     */\n    add(item, repeat) {\n        if (repeat) {\n            this._repeat.push(item);\n        }\n        return this;\n    }\n    /**\n     * Get the time the given item is scheduled for\n     * @param {?} item\n     * @returns {number} time\n     */\n    getTimeOf(item) {\n        return this._queue.getEventTime(item);\n    }\n    /**\n     * Clear all items\n     */\n    clear() {\n        this._queue.clear();\n        this._repeat = [];\n        this._current = null;\n        return this;\n    }\n    /**\n     * Remove a previously added item\n     * @param {?} item\n     * @returns {bool} successful?\n     */\n    remove(item) {\n        let result = this._queue.remove(item);\n        let index = this._repeat.indexOf(item);\n        if (index != -1) {\n            this._repeat.splice(index, 1);\n        }\n        if (this._current == item) {\n            this._current = null;\n        }\n        return result;\n    }\n    /**\n     * Schedule next item\n     * @returns {?}\n     */\n    next() {\n        this._current = this._queue.get();\n        return this._current;\n    }\n}\n","import Scheduler from \"./scheduler.js\";\n/**\n * @class Simple fair scheduler (round-robin style)\n */\nexport default class Simple extends Scheduler {\n    add(item, repeat) {\n        this._queue.add(item, 0);\n        return super.add(item, repeat);\n    }\n    next() {\n        if (this._current !== null && this._repeat.indexOf(this._current) != -1) {\n            this._queue.add(this._current, 0);\n        }\n        return super.next();\n    }\n}\n","import Scheduler from \"./scheduler.js\";\n/**\n * @class Speed-based scheduler\n */\nexport default class Speed extends Scheduler {\n    /**\n     * @param {object} item anything with \"getSpeed\" method\n     * @param {bool} repeat\n     * @param {number} [time=1/item.getSpeed()]\n     * @see ROT.Scheduler#add\n     */\n    add(item, repeat, time) {\n        this._queue.add(item, time !== undefined ? time : 1 / item.getSpeed());\n        return super.add(item, repeat);\n    }\n    /**\n     * @see ROT.Scheduler#next\n     */\n    next() {\n        if (this._current && this._repeat.indexOf(this._current) != -1) {\n            this._queue.add(this._current, 1 / this._current.getSpeed());\n        }\n        return super.next();\n    }\n}\n","import Scheduler from \"./scheduler.js\";\n/**\n * @class Action-based scheduler\n * @augments ROT.Scheduler\n */\nexport default class Action extends Scheduler {\n    constructor() {\n        super();\n        this._defaultDuration = 1; /* for newly added */\n        this._duration = this._defaultDuration; /* for this._current */\n    }\n    /**\n     * @param {object} item\n     * @param {bool} repeat\n     * @param {number} [time=1]\n     * @see ROT.Scheduler#add\n     */\n    add(item, repeat, time) {\n        this._queue.add(item, time || this._defaultDuration);\n        return super.add(item, repeat);\n    }\n    clear() {\n        this._duration = this._defaultDuration;\n        return super.clear();\n    }\n    remove(item) {\n        if (item == this._current) {\n            this._duration = this._defaultDuration;\n        }\n        return super.remove(item);\n    }\n    /**\n     * @see ROT.Scheduler#next\n     */\n    next() {\n        if (this._current !== null && this._repeat.indexOf(this._current) != -1) {\n            this._queue.add(this._current, this._duration || this._defaultDuration);\n            this._duration = this._defaultDuration;\n        }\n        return super.next();\n    }\n    /**\n     * Set duration for the active item\n     */\n    setDuration(time) {\n        if (this._current) {\n            this._duration = time;\n        }\n        return this;\n    }\n}\n","import Simple from \"./simple.js\";\nimport Speed from \"./speed.js\";\nimport Action from \"./action.js\";\nexport default { Simple, Speed, Action };\n","import { DIRS } from \"../constants.js\";\n;\n;\nexport default class FOV {\n    /**\n     * @class Abstract FOV algorithm\n     * @param {function} lightPassesCallback Does the light pass through x,y?\n     * @param {object} [options]\n     * @param {int} [options.topology=8] 4/6/8\n     */\n    constructor(lightPassesCallback, options = {}) {\n        this._lightPasses = lightPassesCallback;\n        this._options = Object.assign({ topology: 8 }, options);\n    }\n    /**\n     * Return all neighbors in a concentric ring\n     * @param {int} cx center-x\n     * @param {int} cy center-y\n     * @param {int} r range\n     */\n    _getCircle(cx, cy, r) {\n        let result = [];\n        let dirs, countFactor, startOffset;\n        switch (this._options.topology) {\n            case 4:\n                countFactor = 1;\n                startOffset = [0, 1];\n                dirs = [\n                    DIRS[8][7],\n                    DIRS[8][1],\n                    DIRS[8][3],\n                    DIRS[8][5]\n                ];\n                break;\n            case 6:\n                dirs = DIRS[6];\n                countFactor = 1;\n                startOffset = [-1, 1];\n                break;\n            case 8:\n                dirs = DIRS[4];\n                countFactor = 2;\n                startOffset = [-1, 1];\n                break;\n            default:\n                throw new Error(\"Incorrect topology for FOV computation\");\n                break;\n        }\n        /* starting neighbor */\n        let x = cx + startOffset[0] * r;\n        let y = cy + startOffset[1] * r;\n        /* circle */\n        for (let i = 0; i < dirs.length; i++) {\n            for (let j = 0; j < r * countFactor; j++) {\n                result.push([x, y]);\n                x += dirs[i][0];\n                y += dirs[i][1];\n            }\n        }\n        return result;\n    }\n}\n","import FOV from \"./fov.js\";\n/**\n * @class Discrete shadowcasting algorithm. Obsoleted by Precise shadowcasting.\n * @augments ROT.FOV\n */\nexport default class DiscreteShadowcasting extends FOV {\n    compute(x, y, R, callback) {\n        /* this place is always visible */\n        callback(x, y, 0, 1);\n        /* standing in a dark place. FIXME is this a good idea?  */\n        if (!this._lightPasses(x, y)) {\n            return;\n        }\n        /* start and end angles */\n        let DATA = [];\n        let A, B, cx, cy, blocks;\n        /* analyze surrounding cells in concentric rings, starting from the center */\n        for (let r = 1; r <= R; r++) {\n            let neighbors = this._getCircle(x, y, r);\n            let angle = 360 / neighbors.length;\n            for (let i = 0; i < neighbors.length; i++) {\n                cx = neighbors[i][0];\n                cy = neighbors[i][1];\n                A = angle * (i - 0.5);\n                B = A + angle;\n                blocks = !this._lightPasses(cx, cy);\n                if (this._visibleCoords(Math.floor(A), Math.ceil(B), blocks, DATA)) {\n                    callback(cx, cy, r, 1);\n                }\n                if (DATA.length == 2 && DATA[0] == 0 && DATA[1] == 360) {\n                    return;\n                } /* cutoff? */\n            } /* for all cells in this ring */\n        } /* for all rings */\n    }\n    /**\n     * @param {int} A start angle\n     * @param {int} B end angle\n     * @param {bool} blocks Does current cell block visibility?\n     * @param {int[][]} DATA shadowed angle pairs\n     */\n    _visibleCoords(A, B, blocks, DATA) {\n        if (A < 0) {\n            let v1 = this._visibleCoords(0, B, blocks, DATA);\n            let v2 = this._visibleCoords(360 + A, 360, blocks, DATA);\n            return v1 || v2;\n        }\n        let index = 0;\n        while (index < DATA.length && DATA[index] < A) {\n            index++;\n        }\n        if (index == DATA.length) { /* completely new shadow */\n            if (blocks) {\n                DATA.push(A, B);\n            }\n            return true;\n        }\n        let count = 0;\n        if (index % 2) { /* this shadow starts in an existing shadow, or within its ending boundary */\n            while (index < DATA.length && DATA[index] < B) {\n                index++;\n                count++;\n            }\n            if (count == 0) {\n                return false;\n            }\n            if (blocks) {\n                if (count % 2) {\n                    DATA.splice(index - count, count, B);\n                }\n                else {\n                    DATA.splice(index - count, count);\n                }\n            }\n            return true;\n        }\n        else { /* this shadow starts outside an existing shadow, or within a starting boundary */\n            while (index < DATA.length && DATA[index] < B) {\n                index++;\n                count++;\n            }\n            /* visible when outside an existing shadow, or when overlapping */\n            if (A == DATA[index - count] && count == 1) {\n                return false;\n            }\n            if (blocks) {\n                if (count % 2) {\n                    DATA.splice(index - count, count, A);\n                }\n                else {\n                    DATA.splice(index - count, count, A, B);\n                }\n            }\n            return true;\n        }\n    }\n}\n","import FOV from \"./fov.js\";\n/**\n * @class Precise shadowcasting algorithm\n * @augments ROT.FOV\n */\nexport default class PreciseShadowcasting extends FOV {\n    compute(x, y, R, callback) {\n        /* this place is always visible */\n        callback(x, y, 0, 1);\n        /* standing in a dark place. FIXME is this a good idea?  */\n        if (!this._lightPasses(x, y)) {\n            return;\n        }\n        /* list of all shadows */\n        let SHADOWS = [];\n        let cx, cy, blocks, A1, A2, visibility;\n        /* analyze surrounding cells in concentric rings, starting from the center */\n        for (let r = 1; r <= R; r++) {\n            let neighbors = this._getCircle(x, y, r);\n            let neighborCount = neighbors.length;\n            for (let i = 0; i < neighborCount; i++) {\n                cx = neighbors[i][0];\n                cy = neighbors[i][1];\n                /* shift half-an-angle backwards to maintain consistency of 0-th cells */\n                A1 = [i ? 2 * i - 1 : 2 * neighborCount - 1, 2 * neighborCount];\n                A2 = [2 * i + 1, 2 * neighborCount];\n                blocks = !this._lightPasses(cx, cy);\n                visibility = this._checkVisibility(A1, A2, blocks, SHADOWS);\n                if (visibility) {\n                    callback(cx, cy, r, visibility);\n                }\n                if (SHADOWS.length == 2 && SHADOWS[0][0] == 0 && SHADOWS[1][0] == SHADOWS[1][1]) {\n                    return;\n                } /* cutoff? */\n            } /* for all cells in this ring */\n        } /* for all rings */\n    }\n    /**\n     * @param {int[2]} A1 arc start\n     * @param {int[2]} A2 arc end\n     * @param {bool} blocks Does current arc block visibility?\n     * @param {int[][]} SHADOWS list of active shadows\n     */\n    _checkVisibility(A1, A2, blocks, SHADOWS) {\n        if (A1[0] > A2[0]) { /* split into two sub-arcs */\n            let v1 = this._checkVisibility(A1, [A1[1], A1[1]], blocks, SHADOWS);\n            let v2 = this._checkVisibility([0, 1], A2, blocks, SHADOWS);\n            return (v1 + v2) / 2;\n        }\n        /* index1: first shadow >= A1 */\n        let index1 = 0, edge1 = false;\n        while (index1 < SHADOWS.length) {\n            let old = SHADOWS[index1];\n            let diff = old[0] * A1[1] - A1[0] * old[1];\n            if (diff >= 0) { /* old >= A1 */\n                if (diff == 0 && !(index1 % 2)) {\n                    edge1 = true;\n                }\n                break;\n            }\n            index1++;\n        }\n        /* index2: last shadow <= A2 */\n        let index2 = SHADOWS.length, edge2 = false;\n        while (index2--) {\n            let old = SHADOWS[index2];\n            let diff = A2[0] * old[1] - old[0] * A2[1];\n            if (diff >= 0) { /* old <= A2 */\n                if (diff == 0 && (index2 % 2)) {\n                    edge2 = true;\n                }\n                break;\n            }\n        }\n        let visible = true;\n        if (index1 == index2 && (edge1 || edge2)) { /* subset of existing shadow, one of the edges match */\n            visible = false;\n        }\n        else if (edge1 && edge2 && index1 + 1 == index2 && (index2 % 2)) { /* completely equivalent with existing shadow */\n            visible = false;\n        }\n        else if (index1 > index2 && (index1 % 2)) { /* subset of existing shadow, not touching */\n            visible = false;\n        }\n        if (!visible) {\n            return 0;\n        } /* fast case: not visible */\n        let visibleLength;\n        /* compute the length of visible arc, adjust list of shadows (if blocking) */\n        let remove = index2 - index1 + 1;\n        if (remove % 2) {\n            if (index1 % 2) { /* first edge within existing shadow, second outside */\n                let P = SHADOWS[index1];\n                visibleLength = (A2[0] * P[1] - P[0] * A2[1]) / (P[1] * A2[1]);\n                if (blocks) {\n                    SHADOWS.splice(index1, remove, A2);\n                }\n            }\n            else { /* second edge within existing shadow, first outside */\n                let P = SHADOWS[index2];\n                visibleLength = (P[0] * A1[1] - A1[0] * P[1]) / (A1[1] * P[1]);\n                if (blocks) {\n                    SHADOWS.splice(index1, remove, A1);\n                }\n            }\n        }\n        else {\n            if (index1 % 2) { /* both edges within existing shadows */\n                let P1 = SHADOWS[index1];\n                let P2 = SHADOWS[index2];\n                visibleLength = (P2[0] * P1[1] - P1[0] * P2[1]) / (P1[1] * P2[1]);\n                if (blocks) {\n                    SHADOWS.splice(index1, remove);\n                }\n            }\n            else { /* both edges outside existing shadows */\n                if (blocks) {\n                    SHADOWS.splice(index1, remove, A1, A2);\n                }\n                return 1; /* whole arc visible! */\n            }\n        }\n        let arcLength = (A2[0] * A1[1] - A1[0] * A2[1]) / (A1[1] * A2[1]);\n        return visibleLength / arcLength;\n    }\n}\n","import FOV from \"./fov.js\";\n/** Octants used for translating recursive shadowcasting offsets */\nconst OCTANTS = [\n    [-1, 0, 0, 1],\n    [0, -1, 1, 0],\n    [0, -1, -1, 0],\n    [-1, 0, 0, -1],\n    [1, 0, 0, -1],\n    [0, 1, -1, 0],\n    [0, 1, 1, 0],\n    [1, 0, 0, 1]\n];\n/**\n * @class Recursive shadowcasting algorithm\n * Currently only supports 4/8 topologies, not hexagonal.\n * Based on Peter Harkins' implementation of Björn Bergström's algorithm described here: http://www.roguebasin.com/index.php?title=FOV_using_recursive_shadowcasting\n * @augments ROT.FOV\n */\nexport default class RecursiveShadowcasting extends FOV {\n    /**\n     * Compute visibility for a 360-degree circle\n     * @param {int} x\n     * @param {int} y\n     * @param {int} R Maximum visibility radius\n     * @param {function} callback\n     */\n    compute(x, y, R, callback) {\n        //You can always see your own tile\n        callback(x, y, 0, 1);\n        for (let i = 0; i < OCTANTS.length; i++) {\n            this._renderOctant(x, y, OCTANTS[i], R, callback);\n        }\n    }\n    /**\n     * Compute visibility for a 180-degree arc\n     * @param {int} x\n     * @param {int} y\n     * @param {int} R Maximum visibility radius\n     * @param {int} dir Direction to look in (expressed in a ROT.DIRS value);\n     * @param {function} callback\n     */\n    compute180(x, y, R, dir, callback) {\n        //You can always see your own tile\n        callback(x, y, 0, 1);\n        let previousOctant = (dir - 1 + 8) % 8; //Need to retrieve the previous octant to render a full 180 degrees\n        let nextPreviousOctant = (dir - 2 + 8) % 8; //Need to retrieve the previous two octants to render a full 180 degrees\n        let nextOctant = (dir + 1 + 8) % 8; //Need to grab to next octant to render a full 180 degrees\n        this._renderOctant(x, y, OCTANTS[nextPreviousOctant], R, callback);\n        this._renderOctant(x, y, OCTANTS[previousOctant], R, callback);\n        this._renderOctant(x, y, OCTANTS[dir], R, callback);\n        this._renderOctant(x, y, OCTANTS[nextOctant], R, callback);\n    }\n    ;\n    /**\n     * Compute visibility for a 90-degree arc\n     * @param {int} x\n     * @param {int} y\n     * @param {int} R Maximum visibility radius\n     * @param {int} dir Direction to look in (expressed in a ROT.DIRS value);\n     * @param {function} callback\n     */\n    compute90(x, y, R, dir, callback) {\n        //You can always see your own tile\n        callback(x, y, 0, 1);\n        let previousOctant = (dir - 1 + 8) % 8; //Need to retrieve the previous octant to render a full 90 degrees\n        this._renderOctant(x, y, OCTANTS[dir], R, callback);\n        this._renderOctant(x, y, OCTANTS[previousOctant], R, callback);\n    }\n    /**\n     * Render one octant (45-degree arc) of the viewshed\n     * @param {int} x\n     * @param {int} y\n     * @param {int} octant Octant to be rendered\n     * @param {int} R Maximum visibility radius\n     * @param {function} callback\n     */\n    _renderOctant(x, y, octant, R, callback) {\n        //Radius incremented by 1 to provide same coverage area as other shadowcasting radiuses\n        this._castVisibility(x, y, 1, 1.0, 0.0, R + 1, octant[0], octant[1], octant[2], octant[3], callback);\n    }\n    /**\n     * Actually calculates the visibility\n     * @param {int} startX The starting X coordinate\n     * @param {int} startY The starting Y coordinate\n     * @param {int} row The row to render\n     * @param {float} visSlopeStart The slope to start at\n     * @param {float} visSlopeEnd The slope to end at\n     * @param {int} radius The radius to reach out to\n     * @param {int} xx\n     * @param {int} xy\n     * @param {int} yx\n     * @param {int} yy\n     * @param {function} callback The callback to use when we hit a block that is visible\n     */\n    _castVisibility(startX, startY, row, visSlopeStart, visSlopeEnd, radius, xx, xy, yx, yy, callback) {\n        if (visSlopeStart < visSlopeEnd) {\n            return;\n        }\n        for (let i = row; i <= radius; i++) {\n            let dx = -i - 1;\n            let dy = -i;\n            let blocked = false;\n            let newStart = 0;\n            //'Row' could be column, names here assume octant 0 and would be flipped for half the octants\n            while (dx <= 0) {\n                dx += 1;\n                //Translate from relative coordinates to map coordinates\n                let mapX = startX + dx * xx + dy * xy;\n                let mapY = startY + dx * yx + dy * yy;\n                //Range of the row\n                let slopeStart = (dx - 0.5) / (dy + 0.5);\n                let slopeEnd = (dx + 0.5) / (dy - 0.5);\n                //Ignore if not yet at left edge of Octant\n                if (slopeEnd > visSlopeStart) {\n                    continue;\n                }\n                //Done if past right edge\n                if (slopeStart < visSlopeEnd) {\n                    break;\n                }\n                //If it's in range, it's visible\n                if ((dx * dx + dy * dy) < (radius * radius)) {\n                    callback(mapX, mapY, i, 1);\n                }\n                if (!blocked) {\n                    //If tile is a blocking tile, cast around it\n                    if (!this._lightPasses(mapX, mapY) && i < radius) {\n                        blocked = true;\n                        this._castVisibility(startX, startY, i + 1, visSlopeStart, slopeStart, radius, xx, xy, yx, yy, callback);\n                        newStart = slopeEnd;\n                    }\n                }\n                else {\n                    //Keep narrowing if scanning across a block\n                    if (!this._lightPasses(mapX, mapY)) {\n                        newStart = slopeEnd;\n                        continue;\n                    }\n                    //Block has ended\n                    blocked = false;\n                    visSlopeStart = newStart;\n                }\n            }\n            if (blocked) {\n                break;\n            }\n        }\n    }\n}\n","import DiscreteShadowcasting from \"./discrete-shadowcasting.js\";\nimport PreciseShadowcasting from \"./precise-shadowcasting.js\";\nimport RecursiveShadowcasting from \"./recursive-shadowcasting.js\";\nexport default { DiscreteShadowcasting, PreciseShadowcasting, RecursiveShadowcasting };\n","import { DEFAULT_WIDTH, DEFAULT_HEIGHT } from \"../constants.js\";\n;\nexport default class Map {\n    /**\n     * @class Base map generator\n     * @param {int} [width=ROT.DEFAULT_WIDTH]\n     * @param {int} [height=ROT.DEFAULT_HEIGHT]\n     */\n    constructor(width = DEFAULT_WIDTH, height = DEFAULT_HEIGHT) {\n        this._width = width;\n        this._height = height;\n    }\n    ;\n    _fillMap(value) {\n        let map = [];\n        for (let i = 0; i < this._width; i++) {\n            map.push([]);\n            for (let j = 0; j < this._height; j++) {\n                map[i].push(value);\n            }\n        }\n        return map;\n    }\n}\n","import Map from \"./map.js\";\n/**\n * @class Simple empty rectangular room\n * @augments ROT.Map\n */\nexport default class Arena extends Map {\n    create(callback) {\n        let w = this._width - 1;\n        let h = this._height - 1;\n        for (let i = 0; i <= w; i++) {\n            for (let j = 0; j <= h; j++) {\n                let empty = (i && j && i < w && j < h);\n                callback(i, j, empty ? 0 : 1);\n            }\n        }\n        return this;\n    }\n}\n","import Map from \"./map.js\";\n/**\n * @class Dungeon map: has rooms and corridors\n * @augments ROT.Map\n */\nexport default class Dungeon extends Map {\n    constructor(width, height) {\n        super(width, height);\n        this._rooms = [];\n        this._corridors = [];\n    }\n    /**\n     * Get all generated rooms\n     * @returns {ROT.Map.Feature.Room[]}\n     */\n    getRooms() { return this._rooms; }\n    /**\n     * Get all generated corridors\n     * @returns {ROT.Map.Feature.Corridor[]}\n     */\n    getCorridors() { return this._corridors; }\n}\n","import RNG from \"../rng.js\";\n;\n/**\n * @class Dungeon feature; has own .create() method\n */\nclass Feature {\n}\n/**\n * @class Room\n * @augments ROT.Map.Feature\n * @param {int} x1\n * @param {int} y1\n * @param {int} x2\n * @param {int} y2\n * @param {int} [doorX]\n * @param {int} [doorY]\n */\nexport class Room extends Feature {\n    constructor(x1, y1, x2, y2, doorX, doorY) {\n        super();\n        this._x1 = x1;\n        this._y1 = y1;\n        this._x2 = x2;\n        this._y2 = y2;\n        this._doors = {};\n        if (doorX !== undefined && doorY !== undefined) {\n            this.addDoor(doorX, doorY);\n        }\n    }\n    ;\n    /**\n     * Room of random size, with a given doors and direction\n     */\n    static createRandomAt(x, y, dx, dy, options) {\n        let min = options.roomWidth[0];\n        let max = options.roomWidth[1];\n        let width = RNG.getUniformInt(min, max);\n        min = options.roomHeight[0];\n        max = options.roomHeight[1];\n        let height = RNG.getUniformInt(min, max);\n        if (dx == 1) { /* to the right */\n            let y2 = y - Math.floor(RNG.getUniform() * height);\n            return new this(x + 1, y2, x + width, y2 + height - 1, x, y);\n        }\n        if (dx == -1) { /* to the left */\n            let y2 = y - Math.floor(RNG.getUniform() * height);\n            return new this(x - width, y2, x - 1, y2 + height - 1, x, y);\n        }\n        if (dy == 1) { /* to the bottom */\n            let x2 = x - Math.floor(RNG.getUniform() * width);\n            return new this(x2, y + 1, x2 + width - 1, y + height, x, y);\n        }\n        if (dy == -1) { /* to the top */\n            let x2 = x - Math.floor(RNG.getUniform() * width);\n            return new this(x2, y - height, x2 + width - 1, y - 1, x, y);\n        }\n        throw new Error(\"dx or dy must be 1 or -1\");\n    }\n    /**\n     * Room of random size, positioned around center coords\n     */\n    static createRandomCenter(cx, cy, options) {\n        let min = options.roomWidth[0];\n        let max = options.roomWidth[1];\n        let width = RNG.getUniformInt(min, max);\n        min = options.roomHeight[0];\n        max = options.roomHeight[1];\n        let height = RNG.getUniformInt(min, max);\n        let x1 = cx - Math.floor(RNG.getUniform() * width);\n        let y1 = cy - Math.floor(RNG.getUniform() * height);\n        let x2 = x1 + width - 1;\n        let y2 = y1 + height - 1;\n        return new this(x1, y1, x2, y2);\n    }\n    /**\n     * Room of random size within a given dimensions\n     */\n    static createRandom(availWidth, availHeight, options) {\n        let min = options.roomWidth[0];\n        let max = options.roomWidth[1];\n        let width = RNG.getUniformInt(min, max);\n        min = options.roomHeight[0];\n        max = options.roomHeight[1];\n        let height = RNG.getUniformInt(min, max);\n        let left = availWidth - width - 1;\n        let top = availHeight - height - 1;\n        let x1 = 1 + Math.floor(RNG.getUniform() * left);\n        let y1 = 1 + Math.floor(RNG.getUniform() * top);\n        let x2 = x1 + width - 1;\n        let y2 = y1 + height - 1;\n        return new this(x1, y1, x2, y2);\n    }\n    addDoor(x, y) {\n        this._doors[x + \",\" + y] = 1;\n        return this;\n    }\n    /**\n     * @param {function}\n     */\n    getDoors(cb) {\n        for (let key in this._doors) {\n            let parts = key.split(\",\");\n            cb(parseInt(parts[0]), parseInt(parts[1]));\n        }\n        return this;\n    }\n    clearDoors() {\n        this._doors = {};\n        return this;\n    }\n    addDoors(isWallCallback) {\n        let left = this._x1 - 1;\n        let right = this._x2 + 1;\n        let top = this._y1 - 1;\n        let bottom = this._y2 + 1;\n        for (let x = left; x <= right; x++) {\n            for (let y = top; y <= bottom; y++) {\n                if (x != left && x != right && y != top && y != bottom) {\n                    continue;\n                }\n                if (isWallCallback(x, y)) {\n                    continue;\n                }\n                this.addDoor(x, y);\n            }\n        }\n        return this;\n    }\n    debug() {\n        console.log(\"room\", this._x1, this._y1, this._x2, this._y2);\n    }\n    isValid(isWallCallback, canBeDugCallback) {\n        let left = this._x1 - 1;\n        let right = this._x2 + 1;\n        let top = this._y1 - 1;\n        let bottom = this._y2 + 1;\n        for (let x = left; x <= right; x++) {\n            for (let y = top; y <= bottom; y++) {\n                if (x == left || x == right || y == top || y == bottom) {\n                    if (!isWallCallback(x, y)) {\n                        return false;\n                    }\n                }\n                else {\n                    if (!canBeDugCallback(x, y)) {\n                        return false;\n                    }\n                }\n            }\n        }\n        return true;\n    }\n    /**\n     * @param {function} digCallback Dig callback with a signature (x, y, value). Values: 0 = empty, 1 = wall, 2 = door. Multiple doors are allowed.\n     */\n    create(digCallback) {\n        let left = this._x1 - 1;\n        let right = this._x2 + 1;\n        let top = this._y1 - 1;\n        let bottom = this._y2 + 1;\n        let value = 0;\n        for (let x = left; x <= right; x++) {\n            for (let y = top; y <= bottom; y++) {\n                if (x + \",\" + y in this._doors) {\n                    value = 2;\n                }\n                else if (x == left || x == right || y == top || y == bottom) {\n                    value = 1;\n                }\n                else {\n                    value = 0;\n                }\n                digCallback(x, y, value);\n            }\n        }\n    }\n    getCenter() {\n        return [Math.round((this._x1 + this._x2) / 2), Math.round((this._y1 + this._y2) / 2)];\n    }\n    getLeft() { return this._x1; }\n    getRight() { return this._x2; }\n    getTop() { return this._y1; }\n    getBottom() { return this._y2; }\n}\n/**\n * @class Corridor\n * @augments ROT.Map.Feature\n * @param {int} startX\n * @param {int} startY\n * @param {int} endX\n * @param {int} endY\n */\nexport class Corridor extends Feature {\n    constructor(startX, startY, endX, endY) {\n        super();\n        this._startX = startX;\n        this._startY = startY;\n        this._endX = endX;\n        this._endY = endY;\n        this._endsWithAWall = true;\n    }\n    static createRandomAt(x, y, dx, dy, options) {\n        let min = options.corridorLength[0];\n        let max = options.corridorLength[1];\n        let length = RNG.getUniformInt(min, max);\n        return new this(x, y, x + dx * length, y + dy * length);\n    }\n    debug() {\n        console.log(\"corridor\", this._startX, this._startY, this._endX, this._endY);\n    }\n    isValid(isWallCallback, canBeDugCallback) {\n        let sx = this._startX;\n        let sy = this._startY;\n        let dx = this._endX - sx;\n        let dy = this._endY - sy;\n        let length = 1 + Math.max(Math.abs(dx), Math.abs(dy));\n        if (dx) {\n            dx = dx / Math.abs(dx);\n        }\n        if (dy) {\n            dy = dy / Math.abs(dy);\n        }\n        let nx = dy;\n        let ny = -dx;\n        let ok = true;\n        for (let i = 0; i < length; i++) {\n            let x = sx + i * dx;\n            let y = sy + i * dy;\n            if (!canBeDugCallback(x, y)) {\n                ok = false;\n            }\n            if (!isWallCallback(x + nx, y + ny)) {\n                ok = false;\n            }\n            if (!isWallCallback(x - nx, y - ny)) {\n                ok = false;\n            }\n            if (!ok) {\n                length = i;\n                this._endX = x - dx;\n                this._endY = y - dy;\n                break;\n            }\n        }\n        /**\n         * If the length degenerated, this corridor might be invalid\n         */\n        /* not supported */\n        if (length == 0) {\n            return false;\n        }\n        /* length 1 allowed only if the next space is empty */\n        if (length == 1 && isWallCallback(this._endX + dx, this._endY + dy)) {\n            return false;\n        }\n        /**\n         * We do not want the corridor to crash into a corner of a room;\n         * if any of the ending corners is empty, the N+1th cell of this corridor must be empty too.\n         *\n         * Situation:\n         * #######1\n         * .......?\n         * #######2\n         *\n         * The corridor was dug from left to right.\n         * 1, 2 - problematic corners, ? = N+1th cell (not dug)\n         */\n        let firstCornerBad = !isWallCallback(this._endX + dx + nx, this._endY + dy + ny);\n        let secondCornerBad = !isWallCallback(this._endX + dx - nx, this._endY + dy - ny);\n        this._endsWithAWall = isWallCallback(this._endX + dx, this._endY + dy);\n        if ((firstCornerBad || secondCornerBad) && this._endsWithAWall) {\n            return false;\n        }\n        return true;\n    }\n    /**\n     * @param {function} digCallback Dig callback with a signature (x, y, value). Values: 0 = empty.\n     */\n    create(digCallback) {\n        let sx = this._startX;\n        let sy = this._startY;\n        let dx = this._endX - sx;\n        let dy = this._endY - sy;\n        let length = 1 + Math.max(Math.abs(dx), Math.abs(dy));\n        if (dx) {\n            dx = dx / Math.abs(dx);\n        }\n        if (dy) {\n            dy = dy / Math.abs(dy);\n        }\n        for (let i = 0; i < length; i++) {\n            let x = sx + i * dx;\n            let y = sy + i * dy;\n            digCallback(x, y, 0);\n        }\n        return true;\n    }\n    createPriorityWalls(priorityWallCallback) {\n        if (!this._endsWithAWall) {\n            return;\n        }\n        let sx = this._startX;\n        let sy = this._startY;\n        let dx = this._endX - sx;\n        let dy = this._endY - sy;\n        if (dx) {\n            dx = dx / Math.abs(dx);\n        }\n        if (dy) {\n            dy = dy / Math.abs(dy);\n        }\n        let nx = dy;\n        let ny = -dx;\n        priorityWallCallback(this._endX + dx, this._endY + dy);\n        priorityWallCallback(this._endX + nx, this._endY + ny);\n        priorityWallCallback(this._endX - nx, this._endY - ny);\n    }\n}\n","import Dungeon from \"./dungeon.js\";\nimport { Room, Corridor } from \"./features.js\";\nimport RNG from \"../rng.js\";\n;\n/**\n * @class Dungeon generator which tries to fill the space evenly. Generates independent rooms and tries to connect them.\n * @augments ROT.Map.Dungeon\n */\nexport default class Uniform extends Dungeon {\n    constructor(width, height, options) {\n        super(width, height);\n        this._options = {\n            roomWidth: [3, 9],\n            roomHeight: [3, 5],\n            roomDugPercentage: 0.1,\n            timeLimit: 1000 /* we stop after this much time has passed (msec) */\n        };\n        Object.assign(this._options, options);\n        this._map = [];\n        this._dug = 0;\n        this._roomAttempts = 20; /* new room is created N-times until is considered as impossible to generate */\n        this._corridorAttempts = 20; /* corridors are tried N-times until the level is considered as impossible to connect */\n        this._connected = []; /* list of already connected rooms */\n        this._unconnected = []; /* list of remaining unconnected rooms */\n        this._digCallback = this._digCallback.bind(this);\n        this._canBeDugCallback = this._canBeDugCallback.bind(this);\n        this._isWallCallback = this._isWallCallback.bind(this);\n    }\n    /**\n     * Create a map. If the time limit has been hit, returns null.\n     * @see ROT.Map#create\n     */\n    create(callback) {\n        let t1 = Date.now();\n        while (1) {\n            let t2 = Date.now();\n            if (t2 - t1 > this._options.timeLimit) {\n                return null;\n            } /* time limit! */\n            this._map = this._fillMap(1);\n            this._dug = 0;\n            this._rooms = [];\n            this._unconnected = [];\n            this._generateRooms();\n            if (this._rooms.length < 2) {\n                continue;\n            }\n            if (this._generateCorridors()) {\n                break;\n            }\n        }\n        if (callback) {\n            for (let i = 0; i < this._width; i++) {\n                for (let j = 0; j < this._height; j++) {\n                    callback(i, j, this._map[i][j]);\n                }\n            }\n        }\n        return this;\n    }\n    /**\n     * Generates a suitable amount of rooms\n     */\n    _generateRooms() {\n        let w = this._width - 2;\n        let h = this._height - 2;\n        let room;\n        do {\n            room = this._generateRoom();\n            if (this._dug / (w * h) > this._options.roomDugPercentage) {\n                break;\n            } /* achieved requested amount of free space */\n        } while (room);\n        /* either enough rooms, or not able to generate more of them :) */\n    }\n    /**\n     * Try to generate one room\n     */\n    _generateRoom() {\n        let count = 0;\n        while (count < this._roomAttempts) {\n            count++;\n            let room = Room.createRandom(this._width, this._height, this._options);\n            if (!room.isValid(this._isWallCallback, this._canBeDugCallback)) {\n                continue;\n            }\n            room.create(this._digCallback);\n            this._rooms.push(room);\n            return room;\n        }\n        /* no room was generated in a given number of attempts */\n        return null;\n    }\n    /**\n     * Generates connectors beween rooms\n     * @returns {bool} success Was this attempt successfull?\n     */\n    _generateCorridors() {\n        let cnt = 0;\n        while (cnt < this._corridorAttempts) {\n            cnt++;\n            this._corridors = [];\n            /* dig rooms into a clear map */\n            this._map = this._fillMap(1);\n            for (let i = 0; i < this._rooms.length; i++) {\n                let room = this._rooms[i];\n                room.clearDoors();\n                room.create(this._digCallback);\n            }\n            this._unconnected = RNG.shuffle(this._rooms.slice());\n            this._connected = [];\n            if (this._unconnected.length) {\n                this._connected.push(this._unconnected.pop());\n            } /* first one is always connected */\n            while (1) {\n                /* 1. pick random connected room */\n                let connected = RNG.getItem(this._connected);\n                if (!connected) {\n                    break;\n                }\n                /* 2. find closest unconnected */\n                let room1 = this._closestRoom(this._unconnected, connected);\n                if (!room1) {\n                    break;\n                }\n                /* 3. connect it to closest connected */\n                let room2 = this._closestRoom(this._connected, room1);\n                if (!room2) {\n                    break;\n                }\n                let ok = this._connectRooms(room1, room2);\n                if (!ok) {\n                    break;\n                } /* stop connecting, re-shuffle */\n                if (!this._unconnected.length) {\n                    return true;\n                } /* done; no rooms remain */\n            }\n        }\n        return false;\n    }\n    ;\n    /**\n     * For a given room, find the closest one from the list\n     */\n    _closestRoom(rooms, room) {\n        let dist = Infinity;\n        let center = room.getCenter();\n        let result = null;\n        for (let i = 0; i < rooms.length; i++) {\n            let r = rooms[i];\n            let c = r.getCenter();\n            let dx = c[0] - center[0];\n            let dy = c[1] - center[1];\n            let d = dx * dx + dy * dy;\n            if (d < dist) {\n                dist = d;\n                result = r;\n            }\n        }\n        return result;\n    }\n    _connectRooms(room1, room2) {\n        /*\n            room1.debug();\n            room2.debug();\n        */\n        let center1 = room1.getCenter();\n        let center2 = room2.getCenter();\n        let diffX = center2[0] - center1[0];\n        let diffY = center2[1] - center1[1];\n        let start;\n        let end;\n        let dirIndex1, dirIndex2, min, max, index;\n        if (Math.abs(diffX) < Math.abs(diffY)) { /* first try connecting north-south walls */\n            dirIndex1 = (diffY > 0 ? 2 : 0);\n            dirIndex2 = (dirIndex1 + 2) % 4;\n            min = room2.getLeft();\n            max = room2.getRight();\n            index = 0;\n        }\n        else { /* first try connecting east-west walls */\n            dirIndex1 = (diffX > 0 ? 1 : 3);\n            dirIndex2 = (dirIndex1 + 2) % 4;\n            min = room2.getTop();\n            max = room2.getBottom();\n            index = 1;\n        }\n        start = this._placeInWall(room1, dirIndex1); /* corridor will start here */\n        if (!start) {\n            return false;\n        }\n        if (start[index] >= min && start[index] <= max) { /* possible to connect with straight line (I-like) */\n            end = start.slice();\n            let value = 0;\n            switch (dirIndex2) {\n                case 0:\n                    value = room2.getTop() - 1;\n                    break;\n                case 1:\n                    value = room2.getRight() + 1;\n                    break;\n                case 2:\n                    value = room2.getBottom() + 1;\n                    break;\n                case 3:\n                    value = room2.getLeft() - 1;\n                    break;\n            }\n            end[(index + 1) % 2] = value;\n            this._digLine([start, end]);\n        }\n        else if (start[index] < min - 1 || start[index] > max + 1) { /* need to switch target wall (L-like) */\n            let diff = start[index] - center2[index];\n            let rotation = 0;\n            switch (dirIndex2) {\n                case 0:\n                case 1:\n                    rotation = (diff < 0 ? 3 : 1);\n                    break;\n                case 2:\n                case 3:\n                    rotation = (diff < 0 ? 1 : 3);\n                    break;\n            }\n            dirIndex2 = (dirIndex2 + rotation) % 4;\n            end = this._placeInWall(room2, dirIndex2);\n            if (!end) {\n                return false;\n            }\n            let mid = [0, 0];\n            mid[index] = start[index];\n            let index2 = (index + 1) % 2;\n            mid[index2] = end[index2];\n            this._digLine([start, mid, end]);\n        }\n        else { /* use current wall pair, but adjust the line in the middle (S-like) */\n            let index2 = (index + 1) % 2;\n            end = this._placeInWall(room2, dirIndex2);\n            if (!end) {\n                return false;\n            }\n            let mid = Math.round((end[index2] + start[index2]) / 2);\n            let mid1 = [0, 0];\n            let mid2 = [0, 0];\n            mid1[index] = start[index];\n            mid1[index2] = mid;\n            mid2[index] = end[index];\n            mid2[index2] = mid;\n            this._digLine([start, mid1, mid2, end]);\n        }\n        room1.addDoor(start[0], start[1]);\n        room2.addDoor(end[0], end[1]);\n        index = this._unconnected.indexOf(room1);\n        if (index != -1) {\n            this._unconnected.splice(index, 1);\n            this._connected.push(room1);\n        }\n        index = this._unconnected.indexOf(room2);\n        if (index != -1) {\n            this._unconnected.splice(index, 1);\n            this._connected.push(room2);\n        }\n        return true;\n    }\n    _placeInWall(room, dirIndex) {\n        let start = [0, 0];\n        let dir = [0, 0];\n        let length = 0;\n        switch (dirIndex) {\n            case 0:\n                dir = [1, 0];\n                start = [room.getLeft(), room.getTop() - 1];\n                length = room.getRight() - room.getLeft() + 1;\n                break;\n            case 1:\n                dir = [0, 1];\n                start = [room.getRight() + 1, room.getTop()];\n                length = room.getBottom() - room.getTop() + 1;\n                break;\n            case 2:\n                dir = [1, 0];\n                start = [room.getLeft(), room.getBottom() + 1];\n                length = room.getRight() - room.getLeft() + 1;\n                break;\n            case 3:\n                dir = [0, 1];\n                start = [room.getLeft() - 1, room.getTop()];\n                length = room.getBottom() - room.getTop() + 1;\n                break;\n        }\n        let avail = [];\n        let lastBadIndex = -2;\n        for (let i = 0; i < length; i++) {\n            let x = start[0] + i * dir[0];\n            let y = start[1] + i * dir[1];\n            avail.push(null);\n            let isWall = (this._map[x][y] == 1);\n            if (isWall) {\n                if (lastBadIndex != i - 1) {\n                    avail[i] = [x, y];\n                }\n            }\n            else {\n                lastBadIndex = i;\n                if (i) {\n                    avail[i - 1] = null;\n                }\n            }\n        }\n        for (let i = avail.length - 1; i >= 0; i--) {\n            if (!avail[i]) {\n                avail.splice(i, 1);\n            }\n        }\n        return (avail.length ? RNG.getItem(avail) : null);\n    }\n    /**\n     * Dig a polyline.\n     */\n    _digLine(points) {\n        for (let i = 1; i < points.length; i++) {\n            let start = points[i - 1];\n            let end = points[i];\n            let corridor = new Corridor(start[0], start[1], end[0], end[1]);\n            corridor.create(this._digCallback);\n            this._corridors.push(corridor);\n        }\n    }\n    _digCallback(x, y, value) {\n        this._map[x][y] = value;\n        if (value == 0) {\n            this._dug++;\n        }\n    }\n    _isWallCallback(x, y) {\n        if (x < 0 || y < 0 || x >= this._width || y >= this._height) {\n            return false;\n        }\n        return (this._map[x][y] == 1);\n    }\n    _canBeDugCallback(x, y) {\n        if (x < 1 || y < 1 || x + 1 >= this._width || y + 1 >= this._height) {\n            return false;\n        }\n        return (this._map[x][y] == 1);\n    }\n}\n","import Map from \"./map.js\";\nimport { DIRS } from \"../constants.js\";\nimport RNG from \"../rng.js\";\n;\n/**\n * @class Cellular automaton map generator\n * @augments ROT.Map\n * @param {int} [width=ROT.DEFAULT_WIDTH]\n * @param {int} [height=ROT.DEFAULT_HEIGHT]\n * @param {object} [options] Options\n * @param {int[]} [options.born] List of neighbor counts for a new cell to be born in empty space\n * @param {int[]} [options.survive] List of neighbor counts for an existing  cell to survive\n * @param {int} [options.topology] Topology 4 or 6 or 8\n */\nexport default class Cellular extends Map {\n    constructor(width, height, options = {}) {\n        super(width, height);\n        this._options = {\n            born: [5, 6, 7, 8],\n            survive: [4, 5, 6, 7, 8],\n            topology: 8\n        };\n        this.setOptions(options);\n        this._dirs = DIRS[this._options.topology];\n        this._map = this._fillMap(0);\n    }\n    /**\n     * Fill the map with random values\n     * @param {float} probability Probability for a cell to become alive; 0 = all empty, 1 = all full\n     */\n    randomize(probability) {\n        for (let i = 0; i < this._width; i++) {\n            for (let j = 0; j < this._height; j++) {\n                this._map[i][j] = (RNG.getUniform() < probability ? 1 : 0);\n            }\n        }\n        return this;\n    }\n    /**\n     * Change options.\n     * @see ROT.Map.Cellular\n     */\n    setOptions(options) { Object.assign(this._options, options); }\n    set(x, y, value) { this._map[x][y] = value; }\n    create(callback) {\n        let newMap = this._fillMap(0);\n        let born = this._options.born;\n        let survive = this._options.survive;\n        for (let j = 0; j < this._height; j++) {\n            let widthStep = 1;\n            let widthStart = 0;\n            if (this._options.topology == 6) {\n                widthStep = 2;\n                widthStart = j % 2;\n            }\n            for (let i = widthStart; i < this._width; i += widthStep) {\n                let cur = this._map[i][j];\n                let ncount = this._getNeighbors(i, j);\n                if (cur && survive.indexOf(ncount) != -1) { /* survive */\n                    newMap[i][j] = 1;\n                }\n                else if (!cur && born.indexOf(ncount) != -1) { /* born */\n                    newMap[i][j] = 1;\n                }\n            }\n        }\n        this._map = newMap;\n        callback && this._serviceCallback(callback);\n    }\n    _serviceCallback(callback) {\n        for (let j = 0; j < this._height; j++) {\n            let widthStep = 1;\n            let widthStart = 0;\n            if (this._options.topology == 6) {\n                widthStep = 2;\n                widthStart = j % 2;\n            }\n            for (let i = widthStart; i < this._width; i += widthStep) {\n                callback(i, j, this._map[i][j]);\n            }\n        }\n    }\n    /**\n     * Get neighbor count at [i,j] in this._map\n     */\n    _getNeighbors(cx, cy) {\n        let result = 0;\n        for (let i = 0; i < this._dirs.length; i++) {\n            let dir = this._dirs[i];\n            let x = cx + dir[0];\n            let y = cy + dir[1];\n            if (x < 0 || x >= this._width || y < 0 || y >= this._height) {\n                continue;\n            }\n            result += (this._map[x][y] == 1 ? 1 : 0);\n        }\n        return result;\n    }\n    /**\n     * Make sure every non-wall space is accessible.\n     * @param {function} callback to call to display map when do\n     * @param {int} value to consider empty space - defaults to 0\n     * @param {function} callback to call when a new connection is made\n     */\n    connect(callback, value, connectionCallback) {\n        if (!value)\n            value = 0;\n        let allFreeSpace = [];\n        let notConnected = {};\n        // find all free space\n        let widthStep = 1;\n        let widthStarts = [0, 0];\n        if (this._options.topology == 6) {\n            widthStep = 2;\n            widthStarts = [0, 1];\n        }\n        for (let y = 0; y < this._height; y++) {\n            for (let x = widthStarts[y % 2]; x < this._width; x += widthStep) {\n                if (this._freeSpace(x, y, value)) {\n                    let p = [x, y];\n                    notConnected[this._pointKey(p)] = p;\n                    allFreeSpace.push([x, y]);\n                }\n            }\n        }\n        let start = allFreeSpace[RNG.getUniformInt(0, allFreeSpace.length - 1)];\n        let key = this._pointKey(start);\n        let connected = {};\n        connected[key] = start;\n        delete notConnected[key];\n        // find what's connected to the starting point\n        this._findConnected(connected, notConnected, [start], false, value);\n        while (Object.keys(notConnected).length > 0) {\n            // find two points from notConnected to connected\n            let p = this._getFromTo(connected, notConnected);\n            let from = p[0]; // notConnected\n            let to = p[1]; // connected\n            // find everything connected to the starting point\n            let local = {};\n            local[this._pointKey(from)] = from;\n            this._findConnected(local, notConnected, [from], true, value);\n            // connect to a connected cell\n            let tunnelFn = (this._options.topology == 6 ? this._tunnelToConnected6 : this._tunnelToConnected);\n            tunnelFn.call(this, to, from, connected, notConnected, value, connectionCallback);\n            // now all of local is connected\n            for (let k in local) {\n                let pp = local[k];\n                this._map[pp[0]][pp[1]] = value;\n                connected[k] = pp;\n                delete notConnected[k];\n            }\n        }\n        callback && this._serviceCallback(callback);\n    }\n    /**\n     * Find random points to connect. Search for the closest point in the larger space.\n     * This is to minimize the length of the passage while maintaining good performance.\n     */\n    _getFromTo(connected, notConnected) {\n        let from = [0, 0], to = [0, 0], d;\n        let connectedKeys = Object.keys(connected);\n        let notConnectedKeys = Object.keys(notConnected);\n        for (let i = 0; i < 5; i++) {\n            if (connectedKeys.length < notConnectedKeys.length) {\n                let keys = connectedKeys;\n                to = connected[keys[RNG.getUniformInt(0, keys.length - 1)]];\n                from = this._getClosest(to, notConnected);\n            }\n            else {\n                let keys = notConnectedKeys;\n                from = notConnected[keys[RNG.getUniformInt(0, keys.length - 1)]];\n                to = this._getClosest(from, connected);\n            }\n            d = (from[0] - to[0]) * (from[0] - to[0]) + (from[1] - to[1]) * (from[1] - to[1]);\n            if (d < 64) {\n                break;\n            }\n        }\n        // console.log(\">>> connected=\" + to + \" notConnected=\" + from + \" dist=\" + d);\n        return [from, to];\n    }\n    _getClosest(point, space) {\n        let minPoint = null;\n        let minDist = null;\n        for (let k in space) {\n            let p = space[k];\n            let d = (p[0] - point[0]) * (p[0] - point[0]) + (p[1] - point[1]) * (p[1] - point[1]);\n            if (minDist == null || d < minDist) {\n                minDist = d;\n                minPoint = p;\n            }\n        }\n        return minPoint;\n    }\n    _findConnected(connected, notConnected, stack, keepNotConnected, value) {\n        while (stack.length > 0) {\n            let p = stack.splice(0, 1)[0];\n            let tests;\n            if (this._options.topology == 6) {\n                tests = [\n                    [p[0] + 2, p[1]],\n                    [p[0] + 1, p[1] - 1],\n                    [p[0] - 1, p[1] - 1],\n                    [p[0] - 2, p[1]],\n                    [p[0] - 1, p[1] + 1],\n                    [p[0] + 1, p[1] + 1],\n                ];\n            }\n            else {\n                tests = [\n                    [p[0] + 1, p[1]],\n                    [p[0] - 1, p[1]],\n                    [p[0], p[1] + 1],\n                    [p[0], p[1] - 1]\n                ];\n            }\n            for (let i = 0; i < tests.length; i++) {\n                let key = this._pointKey(tests[i]);\n                if (connected[key] == null && this._freeSpace(tests[i][0], tests[i][1], value)) {\n                    connected[key] = tests[i];\n                    if (!keepNotConnected) {\n                        delete notConnected[key];\n                    }\n                    stack.push(tests[i]);\n                }\n            }\n        }\n    }\n    _tunnelToConnected(to, from, connected, notConnected, value, connectionCallback) {\n        let a, b;\n        if (from[0] < to[0]) {\n            a = from;\n            b = to;\n        }\n        else {\n            a = to;\n            b = from;\n        }\n        for (let xx = a[0]; xx <= b[0]; xx++) {\n            this._map[xx][a[1]] = value;\n            let p = [xx, a[1]];\n            let pkey = this._pointKey(p);\n            connected[pkey] = p;\n            delete notConnected[pkey];\n        }\n        if (connectionCallback && a[0] < b[0]) {\n            connectionCallback(a, [b[0], a[1]]);\n        }\n        // x is now fixed\n        let x = b[0];\n        if (from[1] < to[1]) {\n            a = from;\n            b = to;\n        }\n        else {\n            a = to;\n            b = from;\n        }\n        for (let yy = a[1]; yy < b[1]; yy++) {\n            this._map[x][yy] = value;\n            let p = [x, yy];\n            let pkey = this._pointKey(p);\n            connected[pkey] = p;\n            delete notConnected[pkey];\n        }\n        if (connectionCallback && a[1] < b[1]) {\n            connectionCallback([b[0], a[1]], [b[0], b[1]]);\n        }\n    }\n    _tunnelToConnected6(to, from, connected, notConnected, value, connectionCallback) {\n        let a, b;\n        if (from[0] < to[0]) {\n            a = from;\n            b = to;\n        }\n        else {\n            a = to;\n            b = from;\n        }\n        // tunnel diagonally until horizontally level\n        let xx = a[0];\n        let yy = a[1];\n        while (!(xx == b[0] && yy == b[1])) {\n            let stepWidth = 2;\n            if (yy < b[1]) {\n                yy++;\n                stepWidth = 1;\n            }\n            else if (yy > b[1]) {\n                yy--;\n                stepWidth = 1;\n            }\n            if (xx < b[0]) {\n                xx += stepWidth;\n            }\n            else if (xx > b[0]) {\n                xx -= stepWidth;\n            }\n            else if (b[1] % 2) {\n                // Won't step outside map if destination on is map's right edge\n                xx -= stepWidth;\n            }\n            else {\n                // ditto for left edge\n                xx += stepWidth;\n            }\n            this._map[xx][yy] = value;\n            let p = [xx, yy];\n            let pkey = this._pointKey(p);\n            connected[pkey] = p;\n            delete notConnected[pkey];\n        }\n        if (connectionCallback) {\n            connectionCallback(from, to);\n        }\n    }\n    _freeSpace(x, y, value) {\n        return x >= 0 && x < this._width && y >= 0 && y < this._height && this._map[x][y] == value;\n    }\n    _pointKey(p) { return p[0] + \".\" + p[1]; }\n}\n","import Dungeon from \"./dungeon.js\";\nimport { Room, Corridor } from \"./features.js\";\nimport RNG from \"../rng.js\";\nimport { DIRS } from \"../constants.js\";\nconst FEATURES = {\n    \"room\": Room,\n    \"corridor\": Corridor\n};\n/**\n * Random dungeon generator using human-like digging patterns.\n * Heavily based on Mike Anderson's ideas from the \"Tyrant\" algo, mentioned at\n * http://www.roguebasin.roguelikedevelopment.org/index.php?title=Dungeon-Building_Algorithm.\n */\nexport default class Digger extends Dungeon {\n    constructor(width, height, options = {}) {\n        super(width, height);\n        this._options = Object.assign({\n            roomWidth: [3, 9],\n            roomHeight: [3, 5],\n            corridorLength: [3, 10],\n            dugPercentage: 0.2,\n            timeLimit: 1000 /* we stop after this much time has passed (msec) */\n        }, options);\n        this._features = {\n            \"room\": 4,\n            \"corridor\": 4\n        };\n        this._map = [];\n        this._featureAttempts = 20; /* how many times do we try to create a feature on a suitable wall */\n        this._walls = {}; /* these are available for digging */\n        this._dug = 0;\n        this._digCallback = this._digCallback.bind(this);\n        this._canBeDugCallback = this._canBeDugCallback.bind(this);\n        this._isWallCallback = this._isWallCallback.bind(this);\n        this._priorityWallCallback = this._priorityWallCallback.bind(this);\n    }\n    create(callback) {\n        this._rooms = [];\n        this._corridors = [];\n        this._map = this._fillMap(1);\n        this._walls = {};\n        this._dug = 0;\n        let area = (this._width - 2) * (this._height - 2);\n        this._firstRoom();\n        let t1 = Date.now();\n        let priorityWalls;\n        do {\n            priorityWalls = 0;\n            let t2 = Date.now();\n            if (t2 - t1 > this._options.timeLimit) {\n                break;\n            }\n            /* find a good wall */\n            let wall = this._findWall();\n            if (!wall) {\n                break;\n            } /* no more walls */\n            let parts = wall.split(\",\");\n            let x = parseInt(parts[0]);\n            let y = parseInt(parts[1]);\n            let dir = this._getDiggingDirection(x, y);\n            if (!dir) {\n                continue;\n            } /* this wall is not suitable */\n            //\t\tconsole.log(\"wall\", x, y);\n            /* try adding a feature */\n            let featureAttempts = 0;\n            do {\n                featureAttempts++;\n                if (this._tryFeature(x, y, dir[0], dir[1])) { /* feature added */\n                    //if (this._rooms.length + this._corridors.length == 2) { this._rooms[0].addDoor(x, y); } /* first room oficially has doors */\n                    this._removeSurroundingWalls(x, y);\n                    this._removeSurroundingWalls(x - dir[0], y - dir[1]);\n                    break;\n                }\n            } while (featureAttempts < this._featureAttempts);\n            for (let id in this._walls) {\n                if (this._walls[id] > 1) {\n                    priorityWalls++;\n                }\n            }\n        } while (this._dug / area < this._options.dugPercentage || priorityWalls); /* fixme number of priority walls */\n        this._addDoors();\n        if (callback) {\n            for (let i = 0; i < this._width; i++) {\n                for (let j = 0; j < this._height; j++) {\n                    callback(i, j, this._map[i][j]);\n                }\n            }\n        }\n        this._walls = {};\n        this._map = [];\n        return this;\n    }\n    _digCallback(x, y, value) {\n        if (value == 0 || value == 2) { /* empty */\n            this._map[x][y] = 0;\n            this._dug++;\n        }\n        else { /* wall */\n            this._walls[x + \",\" + y] = 1;\n        }\n    }\n    _isWallCallback(x, y) {\n        if (x < 0 || y < 0 || x >= this._width || y >= this._height) {\n            return false;\n        }\n        return (this._map[x][y] == 1);\n    }\n    _canBeDugCallback(x, y) {\n        if (x < 1 || y < 1 || x + 1 >= this._width || y + 1 >= this._height) {\n            return false;\n        }\n        return (this._map[x][y] == 1);\n    }\n    _priorityWallCallback(x, y) { this._walls[x + \",\" + y] = 2; }\n    ;\n    _firstRoom() {\n        let cx = Math.floor(this._width / 2);\n        let cy = Math.floor(this._height / 2);\n        let room = Room.createRandomCenter(cx, cy, this._options);\n        this._rooms.push(room);\n        room.create(this._digCallback);\n    }\n    /**\n     * Get a suitable wall\n     */\n    _findWall() {\n        let prio1 = [];\n        let prio2 = [];\n        for (let id in this._walls) {\n            let prio = this._walls[id];\n            if (prio == 2) {\n                prio2.push(id);\n            }\n            else {\n                prio1.push(id);\n            }\n        }\n        let arr = (prio2.length ? prio2 : prio1);\n        if (!arr.length) {\n            return null;\n        } /* no walls :/ */\n        let id = RNG.getItem(arr.sort()); // sort to make the order deterministic\n        delete this._walls[id];\n        return id;\n    }\n    /**\n     * Tries adding a feature\n     * @returns {bool} was this a successful try?\n     */\n    _tryFeature(x, y, dx, dy) {\n        let featureName = RNG.getWeightedValue(this._features);\n        let ctor = FEATURES[featureName];\n        let feature = ctor.createRandomAt(x, y, dx, dy, this._options);\n        if (!feature.isValid(this._isWallCallback, this._canBeDugCallback)) {\n            //\t\tconsole.log(\"not valid\");\n            //\t\tfeature.debug();\n            return false;\n        }\n        feature.create(this._digCallback);\n        //\tfeature.debug();\n        if (feature instanceof Room) {\n            this._rooms.push(feature);\n        }\n        if (feature instanceof Corridor) {\n            feature.createPriorityWalls(this._priorityWallCallback);\n            this._corridors.push(feature);\n        }\n        return true;\n    }\n    _removeSurroundingWalls(cx, cy) {\n        let deltas = DIRS[4];\n        for (let i = 0; i < deltas.length; i++) {\n            let delta = deltas[i];\n            let x = cx + delta[0];\n            let y = cy + delta[1];\n            delete this._walls[x + \",\" + y];\n            x = cx + 2 * delta[0];\n            y = cy + 2 * delta[1];\n            delete this._walls[x + \",\" + y];\n        }\n    }\n    /**\n     * Returns vector in \"digging\" direction, or false, if this does not exist (or is not unique)\n     */\n    _getDiggingDirection(cx, cy) {\n        if (cx <= 0 || cy <= 0 || cx >= this._width - 1 || cy >= this._height - 1) {\n            return null;\n        }\n        let result = null;\n        let deltas = DIRS[4];\n        for (let i = 0; i < deltas.length; i++) {\n            let delta = deltas[i];\n            let x = cx + delta[0];\n            let y = cy + delta[1];\n            if (!this._map[x][y]) { /* there already is another empty neighbor! */\n                if (result) {\n                    return null;\n                }\n                result = delta;\n            }\n        }\n        /* no empty neighbor */\n        if (!result) {\n            return null;\n        }\n        return [-result[0], -result[1]];\n    }\n    /**\n     * Find empty spaces surrounding rooms, and apply doors.\n     */\n    _addDoors() {\n        let data = this._map;\n        function isWallCallback(x, y) {\n            return (data[x][y] == 1);\n        }\n        ;\n        for (let i = 0; i < this._rooms.length; i++) {\n            let room = this._rooms[i];\n            room.clearDoors();\n            room.addDoors(isWallCallback);\n        }\n    }\n}\n","import Map from \"./map.js\";\nimport RNG from \"../rng.js\";\n/**\n * Join lists with \"i\" and \"i+1\"\n */\nfunction addToList(i, L, R) {\n    R[L[i + 1]] = R[i];\n    L[R[i]] = L[i + 1];\n    R[i] = i + 1;\n    L[i + 1] = i;\n}\n/**\n * Remove \"i\" from its list\n */\nfunction removeFromList(i, L, R) {\n    R[L[i]] = R[i];\n    L[R[i]] = L[i];\n    R[i] = i;\n    L[i] = i;\n}\n/**\n * Maze generator - Eller's algorithm\n * See http://homepages.cwi.nl/~tromp/maze.html for explanation\n */\nexport default class EllerMaze extends Map {\n    create(callback) {\n        let map = this._fillMap(1);\n        let w = Math.ceil((this._width - 2) / 2);\n        let rand = 9 / 24;\n        let L = [];\n        let R = [];\n        for (let i = 0; i < w; i++) {\n            L.push(i);\n            R.push(i);\n        }\n        L.push(w - 1); /* fake stop-block at the right side */\n        let j;\n        for (j = 1; j + 3 < this._height; j += 2) {\n            /* one row */\n            for (let i = 0; i < w; i++) {\n                /* cell coords (will be always empty) */\n                let x = 2 * i + 1;\n                let y = j;\n                map[x][y] = 0;\n                /* right connection */\n                if (i != L[i + 1] && RNG.getUniform() > rand) {\n                    addToList(i, L, R);\n                    map[x + 1][y] = 0;\n                }\n                /* bottom connection */\n                if (i != L[i] && RNG.getUniform() > rand) {\n                    /* remove connection */\n                    removeFromList(i, L, R);\n                }\n                else {\n                    /* create connection */\n                    map[x][y + 1] = 0;\n                }\n            }\n        }\n        /* last row */\n        for (let i = 0; i < w; i++) {\n            /* cell coords (will be always empty) */\n            let x = 2 * i + 1;\n            let y = j;\n            map[x][y] = 0;\n            /* right connection */\n            if (i != L[i + 1] && (i == L[i] || RNG.getUniform() > rand)) {\n                /* dig right also if the cell is separated, so it gets connected to the rest of maze */\n                addToList(i, L, R);\n                map[x + 1][y] = 0;\n            }\n            removeFromList(i, L, R);\n        }\n        for (let i = 0; i < this._width; i++) {\n            for (let j = 0; j < this._height; j++) {\n                callback(i, j, map[i][j]);\n            }\n        }\n        return this;\n    }\n}\n","import Map from \"./map.js\";\nimport RNG from \"../rng.js\";\n/**\n * @class Recursively divided maze, http://en.wikipedia.org/wiki/Maze_generation_algorithm#Recursive_division_method\n * @augments ROT.Map\n */\nexport default class DividedMaze extends Map {\n    constructor() {\n        super(...arguments);\n        this._stack = [];\n        this._map = [];\n    }\n    create(callback) {\n        let w = this._width;\n        let h = this._height;\n        this._map = [];\n        for (let i = 0; i < w; i++) {\n            this._map.push([]);\n            for (let j = 0; j < h; j++) {\n                let border = (i == 0 || j == 0 || i + 1 == w || j + 1 == h);\n                this._map[i].push(border ? 1 : 0);\n            }\n        }\n        this._stack = [\n            [1, 1, w - 2, h - 2]\n        ];\n        this._process();\n        for (let i = 0; i < w; i++) {\n            for (let j = 0; j < h; j++) {\n                callback(i, j, this._map[i][j]);\n            }\n        }\n        this._map = [];\n        return this;\n    }\n    _process() {\n        while (this._stack.length) {\n            let room = this._stack.shift(); /* [left, top, right, bottom] */\n            this._partitionRoom(room);\n        }\n    }\n    _partitionRoom(room) {\n        let availX = [];\n        let availY = [];\n        for (let i = room[0] + 1; i < room[2]; i++) {\n            let top = this._map[i][room[1] - 1];\n            let bottom = this._map[i][room[3] + 1];\n            if (top && bottom && !(i % 2)) {\n                availX.push(i);\n            }\n        }\n        for (let j = room[1] + 1; j < room[3]; j++) {\n            let left = this._map[room[0] - 1][j];\n            let right = this._map[room[2] + 1][j];\n            if (left && right && !(j % 2)) {\n                availY.push(j);\n            }\n        }\n        if (!availX.length || !availY.length) {\n            return;\n        }\n        let x = RNG.getItem(availX);\n        let y = RNG.getItem(availY);\n        this._map[x][y] = 1;\n        let walls = [];\n        let w = [];\n        walls.push(w); /* left part */\n        for (let i = room[0]; i < x; i++) {\n            this._map[i][y] = 1;\n            w.push([i, y]);\n        }\n        w = [];\n        walls.push(w); /* right part */\n        for (let i = x + 1; i <= room[2]; i++) {\n            this._map[i][y] = 1;\n            w.push([i, y]);\n        }\n        w = [];\n        walls.push(w); /* top part */\n        for (let j = room[1]; j < y; j++) {\n            this._map[x][j] = 1;\n            w.push([x, j]);\n        }\n        w = [];\n        walls.push(w); /* bottom part */\n        for (let j = y + 1; j <= room[3]; j++) {\n            this._map[x][j] = 1;\n            w.push([x, j]);\n        }\n        let solid = RNG.getItem(walls);\n        for (let i = 0; i < walls.length; i++) {\n            let w = walls[i];\n            if (w == solid) {\n                continue;\n            }\n            let hole = RNG.getItem(w);\n            this._map[hole[0]][hole[1]] = 0;\n        }\n        this._stack.push([room[0], room[1], x - 1, y - 1]); /* left top */\n        this._stack.push([x + 1, room[1], room[2], y - 1]); /* right top */\n        this._stack.push([room[0], y + 1, x - 1, room[3]]); /* left bottom */\n        this._stack.push([x + 1, y + 1, room[2], room[3]]); /* right bottom */\n    }\n}\n","import Map from \"./map.js\";\nimport RNG from \"../rng.js\";\n/**\n * Icey's Maze generator\n * See http://www.roguebasin.roguelikedevelopment.org/index.php?title=Simple_maze for explanation\n */\nexport default class IceyMaze extends Map {\n    constructor(width, height, regularity = 0) {\n        super(width, height);\n        this._regularity = regularity;\n        this._map = [];\n    }\n    create(callback) {\n        let width = this._width;\n        let height = this._height;\n        let map = this._fillMap(1);\n        width -= (width % 2 ? 1 : 2);\n        height -= (height % 2 ? 1 : 2);\n        let cx = 0;\n        let cy = 0;\n        let nx = 0;\n        let ny = 0;\n        let done = 0;\n        let blocked = false;\n        let dirs = [\n            [0, 0],\n            [0, 0],\n            [0, 0],\n            [0, 0]\n        ];\n        do {\n            cx = 1 + 2 * Math.floor(RNG.getUniform() * (width - 1) / 2);\n            cy = 1 + 2 * Math.floor(RNG.getUniform() * (height - 1) / 2);\n            if (!done) {\n                map[cx][cy] = 0;\n            }\n            if (!map[cx][cy]) {\n                this._randomize(dirs);\n                do {\n                    if (Math.floor(RNG.getUniform() * (this._regularity + 1)) == 0) {\n                        this._randomize(dirs);\n                    }\n                    blocked = true;\n                    for (let i = 0; i < 4; i++) {\n                        nx = cx + dirs[i][0] * 2;\n                        ny = cy + dirs[i][1] * 2;\n                        if (this._isFree(map, nx, ny, width, height)) {\n                            map[nx][ny] = 0;\n                            map[cx + dirs[i][0]][cy + dirs[i][1]] = 0;\n                            cx = nx;\n                            cy = ny;\n                            blocked = false;\n                            done++;\n                            break;\n                        }\n                    }\n                } while (!blocked);\n            }\n        } while (done + 1 < width * height / 4);\n        for (let i = 0; i < this._width; i++) {\n            for (let j = 0; j < this._height; j++) {\n                callback(i, j, map[i][j]);\n            }\n        }\n        this._map = [];\n        return this;\n    }\n    _randomize(dirs) {\n        for (let i = 0; i < 4; i++) {\n            dirs[i][0] = 0;\n            dirs[i][1] = 0;\n        }\n        switch (Math.floor(RNG.getUniform() * 4)) {\n            case 0:\n                dirs[0][0] = -1;\n                dirs[1][0] = 1;\n                dirs[2][1] = -1;\n                dirs[3][1] = 1;\n                break;\n            case 1:\n                dirs[3][0] = -1;\n                dirs[2][0] = 1;\n                dirs[1][1] = -1;\n                dirs[0][1] = 1;\n                break;\n            case 2:\n                dirs[2][0] = -1;\n                dirs[3][0] = 1;\n                dirs[0][1] = -1;\n                dirs[1][1] = 1;\n                break;\n            case 3:\n                dirs[1][0] = -1;\n                dirs[0][0] = 1;\n                dirs[3][1] = -1;\n                dirs[2][1] = 1;\n                break;\n        }\n    }\n    _isFree(map, x, y, width, height) {\n        if (x < 1 || y < 1 || x >= width || y >= height) {\n            return false;\n        }\n        return map[x][y];\n    }\n}\n","import Map from \"./map.js\";\nimport RNG from \"../rng.js\";\nimport { DIRS } from \"../constants.js\";\n/**\n * Dungeon generator which uses the \"orginal\" Rogue dungeon generation algorithm. See http://kuoi.com/~kamikaze/GameDesign/art07_rogue_dungeon.php\n * @author hyakugei\n */\nexport default class Rogue extends Map {\n    constructor(width, height, options) {\n        super(width, height);\n        this.map = [];\n        this.rooms = [];\n        this.connectedCells = [];\n        options = Object.assign({\n            cellWidth: 3,\n            cellHeight: 3 //     ie. as an array with min-max values for each direction....\n        }, options);\n        /*\n        Set the room sizes according to the over-all width of the map,\n        and the cell sizes.\n        */\n        if (!options.hasOwnProperty(\"roomWidth\")) {\n            options[\"roomWidth\"] = this._calculateRoomSize(this._width, options[\"cellWidth\"]);\n        }\n        if (!options.hasOwnProperty(\"roomHeight\")) {\n            options[\"roomHeight\"] = this._calculateRoomSize(this._height, options[\"cellHeight\"]);\n        }\n        this._options = options;\n    }\n    create(callback) {\n        this.map = this._fillMap(1);\n        this.rooms = [];\n        this.connectedCells = [];\n        this._initRooms();\n        this._connectRooms();\n        this._connectUnconnectedRooms();\n        this._createRandomRoomConnections();\n        this._createRooms();\n        this._createCorridors();\n        if (callback) {\n            for (let i = 0; i < this._width; i++) {\n                for (let j = 0; j < this._height; j++) {\n                    callback(i, j, this.map[i][j]);\n                }\n            }\n        }\n        return this;\n    }\n    _calculateRoomSize(size, cell) {\n        let max = Math.floor((size / cell) * 0.8);\n        let min = Math.floor((size / cell) * 0.25);\n        if (min < 2) {\n            min = 2;\n        }\n        if (max < 2) {\n            max = 2;\n        }\n        return [min, max];\n    }\n    _initRooms() {\n        // create rooms array. This is the \"grid\" list from the algo.\n        for (let i = 0; i < this._options.cellWidth; i++) {\n            this.rooms.push([]);\n            for (let j = 0; j < this._options.cellHeight; j++) {\n                this.rooms[i].push({ \"x\": 0, \"y\": 0, \"width\": 0, \"height\": 0, \"connections\": [], \"cellx\": i, \"celly\": j });\n            }\n        }\n    }\n    _connectRooms() {\n        //pick random starting grid\n        let cgx = RNG.getUniformInt(0, this._options.cellWidth - 1);\n        let cgy = RNG.getUniformInt(0, this._options.cellHeight - 1);\n        let idx;\n        let ncgx;\n        let ncgy;\n        let found = false;\n        let room;\n        let otherRoom;\n        let dirToCheck;\n        // find  unconnected neighbour cells\n        do {\n            //dirToCheck = [0, 1, 2, 3, 4, 5, 6, 7];\n            dirToCheck = [0, 2, 4, 6];\n            dirToCheck = RNG.shuffle(dirToCheck);\n            do {\n                found = false;\n                idx = dirToCheck.pop();\n                ncgx = cgx + DIRS[8][idx][0];\n                ncgy = cgy + DIRS[8][idx][1];\n                if (ncgx < 0 || ncgx >= this._options.cellWidth) {\n                    continue;\n                }\n                if (ncgy < 0 || ncgy >= this._options.cellHeight) {\n                    continue;\n                }\n                room = this.rooms[cgx][cgy];\n                if (room[\"connections\"].length > 0) {\n                    // as long as this room doesn't already coonect to me, we are ok with it.\n                    if (room[\"connections\"][0][0] == ncgx && room[\"connections\"][0][1] == ncgy) {\n                        break;\n                    }\n                }\n                otherRoom = this.rooms[ncgx][ncgy];\n                if (otherRoom[\"connections\"].length == 0) {\n                    otherRoom[\"connections\"].push([cgx, cgy]);\n                    this.connectedCells.push([ncgx, ncgy]);\n                    cgx = ncgx;\n                    cgy = ncgy;\n                    found = true;\n                }\n            } while (dirToCheck.length > 0 && found == false);\n        } while (dirToCheck.length > 0);\n    }\n    _connectUnconnectedRooms() {\n        //While there are unconnected rooms, try to connect them to a random connected neighbor\n        //(if a room has no connected neighbors yet, just keep cycling, you'll fill out to it eventually).\n        let cw = this._options.cellWidth;\n        let ch = this._options.cellHeight;\n        this.connectedCells = RNG.shuffle(this.connectedCells);\n        let room;\n        let otherRoom;\n        let validRoom;\n        for (let i = 0; i < this._options.cellWidth; i++) {\n            for (let j = 0; j < this._options.cellHeight; j++) {\n                room = this.rooms[i][j];\n                if (room[\"connections\"].length == 0) {\n                    let directions = [0, 2, 4, 6];\n                    directions = RNG.shuffle(directions);\n                    validRoom = false;\n                    do {\n                        let dirIdx = directions.pop();\n                        let newI = i + DIRS[8][dirIdx][0];\n                        let newJ = j + DIRS[8][dirIdx][1];\n                        if (newI < 0 || newI >= cw || newJ < 0 || newJ >= ch) {\n                            continue;\n                        }\n                        otherRoom = this.rooms[newI][newJ];\n                        validRoom = true;\n                        if (otherRoom[\"connections\"].length == 0) {\n                            break;\n                        }\n                        for (let k = 0; k < otherRoom[\"connections\"].length; k++) {\n                            if (otherRoom[\"connections\"][k][0] == i && otherRoom[\"connections\"][k][1] == j) {\n                                validRoom = false;\n                                break;\n                            }\n                        }\n                        if (validRoom) {\n                            break;\n                        }\n                    } while (directions.length);\n                    if (validRoom) {\n                        room[\"connections\"].push([otherRoom[\"cellx\"], otherRoom[\"celly\"]]);\n                    }\n                    else {\n                        console.log(\"-- Unable to connect room.\");\n                    }\n                }\n            }\n        }\n    }\n    _createRandomRoomConnections() {\n        // Empty for now.\n    }\n    _createRooms() {\n        let w = this._width;\n        let h = this._height;\n        let cw = this._options.cellWidth;\n        let ch = this._options.cellHeight;\n        let cwp = Math.floor(this._width / cw);\n        let chp = Math.floor(this._height / ch);\n        let roomw;\n        let roomh;\n        let roomWidth = this._options[\"roomWidth\"];\n        let roomHeight = this._options[\"roomHeight\"];\n        let sx;\n        let sy;\n        let otherRoom;\n        for (let i = 0; i < cw; i++) {\n            for (let j = 0; j < ch; j++) {\n                sx = cwp * i;\n                sy = chp * j;\n                if (sx == 0) {\n                    sx = 1;\n                }\n                if (sy == 0) {\n                    sy = 1;\n                }\n                roomw = RNG.getUniformInt(roomWidth[0], roomWidth[1]);\n                roomh = RNG.getUniformInt(roomHeight[0], roomHeight[1]);\n                if (j > 0) {\n                    otherRoom = this.rooms[i][j - 1];\n                    while (sy - (otherRoom[\"y\"] + otherRoom[\"height\"]) < 3) {\n                        sy++;\n                    }\n                }\n                if (i > 0) {\n                    otherRoom = this.rooms[i - 1][j];\n                    while (sx - (otherRoom[\"x\"] + otherRoom[\"width\"]) < 3) {\n                        sx++;\n                    }\n                }\n                let sxOffset = Math.round(RNG.getUniformInt(0, cwp - roomw) / 2);\n                let syOffset = Math.round(RNG.getUniformInt(0, chp - roomh) / 2);\n                while (sx + sxOffset + roomw >= w) {\n                    if (sxOffset) {\n                        sxOffset--;\n                    }\n                    else {\n                        roomw--;\n                    }\n                }\n                while (sy + syOffset + roomh >= h) {\n                    if (syOffset) {\n                        syOffset--;\n                    }\n                    else {\n                        roomh--;\n                    }\n                }\n                sx = sx + sxOffset;\n                sy = sy + syOffset;\n                this.rooms[i][j][\"x\"] = sx;\n                this.rooms[i][j][\"y\"] = sy;\n                this.rooms[i][j][\"width\"] = roomw;\n                this.rooms[i][j][\"height\"] = roomh;\n                for (let ii = sx; ii < sx + roomw; ii++) {\n                    for (let jj = sy; jj < sy + roomh; jj++) {\n                        this.map[ii][jj] = 0;\n                    }\n                }\n            }\n        }\n    }\n    _getWallPosition(aRoom, aDirection) {\n        let rx;\n        let ry;\n        let door;\n        if (aDirection == 1 || aDirection == 3) {\n            rx = RNG.getUniformInt(aRoom[\"x\"] + 1, aRoom[\"x\"] + aRoom[\"width\"] - 2);\n            if (aDirection == 1) {\n                ry = aRoom[\"y\"] - 2;\n                door = ry + 1;\n            }\n            else {\n                ry = aRoom[\"y\"] + aRoom[\"height\"] + 1;\n                door = ry - 1;\n            }\n            this.map[rx][door] = 0; // i'm not setting a specific 'door' tile value right now, just empty space.\n        }\n        else {\n            ry = RNG.getUniformInt(aRoom[\"y\"] + 1, aRoom[\"y\"] + aRoom[\"height\"] - 2);\n            if (aDirection == 2) {\n                rx = aRoom[\"x\"] + aRoom[\"width\"] + 1;\n                door = rx - 1;\n            }\n            else {\n                rx = aRoom[\"x\"] - 2;\n                door = rx + 1;\n            }\n            this.map[door][ry] = 0; // i'm not setting a specific 'door' tile value right now, just empty space.\n        }\n        return [rx, ry];\n    }\n    _drawCorridor(startPosition, endPosition) {\n        let xOffset = endPosition[0] - startPosition[0];\n        let yOffset = endPosition[1] - startPosition[1];\n        let xpos = startPosition[0];\n        let ypos = startPosition[1];\n        let tempDist;\n        let xDir;\n        let yDir;\n        let move; // 2 element array, element 0 is the direction, element 1 is the total value to move.\n        let moves = []; // a list of 2 element arrays\n        let xAbs = Math.abs(xOffset);\n        let yAbs = Math.abs(yOffset);\n        let percent = RNG.getUniform(); // used to split the move at different places along the long axis\n        let firstHalf = percent;\n        let secondHalf = 1 - percent;\n        xDir = xOffset > 0 ? 2 : 6;\n        yDir = yOffset > 0 ? 4 : 0;\n        if (xAbs < yAbs) {\n            // move firstHalf of the y offset\n            tempDist = Math.ceil(yAbs * firstHalf);\n            moves.push([yDir, tempDist]);\n            // move all the x offset\n            moves.push([xDir, xAbs]);\n            // move sendHalf of the  y offset\n            tempDist = Math.floor(yAbs * secondHalf);\n            moves.push([yDir, tempDist]);\n        }\n        else {\n            //  move firstHalf of the x offset\n            tempDist = Math.ceil(xAbs * firstHalf);\n            moves.push([xDir, tempDist]);\n            // move all the y offset\n            moves.push([yDir, yAbs]);\n            // move secondHalf of the x offset.\n            tempDist = Math.floor(xAbs * secondHalf);\n            moves.push([xDir, tempDist]);\n        }\n        this.map[xpos][ypos] = 0;\n        while (moves.length > 0) {\n            move = moves.pop();\n            while (move[1] > 0) {\n                xpos += DIRS[8][move[0]][0];\n                ypos += DIRS[8][move[0]][1];\n                this.map[xpos][ypos] = 0;\n                move[1] = move[1] - 1;\n            }\n        }\n    }\n    _createCorridors() {\n        // Draw Corridors between connected rooms\n        let cw = this._options.cellWidth;\n        let ch = this._options.cellHeight;\n        let room;\n        let connection;\n        let otherRoom;\n        let wall;\n        let otherWall;\n        for (let i = 0; i < cw; i++) {\n            for (let j = 0; j < ch; j++) {\n                room = this.rooms[i][j];\n                for (let k = 0; k < room[\"connections\"].length; k++) {\n                    connection = room[\"connections\"][k];\n                    otherRoom = this.rooms[connection[0]][connection[1]];\n                    // figure out what wall our corridor will start one.\n                    // figure out what wall our corridor will end on.\n                    if (otherRoom[\"cellx\"] > room[\"cellx\"]) {\n                        wall = 2;\n                        otherWall = 4;\n                    }\n                    else if (otherRoom[\"cellx\"] < room[\"cellx\"]) {\n                        wall = 4;\n                        otherWall = 2;\n                    }\n                    else if (otherRoom[\"celly\"] > room[\"celly\"]) {\n                        wall = 3;\n                        otherWall = 1;\n                    }\n                    else {\n                        wall = 1;\n                        otherWall = 3;\n                    }\n                    this._drawCorridor(this._getWallPosition(room, wall), this._getWallPosition(otherRoom, otherWall));\n                }\n            }\n        }\n    }\n}\n","import Arena from \"./arena.js\";\nimport Uniform from \"./uniform.js\";\nimport Cellular from \"./cellular.js\";\nimport Digger from \"./digger.js\";\nimport EllerMaze from \"./ellermaze.js\";\nimport DividedMaze from \"./dividedmaze.js\";\nimport IceyMaze from \"./iceymaze.js\";\nimport Rogue from \"./rogue.js\";\nexport default { Arena, Uniform, Cellular, Digger, EllerMaze, DividedMaze, IceyMaze, Rogue };\n","/**\n * Base noise generator\n */\nexport default class Noise {\n}\n","import Noise from \"./noise.js\";\nimport RNG from \"../rng.js\";\nimport { mod } from \"../util.js\";\nconst F2 = 0.5 * (Math.sqrt(3) - 1);\nconst G2 = (3 - Math.sqrt(3)) / 6;\n/**\n * A simple 2d implementation of simplex noise by Ondrej Zara\n *\n * Based on a speed-improved simplex noise algorithm for 2D, 3D and 4D in Java.\n * Which is based on example code by Stefan Gustavson (stegu@itn.liu.se).\n * With Optimisations by Peter Eastman (peastman@drizzle.stanford.edu).\n * Better rank ordering method by Stefan Gustavson in 2012.\n */\nexport default class Simplex extends Noise {\n    /**\n     * @param gradients Random gradients\n     */\n    constructor(gradients = 256) {\n        super();\n        this._gradients = [\n            [0, -1],\n            [1, -1],\n            [1, 0],\n            [1, 1],\n            [0, 1],\n            [-1, 1],\n            [-1, 0],\n            [-1, -1]\n        ];\n        let permutations = [];\n        for (let i = 0; i < gradients; i++) {\n            permutations.push(i);\n        }\n        permutations = RNG.shuffle(permutations);\n        this._perms = [];\n        this._indexes = [];\n        for (let i = 0; i < 2 * gradients; i++) {\n            this._perms.push(permutations[i % gradients]);\n            this._indexes.push(this._perms[i] % this._gradients.length);\n        }\n    }\n    get(xin, yin) {\n        let perms = this._perms;\n        let indexes = this._indexes;\n        let count = perms.length / 2;\n        let n0 = 0, n1 = 0, n2 = 0, gi; // Noise contributions from the three corners\n        // Skew the input space to determine which simplex cell we're in\n        let s = (xin + yin) * F2; // Hairy factor for 2D\n        let i = Math.floor(xin + s);\n        let j = Math.floor(yin + s);\n        let t = (i + j) * G2;\n        let X0 = i - t; // Unskew the cell origin back to (x,y) space\n        let Y0 = j - t;\n        let x0 = xin - X0; // The x,y distances from the cell origin\n        let y0 = yin - Y0;\n        // For the 2D case, the simplex shape is an equilateral triangle.\n        // Determine which simplex we are in.\n        let i1, j1; // Offsets for second (middle) corner of simplex in (i,j) coords\n        if (x0 > y0) {\n            i1 = 1;\n            j1 = 0;\n        }\n        else { // lower triangle, XY order: (0,0)->(1,0)->(1,1)\n            i1 = 0;\n            j1 = 1;\n        } // upper triangle, YX order: (0,0)->(0,1)->(1,1)\n        // A step of (1,0) in (i,j) means a step of (1-c,-c) in (x,y), and\n        // a step of (0,1) in (i,j) means a step of (-c,1-c) in (x,y), where\n        // c = (3-sqrt(3))/6\n        let x1 = x0 - i1 + G2; // Offsets for middle corner in (x,y) unskewed coords\n        let y1 = y0 - j1 + G2;\n        let x2 = x0 - 1 + 2 * G2; // Offsets for last corner in (x,y) unskewed coords\n        let y2 = y0 - 1 + 2 * G2;\n        // Work out the hashed gradient indices of the three simplex corners\n        let ii = mod(i, count);\n        let jj = mod(j, count);\n        // Calculate the contribution from the three corners\n        let t0 = 0.5 - x0 * x0 - y0 * y0;\n        if (t0 >= 0) {\n            t0 *= t0;\n            gi = indexes[ii + perms[jj]];\n            let grad = this._gradients[gi];\n            n0 = t0 * t0 * (grad[0] * x0 + grad[1] * y0);\n        }\n        let t1 = 0.5 - x1 * x1 - y1 * y1;\n        if (t1 >= 0) {\n            t1 *= t1;\n            gi = indexes[ii + i1 + perms[jj + j1]];\n            let grad = this._gradients[gi];\n            n1 = t1 * t1 * (grad[0] * x1 + grad[1] * y1);\n        }\n        let t2 = 0.5 - x2 * x2 - y2 * y2;\n        if (t2 >= 0) {\n            t2 *= t2;\n            gi = indexes[ii + 1 + perms[jj + 1]];\n            let grad = this._gradients[gi];\n            n2 = t2 * t2 * (grad[0] * x2 + grad[1] * y2);\n        }\n        // Add contributions from each corner to get the final noise value.\n        // The result is scaled to return values in the interval [-1,1].\n        return 70 * (n0 + n1 + n2);\n    }\n}\n","import Simplex from \"./simplex.js\";\nexport default { Simplex };\n","import { DIRS } from \"../constants.js\";\n/**\n * @class Abstract pathfinder\n * @param {int} toX Target X coord\n * @param {int} toY Target Y coord\n * @param {function} passableCallback Callback to determine map passability\n * @param {object} [options]\n * @param {int} [options.topology=8]\n */\nexport default class Path {\n    constructor(toX, toY, passableCallback, options = {}) {\n        this._toX = toX;\n        this._toY = toY;\n        this._passableCallback = passableCallback;\n        this._options = Object.assign({\n            topology: 8\n        }, options);\n        this._dirs = DIRS[this._options.topology];\n        if (this._options.topology == 8) { /* reorder dirs for more aesthetic result (vertical/horizontal first) */\n            this._dirs = [\n                this._dirs[0],\n                this._dirs[2],\n                this._dirs[4],\n                this._dirs[6],\n                this._dirs[1],\n                this._dirs[3],\n                this._dirs[5],\n                this._dirs[7]\n            ];\n        }\n    }\n    _getNeighbors(cx, cy) {\n        let result = [];\n        for (let i = 0; i < this._dirs.length; i++) {\n            let dir = this._dirs[i];\n            let x = cx + dir[0];\n            let y = cy + dir[1];\n            if (!this._passableCallback(x, y)) {\n                continue;\n            }\n            result.push([x, y]);\n        }\n        return result;\n    }\n}\n","import Path from \"./path.js\";\n/**\n * @class Simplified Dijkstra's algorithm: all edges have a value of 1\n * @augments ROT.Path\n * @see ROT.Path\n */\nexport default class Dijkstra extends Path {\n    constructor(toX, toY, passableCallback, options) {\n        super(toX, toY, passableCallback, options);\n        this._computed = {};\n        this._todo = [];\n        this._add(toX, toY, null);\n    }\n    /**\n     * Compute a path from a given point\n     * @see ROT.Path#compute\n     */\n    compute(fromX, fromY, callback) {\n        let key = fromX + \",\" + fromY;\n        if (!(key in this._computed)) {\n            this._compute(fromX, fromY);\n        }\n        if (!(key in this._computed)) {\n            return;\n        }\n        let item = this._computed[key];\n        while (item) {\n            callback(item.x, item.y);\n            item = item.prev;\n        }\n    }\n    /**\n     * Compute a non-cached value\n     */\n    _compute(fromX, fromY) {\n        while (this._todo.length) {\n            let item = this._todo.shift();\n            if (item.x == fromX && item.y == fromY) {\n                return;\n            }\n            let neighbors = this._getNeighbors(item.x, item.y);\n            for (let i = 0; i < neighbors.length; i++) {\n                let neighbor = neighbors[i];\n                let x = neighbor[0];\n                let y = neighbor[1];\n                let id = x + \",\" + y;\n                if (id in this._computed) {\n                    continue;\n                } /* already done */\n                this._add(x, y, item);\n            }\n        }\n    }\n    _add(x, y, prev) {\n        let obj = {\n            x: x,\n            y: y,\n            prev: prev\n        };\n        this._computed[x + \",\" + y] = obj;\n        this._todo.push(obj);\n    }\n}\n","import Path from \"./path.js\";\n/**\n * @class Simplified A* algorithm: all edges have a value of 1\n * @augments ROT.Path\n * @see ROT.Path\n */\nexport default class AStar extends Path {\n    constructor(toX, toY, passableCallback, options = {}) {\n        super(toX, toY, passableCallback, options);\n        this._todo = [];\n        this._done = {};\n    }\n    /**\n     * Compute a path from a given point\n     * @see ROT.Path#compute\n     */\n    compute(fromX, fromY, callback) {\n        this._todo = [];\n        this._done = {};\n        this._fromX = fromX;\n        this._fromY = fromY;\n        this._add(this._toX, this._toY, null);\n        while (this._todo.length) {\n            let item = this._todo.shift();\n            let id = item.x + \",\" + item.y;\n            if (id in this._done) {\n                continue;\n            }\n            this._done[id] = item;\n            if (item.x == fromX && item.y == fromY) {\n                break;\n            }\n            let neighbors = this._getNeighbors(item.x, item.y);\n            for (let i = 0; i < neighbors.length; i++) {\n                let neighbor = neighbors[i];\n                let x = neighbor[0];\n                let y = neighbor[1];\n                let id = x + \",\" + y;\n                if (id in this._done) {\n                    continue;\n                }\n                this._add(x, y, item);\n            }\n        }\n        let item = this._done[fromX + \",\" + fromY];\n        if (!item) {\n            return;\n        }\n        while (item) {\n            callback(item.x, item.y);\n            item = item.prev;\n        }\n    }\n    _add(x, y, prev) {\n        let h = this._distance(x, y);\n        let obj = {\n            x: x,\n            y: y,\n            prev: prev,\n            g: (prev ? prev.g + 1 : 0),\n            h: h\n        };\n        /* insert into priority queue */\n        let f = obj.g + obj.h;\n        for (let i = 0; i < this._todo.length; i++) {\n            let item = this._todo[i];\n            let itemF = item.g + item.h;\n            if (f < itemF || (f == itemF && h < item.h)) {\n                this._todo.splice(i, 0, obj);\n                return;\n            }\n        }\n        this._todo.push(obj);\n    }\n    _distance(x, y) {\n        switch (this._options.topology) {\n            case 4:\n                return (Math.abs(x - this._fromX) + Math.abs(y - this._fromY));\n                break;\n            case 6:\n                let dx = Math.abs(x - this._fromX);\n                let dy = Math.abs(y - this._fromY);\n                return dy + Math.max(0, (dx - dy) / 2);\n                break;\n            case 8:\n                return Math.max(Math.abs(x - this._fromX), Math.abs(y - this._fromY));\n                break;\n        }\n    }\n}\n","import Dijkstra from \"./dijkstra.js\";\nimport AStar from \"./astar.js\";\nexport default { Dijkstra, AStar };\n","/**\n * @class Asynchronous main loop\n * @param {ROT.Scheduler} scheduler\n */\nexport default class Engine {\n    constructor(scheduler) {\n        this._scheduler = scheduler;\n        this._lock = 1;\n    }\n    /**\n     * Start the main loop. When this call returns, the loop is locked.\n     */\n    start() { return this.unlock(); }\n    /**\n     * Interrupt the engine by an asynchronous action\n     */\n    lock() {\n        this._lock++;\n        return this;\n    }\n    /**\n     * Resume execution (paused by a previous lock)\n     */\n    unlock() {\n        if (!this._lock) {\n            throw new Error(\"Cannot unlock unlocked engine\");\n        }\n        this._lock--;\n        while (!this._lock) {\n            let actor = this._scheduler.next();\n            if (!actor) {\n                return this.lock();\n            } /* no actors */\n            let result = actor.act();\n            if (result && result.then) { /* actor returned a \"thenable\", looks like a Promise */\n                this.lock();\n                result.then(this.unlock.bind(this));\n            }\n        }\n        return this;\n    }\n}\n","import * as Color from \"./color.js\";\n;\n;\n;\n;\n/**\n * Lighting computation, based on a traditional FOV for multiple light sources and multiple passes.\n */\nexport default class Lighting {\n    constructor(reflectivityCallback, options = {}) {\n        this._reflectivityCallback = reflectivityCallback;\n        this._options = {};\n        options = Object.assign({\n            passes: 1,\n            emissionThreshold: 100,\n            range: 10\n        }, options);\n        this._lights = {};\n        this._reflectivityCache = {};\n        this._fovCache = {};\n        this.setOptions(options);\n    }\n    /**\n     * Adjust options at runtime\n     */\n    setOptions(options) {\n        Object.assign(this._options, options);\n        if (options && options.range) {\n            this.reset();\n        }\n        return this;\n    }\n    /**\n     * Set the used Field-Of-View algo\n     */\n    setFOV(fov) {\n        this._fov = fov;\n        this._fovCache = {};\n        return this;\n    }\n    /**\n     * Set (or remove) a light source\n     */\n    setLight(x, y, color) {\n        let key = x + \",\" + y;\n        if (color) {\n            this._lights[key] = (typeof (color) == \"string\" ? Color.fromString(color) : color);\n        }\n        else {\n            delete this._lights[key];\n        }\n        return this;\n    }\n    /**\n     * Remove all light sources\n     */\n    clearLights() { this._lights = {}; }\n    /**\n     * Reset the pre-computed topology values. Call whenever the underlying map changes its light-passability.\n     */\n    reset() {\n        this._reflectivityCache = {};\n        this._fovCache = {};\n        return this;\n    }\n    /**\n     * Compute the lighting\n     */\n    compute(lightingCallback) {\n        let doneCells = {};\n        let emittingCells = {};\n        let litCells = {};\n        for (let key in this._lights) { /* prepare emitters for first pass */\n            let light = this._lights[key];\n            emittingCells[key] = [0, 0, 0];\n            Color.add_(emittingCells[key], light);\n        }\n        for (let i = 0; i < this._options.passes; i++) { /* main loop */\n            this._emitLight(emittingCells, litCells, doneCells);\n            if (i + 1 == this._options.passes) {\n                continue;\n            } /* not for the last pass */\n            emittingCells = this._computeEmitters(litCells, doneCells);\n        }\n        for (let litKey in litCells) { /* let the user know what and how is lit */\n            let parts = litKey.split(\",\");\n            let x = parseInt(parts[0]);\n            let y = parseInt(parts[1]);\n            lightingCallback(x, y, litCells[litKey]);\n        }\n        return this;\n    }\n    /**\n     * Compute one iteration from all emitting cells\n     * @param emittingCells These emit light\n     * @param litCells Add projected light to these\n     * @param doneCells These already emitted, forbid them from further calculations\n     */\n    _emitLight(emittingCells, litCells, doneCells) {\n        for (let key in emittingCells) {\n            let parts = key.split(\",\");\n            let x = parseInt(parts[0]);\n            let y = parseInt(parts[1]);\n            this._emitLightFromCell(x, y, emittingCells[key], litCells);\n            doneCells[key] = 1;\n        }\n        return this;\n    }\n    /**\n     * Prepare a list of emitters for next pass\n     */\n    _computeEmitters(litCells, doneCells) {\n        let result = {};\n        for (let key in litCells) {\n            if (key in doneCells) {\n                continue;\n            } /* already emitted */\n            let color = litCells[key];\n            let reflectivity;\n            if (key in this._reflectivityCache) {\n                reflectivity = this._reflectivityCache[key];\n            }\n            else {\n                let parts = key.split(\",\");\n                let x = parseInt(parts[0]);\n                let y = parseInt(parts[1]);\n                reflectivity = this._reflectivityCallback(x, y);\n                this._reflectivityCache[key] = reflectivity;\n            }\n            if (reflectivity == 0) {\n                continue;\n            } /* will not reflect at all */\n            /* compute emission color */\n            let emission = [0, 0, 0];\n            let intensity = 0;\n            for (let i = 0; i < 3; i++) {\n                let part = Math.round(color[i] * reflectivity);\n                emission[i] = part;\n                intensity += part;\n            }\n            if (intensity > this._options.emissionThreshold) {\n                result[key] = emission;\n            }\n        }\n        return result;\n    }\n    /**\n     * Compute one iteration from one cell\n     */\n    _emitLightFromCell(x, y, color, litCells) {\n        let key = x + \",\" + y;\n        let fov;\n        if (key in this._fovCache) {\n            fov = this._fovCache[key];\n        }\n        else {\n            fov = this._updateFOV(x, y);\n        }\n        for (let fovKey in fov) {\n            let formFactor = fov[fovKey];\n            let result;\n            if (fovKey in litCells) { /* already lit */\n                result = litCells[fovKey];\n            }\n            else { /* newly lit */\n                result = [0, 0, 0];\n                litCells[fovKey] = result;\n            }\n            for (let i = 0; i < 3; i++) {\n                result[i] += Math.round(color[i] * formFactor);\n            } /* add light color */\n        }\n        return this;\n    }\n    /**\n     * Compute FOV (\"form factor\") for a potential light source at [x,y]\n     */\n    _updateFOV(x, y) {\n        let key1 = x + \",\" + y;\n        let cache = {};\n        this._fovCache[key1] = cache;\n        let range = this._options.range;\n        function cb(x, y, r, vis) {\n            let key2 = x + \",\" + y;\n            let formFactor = vis * (1 - r / range);\n            if (formFactor == 0) {\n                return;\n            }\n            cache[key2] = formFactor;\n        }\n        ;\n        this._fov.compute(x, y, range, cb.bind(this));\n        return cache;\n    }\n}\n","export { default as RNG } from \"./rng.js\";\nexport { default as Display } from \"./display/display.js\";\nexport { default as StringGenerator } from \"./stringgenerator.js\";\nexport { default as EventQueue } from \"./eventqueue.js\";\nexport { default as Scheduler } from \"./scheduler/index.js\";\nexport { default as FOV } from \"./fov/index.js\";\nexport { default as Map } from \"./map/index.js\";\nexport { default as Noise } from \"./noise/index.js\";\nexport { default as Path } from \"./path/index.js\";\nexport { default as Engine } from \"./engine.js\";\nexport { default as Lighting } from \"./lighting.js\";\nexport { DEFAULT_WIDTH, DEFAULT_HEIGHT, DIRS, KEYS } from \"./constants.js\";\nimport * as util from \"./util.js\";\nexport const Util = util;\nimport * as color from \"./color.js\";\nexport const Color = color;\nimport * as text from \"./text.js\";\nexport const Text = text;\n","/**\n * @class Abstract display backend module\n * @private\n */\nexport default class Backend {\n    getContainer() { return null; }\n    setOptions(options) { this._options = options; }\n}\n","const ROT = require('rot-js');\r\n// See: http://ondras.github.io/rot.js/manual/#rng\r\n\r\nfunction setSeed(seed) {\r\n\tif (typeof seed !== 'number') {\r\n\t\tseed = makeSeed();\r\n\t}\r\n\tROT.RNG.setSeed(seed);\r\n}\r\n\r\nfunction makeSeed() {\r\n\treturn Math.round(Math.random() * 10000);\r\n}\r\n\r\nfunction roll(n, seed) {\r\n\tif (typeof seed === 'number') {\r\n\t\tsetSeed(seed);\r\n\t}\r\n\tif (typeof n === 'number') {\r\n\t\treturn Math.floor(ROT.RNG.getUniform() * n);\r\n\t}\r\n\tif (typeof n === 'string') {\r\n\t\treturn rollDice(n, seed);\r\n\t}\r\n}\r\n\r\nfunction rollDice(str, seed) { // str like \"1d4\", \"2d6\", \"3d8\", \"1d100\"\r\n\t// TODO: handle \"+\", \"-\", other?\r\n\tconst d = str.split('d');\r\n\tif (d.length !== 2) {\r\n\t\tconst n = Number(str);\r\n\t\t// console.warn('Unexpected value:', str, '. Not valid dice notation. Using:', n);\r\n\t\treturn roll(n, seed);\r\n\t}\r\n\tconst numberOfDice = d[0];\r\n\tconst numberOfSides = d[1];\r\n\tlet sum = 0;\r\n\tfor(let i = 0; i < numberOfDice; i++) {\r\n\t\tsum += roll(numberOfSides, seed);\r\n\t}\r\n\treturn sum;\r\n}\r\n\r\nfunction getWeightedValue(obj, seed) {\r\n\tif (typeof seed === 'number') {\r\n\t\tsetSeed(seed);\r\n\t}\r\n\treturn ROT.RNG.getWeightedValue(obj);\r\n}\r\n\r\nfunction shuffle(arr) {\r\n\treturn ROT.RNG.shuffle(arr);\r\n}\r\n\r\nfunction pickOne(arr) {\r\n\treturn ROT.RNG.getItem(arr);\r\n}\r\n\r\nmodule.exports = {\r\n\tsetSeed,\r\n\tmakeSeed,\r\n\troll,\r\n\tgetWeightedValue,\r\n\tshuffle,\r\n\tpickOne\r\n};\r\n","const Inventory = require('./Inventory');\r\nconst { DIRS_8 } = require('./constants');\r\n\r\nclass Item {\r\n\tconstructor(options = {}) {\r\n\t\tthis.type = options.type || null;\r\n\t\tthis.name = options.name || 'nothing';\r\n\t\tthis.x = options.x || 0;\r\n\t\tthis.y = options.y || 0;\r\n\t\tthis.character = options.character || '^';\r\n\t\tthis.surrounding = options.surrounding || [];\r\n\t\tthis.color = options.color || '#05f';\r\n\t\tthis.background = options.background || null;\r\n\t\tthis.inventory = new Inventory({\r\n\t\t\tsize: options.inventorySize || 0\r\n\t\t});\r\n\t\tthis.isWeapon = Boolean(options.weapon);\r\n\t\tthis.damage = parseInt(options.weapon, 10) || 0;\r\n\t\tthis.isDefense = Boolean(options.defense);\r\n\t\tthis.defense = parseInt(options.defense, 10) || 0;\r\n\t\tthis.illumination = options.illumination || 0;\r\n\t\tthis.portable = (typeof options.portable === 'boolean') ? options.portable : true;\r\n\t\tthis.containedIn = null;\r\n\t\tthis.actions = { ...options.on, ...options.actions }; // TODO: do we need the \"on\" alias?\r\n\t\tif (options.use) {\r\n\t\t\tthis.actions.use = options.use;\r\n\t\t}\r\n\t\tthis.states = options.states || {};\r\n\t\tthis.teleport = null; // can this item move the character to another level, cell\r\n\t}\r\n\r\n\thasAction(verb) {\r\n\t\treturn Boolean(this.actions[verb]);\r\n\t}\r\n\r\n\taction(actionName, who) {\r\n\t\tconst action = this.actions[actionName];\r\n\t\tlet actionOutcome = {};\r\n\t\tif (typeof action === 'function') {\r\n\t\t\tactionOutcome = action(this, who);\r\n\t\t} else if (typeof action === 'object' && action !== null) {\r\n\t\t\tactionOutcome = this.runAction(action, who);\r\n\t\t} else {\r\n\t\t\tconsole.warn('No action', actionName, 'for item', this);\r\n\t\t}\r\n\t\treturn actionOutcome;\r\n\t}\r\n\r\n\trunAction(action = {}, who) { // TODO: move to game/level?\r\n\t\tlet message = '';\r\n\t\tif (!this.requirementMet(action, who)) {\r\n\t\t\tmessage = (action.missingMessage) ? action.missingMessage : `Some requirement is not met to use the ${this.name}`;\r\n\t\t\treturn { message };\r\n\t\t}\r\n\t\tthis.removeRequirements(action);\r\n\t\tmessage = message + ((action.message) ? action.message : '');\r\n\t\tconst effects = action.effects;\r\n\t\treturn { message, effects };\r\n\t}\r\n\r\n\tremoveRequirements(action = {}) {\r\n\t\tif (!action.requires) { return; }\r\n\t\taction.requires.forEach((requirement) => {\r\n\t\t\tconst typeKey = requirement.item;\r\n\t\t\tif (typeKey) {\r\n\t\t\t\tthis.inventory.removeType(typeKey);\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n\r\n\trequirementMet(action = {}, who) {\r\n\t\tif (!action.requires) {\r\n\t\t\treturn true;\r\n\t\t}\r\n\t\tlet met = 0;\r\n\t\taction.requires.forEach((requirement) => {\r\n\t\t\tif (requirement.item && this.inventory.containsType(requirement.item)) {\r\n\t\t\t\tmet += 1;\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn met === action.requires.length;\r\n\t}\r\n\r\n\tdraw(display, lighting = {}, inView = false) {\r\n\t\tif (this.containedIn || !inView) { // Not visible if in a container\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tdisplay.draw(this.x, this.y, this.character, this.color, this.background);\r\n\t\tif (this.surrounding.length) {\r\n\t\t\tthis.surrounding.forEach((char, i) => {\r\n\t\t\t\tlet { x, y } = DIRS_8[i];\r\n\t\t\t\tdisplay.draw(this.x + x, this.y + y, char, this.color, this.background);\r\n\t\t\t});\r\n\t\t}\r\n\t\treturn true;\r\n\t}\r\n\r\n\taddItem(item) { // mutates the item if successful\r\n\t\tif (this.inventory.isFull()) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tconst isAdded = this.inventory.add(item);\r\n\t\tif (!isAdded) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\titem.containedIn = this;\r\n\t\treturn true;\r\n\t}\r\n\r\n\t//---- Inventory\r\n\r\n\tgetContents(n) {\r\n\t\treturn this.inventory.get(n);\r\n\t}\r\n\r\n\thasContents() {\r\n\t\treturn this.inventory.hasContents();\r\n\t}\r\n\r\n\tcontains(itemName) {\r\n\t\treturn this.inventory.contains(itemName);\r\n\t}\r\n\r\n\thasSpace() {\r\n\t\treturn this.inventory.hasSpace();\r\n\t}\r\n\r\n\taddToInventory(item) {\r\n\t\treturn this.inventory.add(item);\r\n\t}\r\n\r\n\t//---- Sets\r\n\r\n\tsetTeleport(options = {}) {\r\n\t\tconst { levelIndex, x, y, verb } = options;\r\n\t\tthis.teleport = { levelIndex, x, y };\r\n\t\tthis.actions[verb] = 'teleport';\r\n\t}\r\n}\r\n\r\nmodule.exports = Item;\r\n","function getDistance(x1, y1, x2, y2) {\r\n\treturn Math.sqrt(\r\n\t\tMath.pow((x1 - x2), 2)\r\n\t\t+ Math.pow((y1 - y2), 2)\r\n\t);\r\n}\r\n\r\nmodule.exports = {\r\n\tgetDistance\r\n};\r\n","const ROT = require('rot-js');\r\nconst Inventory = require('./Inventory');\r\nconst geometer = require('./geometer');\r\nconst random = require('./random');\r\n\r\nclass ActionVerbType {\r\n\tstatic get MOVE () {\r\n\t\treturn 'move';\r\n\t}\r\n\r\n\tstatic get WAIT () {\r\n\t\treturn 'wait';\r\n\t}\r\n}\r\n\r\nclass FactionType {\r\n\r\n\tstatic get HUMAN () {\r\n\t\treturn 'human';\r\n\t}\r\n\r\n\tstatic get MONSTER () {\r\n\t\treturn 'monster';\r\n\t}\r\n\r\n\tstatic get NEUTRAL () {\r\n\t\treturn 'neutral';\r\n\t}\r\n}\r\n\r\n\r\nclass Actor {\r\n\tconstructor(options = {}) {\r\n\t\tthis.type = options.type;\r\n\t\tthis.name = options.name || null;\r\n\t\tthis.faction = options.faction || FactionType.MONSTER;\r\n\t\tthis.isHero = Boolean(options.isHero);\r\n\t\tthis.x = options.x || 0;\r\n\t\tthis.y = options.y || 0;\r\n\t\tthis.character = options.character || 'M';\r\n\t\tthis.originalCharacter = this.character;\r\n\t\tthis.color = options.color || '#df2';\r\n\t\tthis.originalColor = this.color;\r\n\t\tthis.bloodColor = '#611';\r\n\t\t// this.game = options.game || console.error('must tie actor to game');\r\n\t\tthis.inventory = new Inventory({\r\n\t\t\tsize: options.inventorySize || 10\r\n\t\t});\r\n\t\tthis.passable = false;\r\n\t\tthis.actionQueue = [];\r\n\t\tthis.maxMovement = this.isHero ? 1.42 : 1;\r\n\t\tthis.sightRange = (typeof options.sightRange === 'number') ? options.sightRange : 6;\r\n\t\tthis.target = null;\r\n\t\tthis.aggro = options.aggro || 0; // Level will set this to 100 for monsters\r\n\t\t// stats\r\n\t\tthis.hp = (options.hp || typeof options.hp === 'number') ? parseInt(options.hp, 10) : 2;\r\n\t\tthis.hpMax = this.hp;\r\n\t\tthis.ap = options.ap || 0;\t// Attack/Arms\r\n\t\tthis.apMax = this.ap;\r\n\t\tthis.bp = options.bp || 0;\t// Balance\r\n\t\tthis.bpMax = this.bp;\r\n\t\tthis.ep = options.ep || 0;\t// Endurance\r\n\t\tthis.epMax = this.ep;\r\n\t\tthis.fp = options.fp || 0;\t\t// Focus\r\n\t\tthis.fpMax = this.fp;\r\n\t\tthis.wp = options.wp || 0;\t\t// Will(power)\r\n\t\tthis.wpMax = this.wp;\r\n\t\tthis.mp = options.mp || 0;\t\t// Mana\r\n\t\tthis.mpMax = this.mp;\r\n\t\t// advancement\r\n\t\tthis.xp = options.xp || 0;\r\n\t\tthis.score = options.score || 0;\r\n\t\t// abilities\r\n\t\tthis.maxAbilities = 9;\r\n\t\tthis.abilities = {};\r\n\t\tthis.abilityList = [];\r\n\t\t// temporary\r\n\t\tthis.initiativeBoost = 0;\r\n\r\n\t\t// all actors can carry currency\r\n\t\t// everything is boiled down to copper pieces\r\n\t\t// 100 copper = 1 silver\r\n\t\t// 100 silver = 1 gold\r\n\t\tthis.currency = options.currency || 0;\r\n\t}\r\n\r\n\tdraw(display, lighting = {}, inView = false) {\r\n\t\tif (!inView) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\t// TODO: adjust colors based on lighting and inView\r\n\t\tdisplay.draw(this.x, this.y, this.character, this.color);\r\n\t\treturn true;\r\n\t}\r\n\r\n\tqueueAction(verb, params = {}) {\r\n\t\tconst actionParams = { ...params, verb };\r\n\t\tthis.actionQueue.push(actionParams);\r\n\t}\r\n\r\n\tclearQueue() {\r\n\t\tthis.actionQueue.length = 0;\r\n\t}\r\n\r\n\tplanAction(level, hero) {\r\n\t\tif (this.isHero) { return; }\r\n\t\tif (this.dead()) {\r\n\t\t\tthis.clearQueue();\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tconst distanceToHero = geometer.getDistance(this.x, this.y, hero.x, hero.y);\r\n\t\tconst dangerouslyHurt = (this.hp <= 1);\r\n\t\tthis.act = function () {\r\n\t\t\tconsole.log(`${this.name} acts`);\r\n\t\t\t// if (g.getActiveLevel() !== level) { return; }\r\n\t\t};\r\n\t\tif (this.aggro && distanceToHero <= this.getMaxSenseRange() && !hero.dead() && !dangerouslyHurt) {\r\n\t\t\tconst map = level.getMap();\r\n\t\t\tthis.clearQueue();\r\n\t\t\tthis.setTarget(hero);\r\n\t\t\tthis.setPathToTarget(map);\r\n\t\t\tif (this.atEndOfPath()) {\r\n\t\t\t\tthis.queueAction('attack', { target: hero });\r\n\t\t\t} else if (this.actionQueue.length === 1) {\r\n\t\t\t\tthis.clearQueue();\r\n\t\t\t\tthis.queueAction('attack', { target: hero });\r\n\t\t\t}\r\n\t\t} else {\r\n\t\t\tif (this.atEndOfPath()) {\r\n\t\t\t\tthis.setWanderPath(level);\r\n\t\t\t}\r\n\t\t}\r\n\t\t// console.log(`${this.name} plans`, this.actionQueue);\r\n\t}\r\n\r\n\tdoAction() {\r\n\t\tif (this.dead()) { return { verb: 'rot' }; }\r\n\t\tconst waitAction = { verb: ActionVerbType.WAIT };\r\n\t\tif (this.actionQueue.length === 0) { return waitAction; }\r\n\t\tlet action = this.actionQueue.shift();\r\n\t\tconst moveAlreadyThere = (action.verb === ActionVerbType.MOVE && action.x === this.x && action.y === this.y);\r\n\t\tconst moveTooFar = (action.verb === ActionVerbType.MOVE && this.getDistanceToNextMove(action) > this.maxMovement);\r\n\t\t// console.log(this.name, this.x, this.y, action.verb, action.x, action.y, this.getDistanceToNextMove(), this.maxMovement, moveTooFar, 'q', this.actionQueue.length);\r\n\t\tif (moveAlreadyThere) {\r\n\t\t\treturn this.doAction();\r\n\t\t}\r\n\t\tif (moveTooFar) {\r\n\t\t\taction = this.doAction();\r\n\t\t}\r\n\t\tif (!action) {\r\n\t\t\treturn waitAction;\r\n\t\t}\r\n\t\treturn action;\r\n\t}\r\n\r\n\tattack(who) {\r\n\t\tconsole.log(`${this.name} attacks`, who);\r\n\t\t// TODO\r\n\t}\r\n\r\n\tsetWanderPath(level) {\r\n\t\tconst map = level.getMap();\r\n\t\tconst { x, y } = level.findRandomFreeCell();\r\n\t\tthis.setPathTo(map, x, y);\r\n\t}\r\n\r\n\tatEndOfPath() {\r\n\t\tconst nextAction = this.getNextAction();\r\n\t\tif (!nextAction) { return true; }\r\n\t\treturn (nextAction.verb === ActionVerbType.MOVE) ? false : true;\r\n\t}\r\n\r\n\twait() {\r\n\t\tthis.healPools();\r\n\t}\r\n\r\n\t//---- Movement\r\n\r\n\tmove(x, y) {\r\n\t\tthis.x += parseInt(x, 10);\r\n\t\tthis.y += parseInt(y, 10);\r\n\t}\r\n\r\n\tmoveTo(x, y) {\r\n\t\tthis.setCoordinates(x, y);\r\n\t}\r\n\r\n\t//---- Combat\r\n\r\n\tattackDamage(opponent) {\r\n\t\treturn 1;\r\n\t}\r\n\r\n\twound(n) {\r\n\t\treturn this.heal(n * -1);\r\n\t}\r\n\r\n\theal(n) {\r\n\t\tconst originalHp = this.hp;\r\n\t\tthis.hp += parseInt(n, 10);\r\n\t\tthis.hp = Math.min(this.hp, this.hpMax);\r\n\t\tthis.checkDeath();\r\n\t\treturn this.hp - originalHp;\r\n\t}\r\n\r\n\tdead() {\r\n\t\treturn (this.hp <= 0);\r\n\t}\r\n\r\n\tcheckDeath() {\r\n\t\tif (this.dead()) {\r\n\t\t\tthis.character = 'X';\r\n\t\t\tthis.color = this.bloodColor;\r\n\t\t\tthis.passable = true;\r\n\t\t}\r\n\t}\r\n\r\n\t//---- Healing\r\n\r\n\thealPools() {\r\n\t\tthis.healPool(this.getRandomPoolKey());\r\n\t}\r\n\r\n\thealPool(poolKey, amount = 1) {\r\n\t\tconst a = this.getAbilityReadiedAmounts();\r\n\t\tconst max = this[poolKey + 'Max'];\r\n\t\tif (a[poolKey] + this[poolKey] + amount <= max) {\r\n\t\t\tthis[poolKey] += amount;\r\n\t\t} else {\r\n\t\t\tif (this.isHero) {\r\n\t\t\t\tconsole.log('No space to heal', poolKey, this);\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\tdamagePool(poolKey, amount = 1) {\r\n\t\tthis[poolKey] -= amount;\r\n\t\tthis[poolKey] = Math.max(0, this[poolKey]);\r\n\t}\r\n\r\n\t//---- Abilities\r\n\r\n\thasAbility(abilityKey) {\r\n\t\treturn Boolean(this.abilities[abilityKey]);\r\n\t}\r\n\r\n\taddAbility(abilityKey, abilityData) {\r\n\t\tif (this.abilityList.length >= this.maxAbilities) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tif (this.hasAbility(abilityKey)) {\r\n\t\t\tconsole.warn('Cannot add ability twice - would override');\r\n\t\t\treturn;\r\n\t\t}\r\n\t\t// TODO: move to Activity class?\r\n\t\tconst ability = JSON.parse(JSON.stringify(abilityData));\r\n\t\tthis.abilities[abilityKey] = ability;\r\n\t\tability.isReadied = false;\r\n\t\tability.key = abilityKey;\r\n\t\tthis.abilityList.push(abilityKey);\r\n\t\treturn ability;\r\n\t}\r\n\r\n\tgetAbilityByIndex(i) {\r\n\t\tconst key = this.abilityList[i];\r\n\t\treturn this.abilities[key];\r\n\t}\r\n\r\n\tgetAbilityReadiedAmounts() {\r\n\t\tconst a = { hp: 0, ap: 0, bp: 0, ep: 0, fp: 0, wp: 0 };\r\n\t\tthis.abilityList.forEach((abilityKey) => {\r\n\t\t\tconst ability = this.abilities[abilityKey];\r\n\t\t\tActor.loopOverAbilityCosts(ability, (costKey, val) => {\r\n\t\t\t\tif (ability.isReadied) {\r\n\t\t\t\t\ta[costKey] += val;\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t});\r\n\t\treturn a;\r\n\t}\r\n\r\n\tcanReadyAbility(ability) {\r\n\t\tif (ability.isReadied) { return false; }\r\n\t\tlet canReady = true;\r\n\t\tActor.loopOverAbilityCosts(ability, (costKey, val) => {\r\n\t\t\tconst poolAmount = this[costKey];\r\n\t\t\tif (val > poolAmount) {\r\n\t\t\t\tcanReady = false;\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn canReady;\r\n\t}\r\n\r\n\treadyAbilityByIndex(i) {\r\n\t\tconst ability = this.getAbilityByIndex(i);\r\n\t\tif (!ability) { return false; }\r\n\t\tif (!this.canReadyAbility(ability)) { return false; }\r\n\t\tActor.loopOverAbilityCosts(ability, (costKey, val) => {\r\n\t\t\tthis[costKey] -= val;\r\n\t\t});\r\n\t\tability.isReadied = true;\r\n\t\treturn ability;\r\n\t}\r\n\r\n\tactivateAbilities(eventName) {\r\n\t\tconst triggeredAbilities = this.getTriggeredAbilities(eventName);\r\n\t\tlet effects = [];\r\n\t\ttriggeredAbilities.forEach((ability) => {\r\n\t\t\tability.isReadied = false;\r\n\t\t\teffects = effects.concat(ability.effects);\r\n\t\t});\r\n\t\treturn effects;\r\n\t}\r\n\r\n\tgetTriggeredAbilities(eventName) {\r\n\t\tconst triggeredAbilities = [];\r\n\t\tthis.abilityList.forEach((abilityKey) => {\r\n\t\t\tconst ability = this.abilities[abilityKey];\r\n\t\t\tif (ability.isReadied && ability.activateOn === eventName) {\r\n\t\t\t\ttriggeredAbilities.push(ability);\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn triggeredAbilities;\r\n\t}\r\n\r\n\tstatic loopOverAbilityCosts(ability, fn) {\r\n\t\tconst costs = Object.keys(ability.readyCost);\r\n\t\tcosts.forEach((key) => {\r\n\t\t\tfn(key, parseInt(ability.readyCost[key], 10));\r\n\t\t});\r\n\t}\r\n\r\n\tstatic getAbilityEffectsString(ability) {\r\n\t\tlet arr = [];\r\n\t\tability.effects.forEach((effect) => {\r\n\t\t\tconst words = (typeof effect === 'string') ? [effect] : Object.keys(effect);\r\n\t\t\tarr = arr.concat(words);\r\n\t\t});\r\n\t\treturn arr.join(', ');\r\n\t}\r\n\r\n\tstatic getAbilityDescriptionHtml(ability) {\r\n\t\tlet ready = 'Ready with';\r\n\t\tActor.loopOverAbilityCosts(ability, (costKey, val) => {\r\n\t\t\tready += ' ' + val + ' ' + costKey.toUpperCase();\r\n\t\t});\r\n\t\tconst effects = Actor.getAbilityEffectsString(ability);\r\n\t\treturn `<div class=\"ability-description\">${ability.description}</div>\r\n\t\t<div class=\"ability-ready-with\">${ready}</div>\r\n\t\t<div class=\"ability-activates-on\">Activates on: ${ability.activateOn}</div>\r\n\t\t<div class=\"ability-effects\">Causes: ${effects}</div>`;\r\n\t}\r\n\r\n\t//---- Experience\r\n\r\n\tgainRandomPoolMax() {\r\n\t\tconst key = this.getRandomPoolKey() + 'Max';\r\n\t\tthis[key] += 1;\r\n\t}\r\n\r\n\tgainRandomAbility(abilitiesData) {\r\n\t\tconst abilityKeys = Object.keys(abilitiesData);\r\n\t\tlet abilityKey = random.pickOne(abilityKeys);\r\n\t\tlet attempts = 100;\r\n\t\twhile (this.hasAbility(abilityKey) && attempts--) {\r\n\t\t\tabilityKey = random.pickOne(abilityKeys);\r\n\t\t}\r\n\t\tthis.addAbility(abilityKey, abilitiesData[abilityKey]);\r\n\t}\r\n\r\n\t//---- Gets\r\n\r\n\tgetRandomPoolKey() {\r\n\t\treturn random.pickOne(['ap', 'bp', 'ep', 'wp']);\r\n\t}\r\n\r\n\tgetMaxSenseRange() {\r\n\t\treturn this.sightRange;\r\n\t}\r\n\r\n\tgetNextAction() {\r\n\t\treturn this.actionQueue[0];\r\n\t}\r\n\r\n\tgetDistanceToNextMove(nextAction) {\r\n\t\tif (!nextAction) { nextAction = this.getNextAction(); }\r\n\t\tif (!nextAction) { return 0; }\r\n\t\tconst { x, y } = nextAction;\r\n\t\tif (x !== undefined && y !== undefined) {\r\n\t\t\treturn geometer.getDistance(x, y, this.x, this.y);\r\n\t\t}\r\n\t\treturn null; // ?\r\n\t}\r\n\r\n\tgetArmorDefense() {\r\n\t\tif (!this.isHero) {\r\n\t\t\treturn 1; // TODO: change this so there is some kind of natural defense for monsters\r\n\t\t}\r\n\t\tlet highestDefense = 0;\r\n\t\tthis.inventory.loopOverContents((item) => {\r\n\t\t\tif (item.defense > highestDefense) {\r\n\t\t\t\thighestDefense = item.defense;\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn highestDefense;\r\n\t}\r\n\r\n\tgetWeaponDamage() {\r\n\t\tif (!this.isHero) {\r\n\t\t\treturn 1; // TODO: change this so there is some kind of natural damage for monsters\r\n\t\t}\r\n\t\tlet highestDamage = 0;\r\n\t\tthis.inventory.loopOverContents((item) => {\r\n\t\t\tif (item.damage > highestDamage) {\r\n\t\t\t\thighestDamage = item.damage;\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn highestDamage;\r\n\t}\r\n\r\n\t//---- Sets\r\n\r\n\tsetCoordinates(x, y) {\r\n\t\tthis.x = parseInt(x, 10);\r\n\t\tthis.y = parseInt(y, 10);\r\n\t}\r\n\r\n\tsetPathTo(map, x = 0, y = 0) {\r\n\t\tconst passableCallback = function(x, y) {\r\n\t\t\treturn map.getCellPassability(x, y);\r\n\t\t};\r\n\t\tconst astar = new ROT.Path.AStar(x, y, passableCallback, { topology: 4 });\r\n\t\tconst path = this.actionQueue;\r\n\t\tconst pathCallback = function(x, y) {\r\n\t\t\tpath.push({ x, y, verb: ActionVerbType.MOVE });\r\n\t\t};\r\n\t\tif (path[0] && path[0].x === this.x && path[0].y === this.y) {\r\n\t\t\tconsole.alert('removing first');\r\n\t\t\tpath.shift();\r\n\t\t}\r\n\t\tastar.compute(this.x, this.y, pathCallback);\r\n\t\treturn true;\r\n\t}\r\n\r\n\tsetPathToTarget(map) {\r\n\t\treturn this.setPathTo(map, this.target.x, this.target.y);\r\n\t}\r\n\r\n\tsetTarget(target) {\r\n\t\tif (typeof target.x !== 'number' || typeof target.y !== 'number') {\r\n\t\t\tconsole.warn('Cannot set target to something without x,y');\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tthis.target = target;\r\n\t}\r\n}\r\n\r\nmodule.exports = Actor;\r\n","import Backend from \"./backend.js\";\nimport * as Color from \"../color.js\";\nfunction clearToAnsi(bg) {\n    return `\\x1b[0;48;5;${termcolor(bg)}m\\x1b[2J`;\n}\nfunction colorToAnsi(fg, bg) {\n    return `\\x1b[0;38;5;${termcolor(fg)};48;5;${termcolor(bg)}m`;\n}\nfunction positionToAnsi(x, y) {\n    return `\\x1b[${y + 1};${x + 1}H`;\n}\nfunction termcolor(color) {\n    const SRC_COLORS = 256.0;\n    const DST_COLORS = 6.0;\n    const COLOR_RATIO = DST_COLORS / SRC_COLORS;\n    let rgb = Color.fromString(color);\n    let r = Math.floor(rgb[0] * COLOR_RATIO);\n    let g = Math.floor(rgb[1] * COLOR_RATIO);\n    let b = Math.floor(rgb[2] * COLOR_RATIO);\n    return r * 36 + g * 6 + b * 1 + 16;\n}\nexport default class Term extends Backend {\n    constructor() {\n        super();\n        this._offset = [0, 0];\n        this._cursor = [-1, -1];\n        this._lastColor = \"\";\n    }\n    schedule(cb) { setTimeout(cb, 1000 / 60); }\n    setOptions(options) {\n        super.setOptions(options);\n        let size = [options.width, options.height];\n        let avail = this.computeSize();\n        this._offset = avail.map((val, index) => Math.floor((val - size[index]) / 2));\n    }\n    clear() {\n        process.stdout.write(clearToAnsi(this._options.bg));\n    }\n    draw(data, clearBefore) {\n        // determine where to draw what with what colors\n        let [x, y, ch, fg, bg] = data;\n        // determine if we need to move the terminal cursor\n        let dx = this._offset[0] + x;\n        let dy = this._offset[1] + y;\n        let size = this.computeSize();\n        if (dx < 0 || dx >= size[0]) {\n            return;\n        }\n        if (dy < 0 || dy >= size[1]) {\n            return;\n        }\n        if (dx !== this._cursor[0] || dy !== this._cursor[1]) {\n            process.stdout.write(positionToAnsi(dx, dy));\n            this._cursor[0] = dx;\n            this._cursor[1] = dy;\n        }\n        // terminals automatically clear, but if we're clearing when we're\n        // not otherwise provided with a character, just use a space instead\n        if (clearBefore) {\n            if (!ch) {\n                ch = \" \";\n            }\n        }\n        // if we're not clearing and not provided with a character, do nothing\n        if (!ch) {\n            return;\n        }\n        // determine if we need to change colors\n        let newColor = colorToAnsi(fg, bg);\n        if (newColor !== this._lastColor) {\n            process.stdout.write(newColor);\n            this._lastColor = newColor;\n        }\n        // write the provided symbol to the display\n        let chars = [].concat(ch);\n        process.stdout.write(chars[0]);\n        // update our position, given that we wrote a character\n        this._cursor[0]++;\n        if (this._cursor[0] >= size[0]) {\n            this._cursor[0] = 0;\n            this._cursor[1]++;\n        }\n    }\n    computeFontSize() { throw new Error(\"Terminal backend has no notion of font size\"); }\n    eventToPosition(x, y) { return [x, y]; }\n    computeSize() { return [process.stdout.columns, process.stdout.rows]; }\n}\n","const FontFaceObserver = require('fontfaceobserver');\r\n\r\nfunction domReady() {\r\n\treturn new Promise((resolve, reject) => {\r\n\t\tif (document.readyState === \"complete\" || document.readyState === \"loaded\") {\r\n\t\t\tresolve();\r\n\t\t} else {\r\n\t\t\tdocument.addEventListener(\"DOMContentLoaded\", () => {\r\n\t\t\t\tresolve();\r\n\t\t\t});\r\n\t\t}\r\n\t});\r\n}\r\n\r\nfunction ready(fn, fonts = []) {\r\n\tif (fonts.length > 0) {\r\n\t\t// TODO: allow multiple fonts ~ https://github.com/bramstein/fontfaceobserver\r\n\t\tconst font = new FontFaceObserver(fonts[0]);\r\n\t\tfont.load()\r\n\t\t\t.then(() => { domReady().then(fn); })\r\n\t\t\t.catch(() => { console.warn('error loading font'); domReady().then(fn); });\r\n\t\treturn;\r\n\t}\r\n\tdomReady().then(fn);\r\n}\r\n\r\nmodule.exports = ready;\r\n","const ROT = require('rot-js');\r\n\r\nclass Display {\r\n\tconstructor(options = {}) {\r\n\t\toptions = { width: 60, height: 30, ...options };\r\n\t\tthis.width = null;\r\n\t\tthis.height = null;\r\n\t\tthis.center = {};\r\n\t\tthis.setDimensions(options.width, options.height);\r\n\r\n\t\tthis.id = options.id || 'display';\r\n\t\tthis.rotDisplay = new ROT.Display(options); // , layout:\"term\"});\r\n\t\tthis.displayContainer = null;\r\n\t\tthis.elt = null\r\n\t\tthis.cameraTarget = null;\r\n\t\tthis.setupElements();\r\n\t}\r\n\r\n\tsetDimensions(x, y) {\r\n\t\tthis.width = x;\r\n\t\tthis.height = y;\r\n\t\tthis.center.x = Math.round(x/2);\r\n\t\tthis.center.y = Math.round(y/2);\r\n\t}\r\n\r\n\tsetupElements() {\r\n\t\tthis.displayContainer = document.getElementById(this.id);\r\n\t\tthis.elt = this.rotDisplay.getContainer(); // canvas\r\n\t\tthis.appendToElement(this.elt);\r\n\t}\r\n\r\n\tappendToElement(elt) {\r\n\t\tthis.displayContainer.appendChild(elt);\r\n\t}\r\n\r\n\tsetCameraTarget(cameraTarget) {\r\n\t\tif (!cameraTarget) {\r\n\t\t\tconsole.warn(\"No target\", cameraTarget);\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tif (typeof cameraTarget.x !== 'number' || typeof cameraTarget.y !== 'number') {\r\n\t\t\tconsole.warn(\"Couldn't target\", cameraTarget);\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tthis.cameraTarget = cameraTarget;\r\n\t\treturn true;\r\n\t}\r\n\r\n\tclear() {\r\n\t\tthis.rotDisplay.clear();\r\n\t}\r\n\r\n\tdraw(x, y, character, fgColor, bgColor) {\r\n\t\tif (this.cameraTarget) {\r\n\t\t\tx += (this.center.x - this.cameraTarget.x);\r\n\t\t\ty += (this.center.y - this.cameraTarget.y);\r\n\t\t}\r\n\t\treturn this.rotDisplay.draw(x, y, character, fgColor, bgColor);\r\n\t}\r\n\r\n\tdrawLevel(game, level, hero) {\r\n\t\tlevel.draw(this);\r\n\t\tif (!hero) { return; }\r\n\t\thero.draw(this);\r\n\t\tthis.drawInterface(game, hero);\r\n\t}\r\n\r\n\tdrawHero(hero) {\r\n\t\tif (!hero) { return; }\r\n\t\thero.draw(this);\r\n\t}\r\n\r\n\tdrawDamage(isDamaged = false, options = {}) {\r\n\t\t// Override this\r\n\t}\r\n\r\n\tdrawInterface(game = {}, hero = {}, options = {}) {\r\n\t\t// Override this\r\n\t}\r\n\r\n\tstatic getPoolSquares(value, max, used) {\r\n\t\tconst maxLeft = max - value - used;\r\n\t\tlet str = '';\r\n\t\tlet i;\r\n\t\tfor(i = 0; i < value; i++) { str += '■'; }\r\n\t\tfor(i = 0; i < used; i++) { str += '▣'; }\r\n\t\tfor(i = 0; i < maxLeft; i++) { str += '▢'; }\r\n\t\treturn str;\r\n\t}\r\n\r\n}\r\n\r\nmodule.exports = Display;","const Map = require('./Map');\r\nconst Actor = require('./Actor');\r\nconst Item = require('./Item');\r\nconst Prop = require('./Prop');\r\nconst geometer = require('./geometer');\r\nconst random = require('./random');\r\nconst { DIRS_4, DIRS_8_DIAGNOLS } = require('./constants');\r\n\r\nclass Level {\r\n\tconstructor(options = {}, refData = {}) {\r\n\t\tthis.seed = options.seed || 1;\r\n\t\tthis.name = options.name || 'Unknown level';\r\n\t\tthis.description = options.description || null;\r\n\t\tthis.levelIndex = options.levelIndex || 0;\r\n\t\tthis.color = options.color || '#777';\r\n\t\tthis.background = options.background || '#222';\r\n\t\tconst mapOptions = {\r\n\t\t\tcolor: this.color,\r\n\t\t\tbackground: this.background,\r\n\t\t\t...options.map,\r\n\t\t\tseed: this.seed,\r\n\t\t\tgenerators: options.generators || {}\r\n\t\t};\r\n\t\tthis.map = new Map(mapOptions);\r\n\t\tthis.actors = [];\r\n\t\tthis.items = [];\r\n\t\tthis.props = [];\r\n\t\tthis.actors = this.generateActors(options, refData);\r\n\t\tthis.items = this.generateItems(options, refData.items);\r\n\t\tthis.props = this.generateProps(options, refData.props);\r\n\t\tthis.eye = { x: 0, y: 0, sightRange: 7 };\r\n\t\tthis.customEffects = { ...options.customEffects };\r\n\t}\r\n\r\n\tdraw(display) {\r\n\t\tdisplay.clear();\r\n\t\tthis.drawMap(display);\r\n\t\tthis.drawProps(display);\r\n\t\tthis.drawItems(display);\r\n\t\tthis.drawActors(display);\r\n\t}\r\n\r\n\tdrawMap(display) {\r\n\t\tthis.map.forEachCell((cell, x, y) => {\r\n\t\t\tconst inView = this.isInView(x, y);\r\n\t\t\t// TODO: improve this\r\n\t\t\tconst fg = cell.getForegroundColor(inView);\r\n\t\t\tconst bg = cell.getBackgroundColor(inView);\r\n\t\t\tdisplay.draw(x, y, cell.character, fg, bg);\r\n\t\t});\r\n\t}\r\n\r\n\tdrawProps(display) {\r\n\t\tthis.props.forEach((prop) => {\r\n\t\t\tconst lighting = this.map.getLightingAt(this.eye.x, this.eye.y);\r\n\t\t\tconst inView = this.isInView(prop.x, prop.y);\r\n\t\t\tprop.draw(display, lighting, inView);\r\n\t\t});\r\n\t}\r\n\r\n\tdrawItems(display) {\r\n\t\tthis.items.forEach((item) => {\r\n\t\t\tconst lighting = this.map.getLightingAt(this.eye.x, this.eye.y);\r\n\t\t\tconst inView = this.isInView(item.x, item.y);\r\n\t\t\titem.draw(display, lighting, inView);\r\n\t\t});\r\n\t}\r\n\r\n\tdrawActors(display) {\r\n\t\t// Draw dead first, then non-dead\r\n\t\tthis.actors.forEach((actor) => {\r\n\t\t\tif (actor.dead()) {\r\n\t\t\t\tthis.drawActor(display, actor);\r\n\t\t\t}\r\n\t\t});\r\n\t\tthis.actors.forEach((actor) => {\r\n\t\t\tif (!actor.dead()) {\r\n\t\t\t\tthis.drawActor(display, actor);\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n\r\n\tdrawActor(display, actor) {\r\n\t\tconst lighting = this.map.getLightingAt(this.eye.x, this.eye.y);\r\n\t\tconst inView = this.isInView(actor.x, actor.y);\r\n\t\tactor.draw(display, lighting, inView);\r\n\t}\r\n\r\n\tisInView(x, y) { // TODO: optimize\r\n\t\tconst r = geometer.getDistance(this.eye.x, this.eye.y, x, y); // TODO: allow more complicated POV\r\n\t\treturn (r <= this.eye.sightRange);\t\t\r\n\t}\r\n\r\n\taddItem(item) {\r\n\t\tthis.items.push(item);\r\n\t}\r\n\r\n\tremoveItem(item) {\r\n\t\treturn this.removeThing('items', item);\r\n\t}\r\n\r\n\taddActor(actor) {\r\n\t\tthis.actors.push(actor);\r\n\t}\r\n\r\n\tremoveActor(actor) {\r\n\t\treturn this.removeThing('actors', actor);\r\n\t}\r\n\r\n\tremoveThing(property, thing) {\r\n\t\tconst i = this[property].findIndex((a) => { return a === thing; });\r\n\t\tif (i <= -1) {\r\n\t\t\tconsole.warn('nothing found in', property);\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tconst arr = this[property].splice(i, 1);\r\n\t\treturn arr[0];\t\t\r\n\t}\r\n\r\n\tfindItem(x, y) {\r\n\t\tconst foundThings = this.findItems(x, y);\r\n\t\treturn (foundThings.length) ? foundThings[0] : null;\r\n\t}\r\n\tfindItems(x, y) {\r\n\t\treturn this.items.filter((item) => {\r\n\t\t\treturn item.x === x && item.y === y && !item.containedIn;\r\n\t\t});\r\n\t}\r\n\r\n\tfindProp(x, y) {\r\n\t\tconst foundThings = this.findProps(x, y);\r\n\t\treturn (foundThings.length) ? foundThings[0] : null;\r\n\t}\r\n\tfindProps(x, y) {\r\n\t\treturn this.props.filter((prop) => { return prop.x === x && prop.y === y; });\r\n\t}\r\n\tfindPropsByType(type) {\r\n\t\treturn this.props.filter((prop) => { return prop.type === type; });\r\n\t}\r\n\r\n\tfindActor(x, y) {\r\n\t\tconst foundThings = this.findActors(x, y);\r\n\t\treturn (foundThings.length) ? foundThings[0] : null;\r\n\t}\r\n\tfindActors(x, y) {\r\n\t\treturn this.actors.filter((actor) => {\r\n\t\t\treturn actor.x === x && actor.y === y && !actor.dead();\r\n\t\t});\r\n\t}\r\n\r\n\tfindThingInView(what) { // 'actors', 'items', 'props'\r\n\t\treturn this[what].filter((a) => this.isInView(a.x, a.y));\r\n\t}\r\n\r\n\tfindActorsInView(excludeHero) {\r\n\t\treturn this.actors.filter((a) => {\r\n\t\t\tconst inView = this.isInView(a.x, a.y);\r\n\t\t\tif (excludeHero) {\r\n\t\t\t\treturn inView && !a.isHero;\r\n\t\t\t}\r\n\t\t\treturn inView;\r\n\t\t});\r\n\t}\r\n\r\n\tfindEverythingInView(options = {}) {\r\n\t\treturn this.findActorsInView(options.excludeHero)\r\n\t\t\t.concat(this.findThingInView('items'))\r\n\t\t\t.concat(this.findThingInView('props'));\r\n\t}\r\n\r\n\tfindThings(x, y) {\r\n\t\tconst props = this.findProps(x, y);\r\n\t\tconst items = this.findItems(x, y);\r\n\t\tconst allThings = props.concat(items);\r\n\t\treturn allThings;\r\n\t}\r\n\r\n\tfindThingsCardinal(x, y) {\r\n\t\tconst props = this.findThingsByDirections('props', DIRS_4, x, y);\r\n\t\tconst items = this.findThingsByDirections('items', DIRS_4, x, y);\r\n\t\tconst allThings = props.concat(items);\r\n\t\treturn allThings;\r\n\t}\r\n\r\n\tfindThingsDiagnol(x, y) {\r\n\t\tconst props = this.findThingsByDirections('props', DIRS_8_DIAGNOLS, x, y);\r\n\t\tconst items = this.findThingsByDirections('items', DIRS_8_DIAGNOLS, x, y);\r\n\t\tconst allThings = props.concat(items);\r\n\t\treturn allThings;\r\n\t}\r\n\r\n\tfindThingsByDirections(thingName, dirs, x, y) {\r\n\t\tconst coords = [];\r\n\t\tdirs.forEach((dir) => { coords.push({ x: x + dir.x, y: y + dir.y }); });\r\n\t\treturn this[thingName].filter((thing) => {\r\n\t\t\tconst matches = coords.filter((xy) => {\r\n\t\t\t\treturn xy.x === thing.x && xy.y === thing.y && !thing.containedIn\r\n\t\t\t});\r\n\t\t\treturn matches.length > 0;\r\n\t\t});\t\t\r\n\t}\r\n\r\n\tfindThingSmart(x, y, perferredProperty) {\r\n\t\tlet things = this.findThings(x, y);\r\n\t\t// console.log('find smart - on spot', things);\r\n\t\tif (!things.length) {\r\n\t\t\tthings = this.findThingsCardinal(x, y);\r\n\t\t\t// console.log('find smart - cardinal', things);\r\n\t\t\tif (!things.length) {\r\n\t\t\t\tthings = this.findThingsDiagnol(x, y);\r\n\t\t\t\t// console.log('find smart - diagnols', things);\r\n\t\t\t}\r\n\t\t}\r\n\t\tif (perferredProperty) {\r\n\t\t\tthings.sort((a, b) => {\r\n\t\t\t\t// console.log(a, b, a[perferredProperty]);\r\n\t\t\t\treturn b[perferredProperty];\r\n\t\t\t});\r\n\t\t\t// console.log(\"sorted\", things);\r\n\t\t}\r\n\t\treturn things[0];\r\n\t}\r\n\r\n\tfindRandomFreeCell(seed, clearing, retries = 50) {\r\n\t\tlet cell = this.map.getRandomFreeCell();\r\n\t\tif (!retries) {\r\n\t\t\treturn cell;\r\n\t\t}\r\n\t\tconst tryAgain = () => {\r\n\t\t\treturn this.findRandomFreeCell(seed, clearing, (retries - 1));\r\n\t\t};\r\n\t\tif (this.findActors(cell.x, cell.y).length > 0) {\r\n\t\t\treturn tryAgain();\r\n\t\t}\r\n\t\tif (this.findThings(cell.x, cell.y).length > 0) {\r\n\t\t\treturn tryAgain();\r\n\t\t}\r\n\t\tif (this.findMapClearing(cell.x, cell.y) >= clearing) {\r\n\t\t\treturn tryAgain();\r\n\t\t}\r\n\t\treturn cell;\r\n\t}\r\n\r\n\tfindMapClearing(x, y) {\r\n\t\t// TODO: loop\r\n\t\treturn 0;\r\n\t}\r\n\r\n\tdiscoverCircle(x, y, radius) {\r\n\t\treturn this.map.discoverCircle(x, y, radius);\r\n\t}\r\n\r\n\t// Actions\r\n\r\n\tuseThing(actor, actionName, thing) {\r\n\t\tconst outcome = thing.action(actionName, actor);\r\n\t\tif (typeof outcome !== 'object') {\r\n\t\t\tconsole.warn('action returns outcome that is not object', actionName, thing, outcome);\r\n\t\t}\r\n\t\tthis.doEffects(outcome.effects, actor, actor);\r\n\t\treturn outcome;\r\n\t}\r\n\r\n\tthrow(actor, what, x, y) {\r\n\t\tconst item = actor.inventory.remove(what);\r\n\t\tif (!item) { return false; }\r\n\t\titem.x = (typeof x === 'number') ? x : actor.x;\r\n\t\titem.y = (typeof y === 'number') ? y : actor.y;\r\n\t\tconst containers = this.findThings(x, y).filter((thing) => {\r\n\t\t\tconsole.log(thing);\r\n\t\t\treturn thing.hasSpace();\r\n\t\t});\r\n\t\tif (containers.length) {\r\n\t\t\tconst container = containers[0];\r\n\t\t\tcontainer.addToInventory(item);\r\n\t\t\treturn `${actor.name} puts ${what.name} into the ${container.name}.`;\r\n\t\t}\r\n\t\tthis.addItem(item);\r\n\t\treturn `${actor.name} throws down a ${what.name}.`;\r\n\t}\r\n\r\n\tdoInitiative() {\r\n\t\tconst livingActors = this.actors.filter((actor) => !actor.dead());\r\n\t\tlet orderedActors = random.shuffle(livingActors);\r\n\t\t// TODO: Look for initiative boost, put at top of list\r\n\t}\r\n\r\n\tstatic removeEffects(effects, key) {\r\n\t\tlet i = effects.indexOf(key);\r\n\t\twhile (i > -1) {\r\n\t\t\teffects.splice(i, 1);\r\n\t\t\ti = effects.indexOf(key);\r\n\t\t}\r\n\t}\r\n\r\n\tresolveRoundEffects() {\r\n\t\tthis.actors.forEach((actor) => {\r\n\t\t\tconst roundEffects = actor.activateAbilities('round');\r\n\t\t\tthis.doEffects(roundEffects, actor, actor);\r\n\t\t});\r\n\t}\r\n\r\n\tresolveCombatEffects(attacker, defender) {\r\n\t\tlet attackEffects = attacker.activateAbilities('attack');\r\n\t\tlet defendEffects = defender.activateAbilities('attacked');\r\n\t\tlet damageEffects;\r\n\t\tlet damagedEffects;\r\n\r\n\t\tattackEffects.push('attack');\r\n\r\n\t\tconsole.log(attacker.name, JSON.stringify(attackEffects), 'vs', defender.name, JSON.stringify(defendEffects), defender);\r\n\r\n\t\tif (defendEffects.includes('cancelAttack')) {\r\n\t\t\tLevel.removeEffects(attackEffects, 'attack');\r\n\t\t}\r\n\r\n\t\tif (attackEffects.includes('attack')) {\r\n\t\t\tattackEffects.push('weaponDamage');\r\n\t\t}\r\n\t\tif (attackEffects.includes('damage') || attackEffects.includes('weaponDamage')) {\r\n\t\t\tdamageEffects = defender.activateAbilities('damage');\r\n\t\t\tattackEffects = attackEffects.concat(damageEffects);\r\n\t\t\tdamagedEffects = defender.activateAbilities('damaged');\r\n\t\t\tdefendEffects = defendEffects.concat(damagedEffects);\r\n\t\t}\r\n\t\tif (defendEffects.includes('cancelDamage') || attackEffects.includes('cancelDamage')) {\r\n\t\t\tLevel.removeEffects(attackEffects, 'damage');\r\n\t\t\tLevel.removeEffects(attackEffects, 'weaponDamage');\r\n\t\t}\r\n\r\n\t\tconsole.log(attacker.name, JSON.stringify(attackEffects), 'vs', JSON.stringify(defendEffects));\r\n\r\n\t\tconst outcomeAttack = this.doEffects(attackEffects, attacker, defender);\r\n\t\tconst outcomeDefend = this.doEffects(defendEffects, defender, attacker);\r\n\r\n\t\t// TODO: generate messages\r\n\r\n\t\treturn { outcomeAttack, outcomeDefend, attackEffects, defendEffects };\r\n\t}\r\n\r\n\tdoEffects(effects, actor, opponent) {\r\n\t\tlet damage = 0;\r\n\t\tif (!effects) { return { damage }; }\r\n\t\teffects.forEach((effect) => {\r\n\t\t\tdamage += this.doEffect(effect, actor, opponent);\r\n\t\t});\r\n\t\treturn { damage };\r\n\t}\r\n\r\n\tdoEffect(effect, actor, opponent) {\r\n\t\tconsole.log('doEffect', effect);\r\n\t\tlet damage = 0;\r\n\t\tswitch(effect) {\r\n\t\t\tcase 'damage':\r\n\t\t\t\tdamage += 1;\r\n\t\t\t\topponent.wound(1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'weaponDamage':\r\n\t\t\t\tdamage += actor.getWeaponDamage();\r\n\t\t\t\topponent.wound(damage);\r\n\t\t\tbreak;\r\n\t\t\tcase 'heal':\r\n\t\t\tcase 'hp':\r\n\t\t\t\tactor.heal(1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'ap':\r\n\t\t\t\tactor.healPool('ap', 1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'bp':\r\n\t\t\t\tactor.healPool('bp', 1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'ep':\r\n\t\t\t\tactor.healPool('ep', 1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'moveSwitch': {\r\n\t\t\t\tconst { x, y } = actor;\r\n\t\t\t\tactor.setCoordinates(opponent.x, opponent.y);\r\n\t\t\t\topponent.setCoordinates(x, y);\r\n\t\t\t}\r\n\t\t\tbreak;\r\n\t\t\tcase 'apDamage':\r\n\t\t\t\tactor.damagePool('ap', 1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'bpDamage':\r\n\t\t\t\tactor.damagePool('bp', 1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'epDamage':\r\n\t\t\t\tactor.damagePool('ep', 1);\r\n\t\t\tbreak;\r\n\t\t\tcase 'push':\r\n\t\t\t\tthis.pushActor(actor, opponent);\r\n\t\t\tbreak;\r\n\t\t\tcase 'pushAoe':\r\n\t\t\t\tthis.pushActor(actor, opponent);\r\n\t\t\t\t// TODO: Handle aoe --> everyone (accept opponent) around hitX, hitY gets knocked back\r\n\t\t\tbreak;\r\n\t\t\tcase 'moveBack':\r\n\t\t\t\tthis.pushActor(opponent, actor);\r\n\t\t\tbreak;\r\n\t\t\tcase 'initiative':\r\n\t\t\t\tactor.initiativeBoost = 1;\r\n\t\t\tbreak;\r\n\t\t\tcase \"fire\":\r\n\t\t\t\t// TODO:\r\n\t\t\tbreak;\r\n\t\t\tcase \"endGame\":\r\n\t\t\t\t// TODO\r\n\t\t\tbreak;\r\n\t\t\tcase \"score1000\":\r\n\t\t\t\tactor.score += 1000;\r\n\t\t\tbreak;\r\n\t\t\tdefault:\r\n\t\t\t\tthis.doCustomEffect(effect, actor, opponent);\r\n\t\t\tbreak;\r\n\t\t}\r\n\t\treturn damage;\r\n\t}\r\n\r\n\tdoCustomEffect(effect, actor, opponent) {\r\n\t\tif (typeof this.customEffects[effect] === 'function') {\r\n\t\t\tthis.customEffects[effect](effect, actor, opponent);\r\n\t\t}\r\n\t}\r\n\r\n\tpushActor(pusher, pushee) {\r\n\t\tconst { x, y } = pusher;\r\n\t\tlet moveX = pushee.x - x;\r\n\t\tlet moveY = pushee.y - y;\r\n\t\tmoveX = moveX / (moveX === 0 ? 1 : Math.abs(moveX));\r\n\t\tmoveY = moveY / (moveY === 0 ? 1 : Math.abs(moveY));\r\n\t\t// console.log('pushing', pushee.name, moveX, moveY);\r\n\t\tpushee.move(moveX, moveY);\r\n\t}\r\n\r\n\tgetActorsInitiativeOrdered() {\r\n\t\tconst randomActors = random.shuffle(this.actors);\r\n\t\tconst firstActors = [];\r\n\t\tlet i = randomActors.length - 1;\r\n\t\twhile (i--) {\r\n\t\t\tconst actor = randomActors[i];\r\n\t\t\tif (actor.initiativeBoost > 0) {\r\n\t\t\t\tfirstActors.push(actor);\r\n\t\t\t\trandomActors.splice(i, 1);\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn firstActors.concat(randomActors);\r\n\t}\r\n\r\n\tcoolOffInitiativeBoosts() {\r\n\t\tthis.actors.forEach((actor) => {\r\n\t\t\tactor.initiativeBoost = 0;\r\n\t\t});\r\n\t}\r\n\r\n\t// Generation\r\n\r\n\tgenerateItem(levelItem = {}, Class, seed = 0, types = [], background = undefined) {\r\n\t\tconst { x, y } = this.findRandomFreeCell(seed, levelItem.clearing);\r\n\t\tconst itemTypeOptions = (levelItem.type && types[levelItem.type]) ? types[levelItem.type] : {};\r\n\t\tconst itemOptions = {\r\n\t\t\tx, y,\r\n\t\t\t...itemTypeOptions,\r\n\t\t\t...levelItem\r\n\t\t};\r\n\t\tif (background) { itemOptions.background = background; }\r\n\t\tconst item = new Class(itemOptions);\r\n\t\treturn item;\r\n\t}\r\n\r\n\tgenerateItems(options = {}, itemTypes = {}) {\r\n\t\tlet seed = this.seed + 200;\r\n\t\tlet { items = [] } = options;\r\n\r\n\t\tconst arr = [];\r\n\t\titems.forEach((levelItem) => {\r\n\t\t\tconst quantity = (typeof levelItem.quantity === 'number') ? levelItem.quantity : 1;\r\n\t\t\t// TODO: handle weight, etc.\r\n\t\t\tfor (let i = 0; i < quantity; i++) {\r\n\t\t\t\tconst item = this.generateItem(levelItem, Item, ++seed, itemTypes);\r\n\t\t\t\tarr.push(item);\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn arr;\r\n\t}\r\n\r\n\tgenerateProps(options = {}, propTypes = {}) {\r\n\t\tlet seed = this.seed + 100;\r\n\t\tlet { props = [] } = options;\r\n\t\tconst background = this.background;\r\n\r\n\t\tconst arr = [];\r\n\t\tprops.forEach((levelProp) => {\r\n\t\t\tconst quantity = (typeof levelProp.quantity === 'number') ? levelProp.quantity : 1;\r\n\t\t\t// TODO: handle weight, etc.\r\n\t\t\tfor (let i = 0; i < quantity; i++) {\r\n\t\t\t\tconst prop = this.generateItem(levelProp, Prop, ++seed, propTypes, background);\r\n\t\t\t\tarr.push(prop);\r\n\t\t\t}\r\n\t\t});\r\n\t\treturn arr;\r\n\t}\r\n\r\n\tgenerateActors(options = {}, refData = {}) {\r\n\t\tconsole.log('generateActors', options);\r\n\t\tlet seed = this.seed + 999; // ?\r\n\t\tconst { monsterSpawn, monsters = [] } = options;\r\n\t\tconst monsterTypes = refData.monsters;\r\n\t\tconst depth = options.levelIndex;\r\n\t\tconst availableMonsters = monsters.filter((levelMonster) => {\r\n\t\t\treturn (!levelMonster.minDepth) || levelMonster.minDepth <= depth;\r\n\t\t});\r\n\t\tconst availableMonsterWeights = {};\r\n\t\tavailableMonsters.forEach((levelMonster) => {\r\n\t\t\tif (levelMonster.weight) {\r\n\t\t\t\tavailableMonsterWeights[levelMonster.type] = levelMonster.weight;\r\n\t\t\t}\r\n\t\t});\r\n\t\tconst hasMonstersWithWeights = Object.keys(availableMonsterWeights).length > 0;\r\n\t\tconst monsterSpawnNumber = (monsterSpawn === undefined) ? 10 : random.roll(monsterSpawn);\r\n\t\tconst totalMonsterSpawnQuantity = monsterSpawnNumber;\r\n\t\tconst actors = [];\r\n\t\t// Create monsters with fixed quantities\r\n\t\t// Note: this could exceed the total quantity\r\n\t\tconst availableMonstersFixedQuantities = availableMonsters.filter((levelMonster) => {\r\n\t\t\treturn levelMonster.quantity;\r\n\t\t});\r\n\t\tavailableMonstersFixedQuantities.forEach((levelMonster) => {\r\n\t\t\tconst monsterTypeKey = levelMonster.type;\r\n\t\t\tfor (let i = 0; i < levelMonster.quantity; i++) {\r\n\t\t\t\tconst monster = this.createActor(monsterTypes, monsterTypeKey, ++seed);\r\n\t\t\t\tactors.push(monster);\r\n\t\t\t}\r\n\t\t});\r\n\t\t// Create weighted monsters\r\n\t\tif (hasMonstersWithWeights) {\r\n\t\t\tlet stopper = 0;\r\n\t\t\twhile (actors.length < totalMonsterSpawnQuantity && stopper < 9000) {\r\n\t\t\t\tstopper++;\r\n\t\t\t\t(() => {\r\n\t\t\t\t\tconst monsterTypeKey = random.getWeightedValue(availableMonsterWeights);\r\n\t\t\t\t\tif (!monsterTypeKey) { return; }\r\n\t\t\t\t\tconst monster = this.createActor(monsterTypes, monsterTypeKey, ++seed);\r\n\t\t\t\t\tactors.push(monster);\r\n\t\t\t\t})();\r\n\t\t\t}\r\n\t\t}\r\n\t\t// console.log('Actors at depth', depth, actors);\r\n\t\treturn actors;\r\n\t}\r\n\r\n\tcreateActor(monsterTypes, monsterTypeKey, seed) {\r\n\t\tif (!monsterTypeKey) { return; }\r\n\t\tconst { x, y } = this.findRandomFreeCell(seed);\r\n\t\tlet monsterOptions = monsterTypes[monsterTypeKey];\r\n\t\tmonsterOptions = { type: monsterTypeKey, aggro: 100, ...monsterOptions, x, y };\r\n\t\t// console.log(monsterTypes, monsterTypeKey, monsterOptions);\r\n\t\treturn new Actor(monsterOptions);\t\t\r\n\t}\r\n\r\n\t// Gets\r\n\r\n\tgetMap() {\r\n\t\treturn this.map;\r\n\t}\r\n\r\n\tgetCellPassability(x, y) {\r\n\t\tconst isMapPassable = this.map.getCellPassability(x, y);\r\n\t\tif (!isMapPassable) { return false; }\r\n\t\tconst actorsHere = this.actors.filter((actor) => {\r\n\t\t\treturn actor.x === x && actor.y === y && !actor.passable;\r\n\t\t});\r\n\t\treturn (actorsHere.length === 0);\r\n\t}\r\n\r\n\t// Sets\r\n\r\n\tsetEye(actorThing) {\r\n\t\tthis.eye.x = actorThing.x;\r\n\t\tthis.eye.y = actorThing.y;\r\n\t\tthis.eye.sightRange = actorThing.sightRange;\r\n\t}\r\n}\r\n\r\nmodule.exports = Level;\r\n","const ROT = require('rot-js');\r\nconst Cell = require('./Cell');\r\nconst geometer = require('./geometer');\r\nconst random = require('./random');\r\n\r\nconst DIGGER_TYPE = 'digger';\r\n\r\nclass Map {\r\n\tconstructor(options = {}) {\r\n\t\tthis.baseSeed = options.seed || 1;\r\n\t\tthis.seed = this.baseSeed;\r\n\t\tthis.color = options.color || '#777';\r\n\t\tthis.background = options.background || '#222';\r\n\t\tthis.type = options.type || DIGGER_TYPE;\r\n\t\tthis.rotMap = options.rotMap;\r\n\t\tthis.cells = {};\r\n\t\tthis.freeCells = [];\r\n\t\tthis.walls = Boolean(options.walls) || Boolean(options.wallsCharacter);\r\n\t\tthis.wallsCharacter = options.wallsCharacter || '#'; // ▧\r\n\t\tthis.floorCharacter = options.floorCharacter || '.';\r\n\t\tthis.generate(options);\r\n\t}\r\n\r\n\tgenerate(options = {}) {\r\n\t\tconst generators = options.generators || {};\r\n\r\n\t\tif (typeof generators[this.type] === 'function') {\r\n\t\t\tthis.clearCells();\r\n\t\t\tgenerators[this.type](this.seed, this, options);\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tif (this.type === DIGGER_TYPE) {\r\n\t\t\tthis.generateDigger();\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (this.type === ARENA_TYPE) {\r\n\t\t\tthis.generateArena(options.x, options.y);\r\n\t\t\treturn;\r\n\t\t}\r\n\t\t// TODO: handle other rot-js types\r\n\r\n\t\tconsole.warn('Undefined map type:', this.type, generators);\r\n\t\tthis.generateArena(3, 3);\r\n\t\t// TODO: Have default be a big empty room instead?\r\n\t}\r\n\r\n\tgenerateArena(x, y) {\r\n\t\tROT.RNG.setSeed(this.seed);\r\n\t\tthis.rotMap = new ROT.Map.Arena(x, y);\r\n\t\tthis.setupRotMap();\t\r\n\t}\r\n\r\n\tgenerateDigger() {\r\n\t\tROT.RNG.setSeed(this.seed);\r\n\t\tthis.rotMap = new ROT.Map.Digger();\r\n\t\tthis.setupRotMap();\t\t\r\n\t}\r\n\r\n\tsetupRotMap() {\r\n\t\tthis.clearCells();\r\n\t\tthis.rotMap.create((x, y, value) => {\r\n\t\t\tif (value) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tthis.setFloorAt(x, y);\r\n\t\t});\r\n\r\n\t\tif (this.walls) {\r\n\t\t\tthis.addWalls();\r\n\t\t}\t\t\r\n\t}\r\n\r\n\taddWalls() {\r\n\t\tthis.forEachCell((cell, x, y) => {\r\n\t\t\tMap.forEachDirection((dir, dirX, dirY) => {\r\n\t\t\t\tconst newX = x + dirX;\r\n\t\t\t\tconst newY = y + dirY;\r\n\t\t\t\tconst wallCell = this.getCellAt(newX, newY);\r\n\t\t\t\tif (!wallCell) {\r\n\t\t\t\t\tthis.setWallAt(newX, newY);\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t});\r\n\t}\r\n\r\n\tdiscoverCircle(x, y, radius) {\r\n\t\tthis.forEachCellInCircle(x, y, radius, (cell) => {\r\n\t\t\tcell.discover();\r\n\t\t});\r\n\t}\r\n\r\n\tstatic parseKeyCoordinates(key) {\r\n\t\tconst parts = key.split(\",\");\r\n\t\tconst x = parseInt(parts[0]);\r\n\t\tconst y = parseInt(parts[1]);\r\n\t\treturn { x, y };\r\n\t}\r\n\r\n\tstatic makeKey(x, y) {\r\n\t\treturn x + ',' + y;\r\n\t}\r\n\r\n\tstatic forEachDirection(callback) {\r\n\t\tconst dirCoords = [\r\n\t\t\t{x: 0, y: -1}, // top\r\n\t\t\t{x: 1, y: -1},\r\n\t\t\t{x: 1, y: 0}, // right\r\n\t\t\t{x: 1, y: 1},\r\n\t\t\t{x: 0, y: 1}, // bottom\r\n\t\t\t{x: -1, y: 1},\r\n\t\t\t{x: -1, y: 0}, // left\r\n\t\t\t{x: -1, y: -1},\r\n\t\t];\r\n\t\tfor (let i = 0; i < 8; i++) {\r\n\t\t\tcallback(i, dirCoords[i].x, dirCoords[i].y);\r\n\t\t}\r\n\t}\r\n\r\n\tclearCells() {\r\n\t\tthis.cells = {};\r\n\t\tthis.freeCells.length = 0;\t\t\r\n\t}\r\n\r\n\tforEachCell(callback) {\r\n\t\tfor (let key in this.cells) {\r\n\t\t\tconst { x, y } = Map.parseKeyCoordinates(key);\r\n\t\t\tcallback(this.cells[key], x, y, key);\r\n\t\t}\r\n\t}\r\n\r\n\tforEachCellInCircle(centerX, centerY, radius, callback, includeEmptyCells = false) {\r\n\t\tconst maxX = centerX + radius;\r\n\t\tconst maxY = centerY + radius;\r\n\t\tlet x;\r\n\t\tfor (x = centerX - radius; x <= maxX; x++) {\r\n\t\t\tlet y;\r\n\t\t\tfor (y = centerY - radius; y <= maxY; y++) {\r\n\t\t\t\tconst r = Math.round(geometer.getDistance(centerX, centerY, x, y));\r\n\t\t\t\tif (r < radius) {\r\n\t\t\t\t\tconst cell = this.getCellAt(x, y);\r\n\t\t\t\t\tif (cell || includeEmptyCells) {\r\n\t\t\t\t\t\tcallback(cell, x, y)\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\tgetRandomFreeCell(seed) {\t\t\r\n\t\tconst i = random.roll(this.freeCells.length, seed);\r\n\t\t\r\n\t\t// TODO: TBD- Is it still a free cell?\r\n\t\t// var key = freeCells.splice(index, 1)[0];\r\n\t\t// this.map[key] = \"*\";\r\n\t\tconst key = this.freeCells[i];\r\n\t\tconst cell = this.cells[key];\r\n\t\t\r\n\t\tconst { x, y } = Map.parseKeyCoordinates(key);\r\n\t\t// console.log(seed, key, i, x, y);\r\n\t\treturn { x, y, cell };\r\n\t}\r\n\r\n\tgetCellAt(x, y) {\r\n\t\tconst key = Map.makeKey(x, y);\r\n\t\treturn this.cells[key];\t\t\r\n\t}\r\n\r\n\tgetCharacterAt(x, y) {\r\n\t\tconst cell = this.getCellAt(x, y);\r\n\t\treturn (cell) ? cell.getCharacter() : null;\r\n\t}\r\n\r\n\tsetFloorAt(x, y) {\r\n\t\tconst key = this.setCharacterAt(this.floorCharacter, x, y);\r\n\t\tthis.freeCells.push(key);\r\n\t\treturn key;\r\n\t}\r\n\r\n\tsetWallAt(x, y) {\r\n\t\treturn this.setCharacterAt(this.wallsCharacter, x, y);\r\n\t}\r\n\r\n\tsetCharacterAt(char, x, y) {\r\n\t\tconst key = Map.makeKey(x, y);\r\n\t\tconst cell = this.cells[key];\r\n\t\tif (cell) {\r\n\t\t\tcell.setCharacter(char);\r\n\t\t} else {\r\n\t\t\tconst { color, background } = this;\r\n\t\t\tthis.cells[key] = new Cell({ color, background, character: char });\r\n\t\t}\r\n\t\treturn key;\r\n\t}\r\n\r\n\tgetCellPassability(x, y) {\r\n\t\tconst cell = this.getCellAt(x, y);\r\n\t\treturn (cell) ? cell.getPassability() : false;\r\n\t}\r\n\r\n\tgetLightingAt(x, y) {\r\n\t\treturn {}; // TODO\r\n\t}\r\n\r\n\t// _generateBoxes(freeCells) {\r\n\t// \tfor (var i=0;i<10;i++) {\r\n\t// \t\tvar index = Math.floor(ROT.RNG.getUniform() * freeCells.length);\r\n\t// \t\tvar key = freeCells.splice(index, 1)[0];\r\n\t// \t\tthis.map[key] = \"*\";\r\n\t// \t}\r\n\t// }\r\n}\r\n\r\nmodule.exports = Map;\r\n","class Inventory {\r\n\tconstructor(options = {}) {\r\n\t\tthis.size = (typeof options.size === 'number') ? options.size : 10;\r\n\t\tthis.items = [];\r\n\t}\r\n\r\n\tisFull() {\r\n\t\treturn (this.items.length >= this.size);\r\n\t}\r\n\r\n\thasSpace() {\r\n\t\treturn !this.isFull();\r\n\t}\r\n\r\n\tadd(item) {\r\n\t\tif (this.isFull()) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tif (!item.portable) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tthis.items.push(item);\r\n\t\treturn true;\r\n\t}\r\n\r\n\tremove(item) {\r\n\t\tconst i = this.items.indexOf(item);\r\n\t\tif (i <= -1) {\r\n\t\t\tconsole.warn('nothing found in', this.items, item);\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tconst arr = this.items.splice(i, 1);\r\n\t\treturn arr[0];\r\n\t}\r\n\r\n\tremoveType(typeKey) {\r\n\t\tconst itemsOfType = this.items.filter((item) => { return item.type === typeKey; });\r\n\t\tif (itemsOfType.length === 0) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\treturn this.remove(itemsOfType[0]);\r\n\t}\r\n\r\n\tget(n) {\r\n\t\tif (typeof n === 'number') {\r\n\t\t\treturn this.items[n];\r\n\t\t} else if (typeof n === 'string') {\r\n\t\t\treturn this.items.find((item) => { return item.name === n; });\r\n\t\t}\r\n\t\treturn this.items;\r\n\t}\r\n\r\n\tgetString() {\r\n\t\tconst arr = this.items.map((item, i) => { return `[${(i + 1)}] ${item.name}`; });\r\n\t\treturn (arr.length) ? arr.join(', ') : 'nothing';\r\n\t}\r\n\r\n\tloopOverContents(fn) {\r\n\t\tthis.items.forEach((item, i) => {\r\n\t\t\tfn(item, i);\r\n\t\t});\r\n\t}\r\n\r\n\thasContents() {\r\n\t\treturn (this.items.length > 0);\r\n\t}\r\n\r\n\tcontains(itemName) {\r\n\t\tlet foundItem = this.items.find((item) => { return (item.name === itemName); });\r\n\t\treturn Boolean(foundItem);\r\n\t}\r\n\r\n\tcontainsType(typeName) {\r\n\t\tlet foundItem = this.items.find((item) => { return (item.type === typeName); });\r\n\t\treturn Boolean(foundItem);\r\n\t}\r\n}\r\n\r\nmodule.exports = Inventory;\r\n","\r\nmodule.exports = {\r\n\tDIRS_4: Object.freeze([\r\n\t\t{ x: 0, y: -1 },\r\n\t\t{ x: 1, y: 0 },\r\n\t\t{ x: 0, y: 1 },\r\n\t\t{ x: -1, y: 0 },\r\n\t]),\r\n\tDIRS_8: Object.freeze([\r\n\t\t{ x: 0, y: -1 },\r\n\t\t{ x: 1, y: -1 },\r\n\t\t{ x: 1, y: 0 },\r\n\t\t{ x: 1, y: 1 },\r\n\t\t{ x: 0, y: 1 },\r\n\t\t{ x: -1, y: 1 },\r\n\t\t{ x: -1, y: 0 },\r\n\t\t{ x: -1, y: -1 },\t\r\n\t]),\r\n\tDIRS_8_DIAGNOLS: Object.freeze([\r\n\t\t{ x: 1, y: -1 },\r\n\t\t{ x: 1, y: 1 },\r\n\t\t{ x: -1, y: 1 },\r\n\t\t{ x: -1, y: -1 },\t\r\n\t]),\r\n};\r\n","const Item = require('./Item');\r\n\r\nclass Prop extends Item {\r\n\tconstructor(options = {}) {\r\n\t\toptions = { portable: false, ...options };\r\n\t\tsuper(options);\r\n\t}\r\n}\r\n\r\nmodule.exports = Prop;\r\n","const ROT = require('rot-js');\r\nconst Game = require('./Game');\r\nconst Item = require('./Item');\r\nconst Map = require('./Map');\r\nconst Actor = require('./Actor');\r\nconst Prop = require('./Prop');\r\nconst Level = require('./Level');\r\nconst Display = require('./Display');\r\nconst random = require('./random');\r\nconst ready = require('./ready');\r\n\r\nconst rote = {\r\n    ROT,\r\n    Game, Level, Map, Item, Prop, Actor, Display,\r\n    random,\r\n    ready\r\n};\r\n\r\nif (window) {\r\n    window.rote = rote;\r\n}\r\n\r\nmodule.exports = rote;\r\n","// shim for using process in browser\nvar process = module.exports = {};\n\n// cached from whatever global is present so that test runners that stub it\n// don't break things.  But we need to wrap it in a try catch in case it is\n// wrapped in strict mode code which doesn't define any globals.  It's inside a\n// function because try/catches deoptimize in certain engines.\n\nvar cachedSetTimeout;\nvar cachedClearTimeout;\n\nfunction defaultSetTimout() {\n    throw new Error('setTimeout has not been defined');\n}\nfunction defaultClearTimeout () {\n    throw new Error('clearTimeout has not been defined');\n}\n(function () {\n    try {\n        if (typeof setTimeout === 'function') {\n            cachedSetTimeout = setTimeout;\n        } else {\n            cachedSetTimeout = defaultSetTimout;\n        }\n    } catch (e) {\n        cachedSetTimeout = defaultSetTimout;\n    }\n    try {\n        if (typeof clearTimeout === 'function') {\n            cachedClearTimeout = clearTimeout;\n        } else {\n            cachedClearTimeout = defaultClearTimeout;\n        }\n    } catch (e) {\n        cachedClearTimeout = defaultClearTimeout;\n    }\n} ())\nfunction runTimeout(fun) {\n    if (cachedSetTimeout === setTimeout) {\n        //normal enviroments in sane situations\n        return setTimeout(fun, 0);\n    }\n    // if setTimeout wasn't available but was latter defined\n    if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {\n        cachedSetTimeout = setTimeout;\n        return setTimeout(fun, 0);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedSetTimeout(fun, 0);\n    } catch(e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally\n            return cachedSetTimeout.call(null, fun, 0);\n        } catch(e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error\n            return cachedSetTimeout.call(this, fun, 0);\n        }\n    }\n\n\n}\nfunction runClearTimeout(marker) {\n    if (cachedClearTimeout === clearTimeout) {\n        //normal enviroments in sane situations\n        return clearTimeout(marker);\n    }\n    // if clearTimeout wasn't available but was latter defined\n    if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {\n        cachedClearTimeout = clearTimeout;\n        return clearTimeout(marker);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedClearTimeout(marker);\n    } catch (e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't  trust the global object when called normally\n            return cachedClearTimeout.call(null, marker);\n        } catch (e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.\n            // Some versions of I.E. have different rules for clearTimeout vs setTimeout\n            return cachedClearTimeout.call(this, marker);\n        }\n    }\n\n\n\n}\nvar queue = [];\nvar draining = false;\nvar currentQueue;\nvar queueIndex = -1;\n\nfunction cleanUpNextTick() {\n    if (!draining || !currentQueue) {\n        return;\n    }\n    draining = false;\n    if (currentQueue.length) {\n        queue = currentQueue.concat(queue);\n    } else {\n        queueIndex = -1;\n    }\n    if (queue.length) {\n        drainQueue();\n    }\n}\n\nfunction drainQueue() {\n    if (draining) {\n        return;\n    }\n    var timeout = runTimeout(cleanUpNextTick);\n    draining = true;\n\n    var len = queue.length;\n    while(len) {\n        currentQueue = queue;\n        queue = [];\n        while (++queueIndex < len) {\n            if (currentQueue) {\n                currentQueue[queueIndex].run();\n            }\n        }\n        queueIndex = -1;\n        len = queue.length;\n    }\n    currentQueue = null;\n    draining = false;\n    runClearTimeout(timeout);\n}\n\nprocess.nextTick = function (fun) {\n    var args = new Array(arguments.length - 1);\n    if (arguments.length > 1) {\n        for (var i = 1; i < arguments.length; i++) {\n            args[i - 1] = arguments[i];\n        }\n    }\n    queue.push(new Item(fun, args));\n    if (queue.length === 1 && !draining) {\n        runTimeout(drainQueue);\n    }\n};\n\n// v8 likes predictible objects\nfunction Item(fun, array) {\n    this.fun = fun;\n    this.array = array;\n}\nItem.prototype.run = function () {\n    this.fun.apply(null, this.array);\n};\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\nprocess.version = ''; // empty string to avoid regexp issues\nprocess.versions = {};\n\nfunction noop() {}\n\nprocess.on = noop;\nprocess.addListener = noop;\nprocess.once = noop;\nprocess.off = noop;\nprocess.removeListener = noop;\nprocess.removeAllListeners = noop;\nprocess.emit = noop;\nprocess.prependListener = noop;\nprocess.prependOnceListener = noop;\n\nprocess.listeners = function (name) { return [] }\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n};\n\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\nprocess.umask = function() { return 0; };\n","const ROT = require('rot-js');\r\nconst ready = require('./ready');\r\nconst Display = require('./Display');\r\nconst Level = require('./Level');\r\nconst Actor = require('./Actor');\r\nconst Item = require('./Item');\r\nconst Keyboard = require('./KeyboardListener');\r\nconst MusicBox = require('./MusicBox');\r\nconst Console = require('./Console');\r\n\r\nconst INIT_STATE = 'INIT';\r\nconst MAIN_GAME_STATE = 'GAME';\r\nconst SPLASH_STATE = 'SPLASH';\r\nconst OFF_STATE = 'OFF';\r\n\r\nclass Game {\r\n\tconstructor(options) {\r\n\t\tconst { id, consoleId, data, customEffects,\thaveSplash,\tfontFamilies } = options;\r\n\t\tthis.id = id;\r\n\t\tthis.displayContainer = document.getElementById(id || 'display');\r\n\t\tthis.console = new Console({ id: consoleId });\r\n\t\tthis.display = null;\r\n\t\tthis.haveSplash = Boolean(haveSplash);\r\n\t\tthis.fontFamilies = fontFamilies || [];\r\n\t\tthis.activeLevelIndex = 0;\r\n\t\t// The generated levels\r\n\t\tthis.levels = [];\r\n\t\t// Custom funcitons for generating things\r\n\t\tthis.generators = options.generators || {};\r\n\t\t// Reference data on prototypical \"things\" (monsters, items)\r\n\t\tthis.data = {\r\n\t\t\tmonsters: {},\r\n\t\t\titems: {},\r\n\t\t\tprops: {},\r\n\t\t\tplaylist: [],\r\n\t\t};\r\n\t\t// The main actor\r\n\t\tthis.hero = null; // player character / player actor\r\n\t\t// Guts\r\n\t\tthis.scheduler = new ROT.Scheduler.Simple();\r\n\t\tthis.engine = null;\r\n\t\tthis.keyboard = null;\r\n\t\tthis.state = INIT_STATE;\r\n\t\tthis.states = new Set([INIT_STATE, SPLASH_STATE, MAIN_GAME_STATE, OFF_STATE]);\r\n\t\t// this.setupEngine();\r\n\t\tthis.loadingPromise = null;\r\n\t\tthis.console.setup();\r\n\t\tthis.loadData(data);\r\n\t\tthis.hooks = {};\r\n\t\tthis.customEffects = { ...customEffects };\r\n\t}\r\n\r\n\tsetupEngine() {\r\n\t\tthis.engine = new ROT.Engine(this.scheduler);\r\n\t\tthis.engine.start();\r\n\t\treturn this.engine;\r\n\t}\r\n\r\n\tsetupKeyboard() {\r\n\t\tthis.keyboard = new Keyboard({ state: MAIN_GAME_STATE, autoStart: true });\r\n\t\t// Splash state\r\n\t\tthis.keyboard.on(SPLASH_STATE, 'ENTER', () => {\r\n\t\t\tthis.setState(MAIN_GAME_STATE);\r\n\t\t});\r\n\t\t// Main state\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 'DIRECTION', (keyName, keyCode, direction) => {\r\n\t\t\t// TODO: Lock and unlock the game? or do something else to determine if it's OK to move\r\n\t\t\tthis.hero.queueAction('move', { direction });\r\n\t\t\tthis.advance();\r\n\t\t});\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 'ENTER', () => {\r\n\t\t\t// this.actorDefaultAction(this.hero); // TODO: Remove me\r\n\t\t\tthis.actorAddDefaultAction(this.hero);\r\n\t\t\tthis.advance();\r\n\t\t});\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 'SPACE', () => {\r\n\t\t\tthis.hero.queueAction('wait');\r\n\t\t\tthis.advance();\r\n\t\t});\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 't', () => {\r\n\t\t\tthis.showInventory();\r\n\t\t\tthis.print('> Throw which item?');\r\n\t\t\tlet n = prompt('Throw which item? \\n\\n' + this.hero.inventory.getString());\r\n\t\t\tif (!n || n.length === 0) {\r\n\t\t\t\tthis.print('None');\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tn = parseInt(n, 10);\r\n\t\t\tconst i = (isNaN(n)) ? -1 : n - 1;\r\n\t\t\tconst item = this.hero.inventory.get(i);\r\n\t\t\tif (item) {\r\n\t\t\t\tthis.hero.queueAction('throw', { what: item, x: this.hero.x, y: this.hero.y });\r\n\t\t\t\tthis.advance();\r\n\t\t\t} else {\r\n\t\t\t\tthis.print(`Invalid item [${n}]`);\r\n\t\t\t}\r\n\t\t});\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 'i', () => { this.showInventory(); });\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 'p', () => {\r\n\t\t\tthis.hero.queueAction('pickup');\r\n\t\t\tthis.advance();\r\n\t\t});\r\n\t\tthis.keyboard.on(MAIN_GAME_STATE, 'o', () => {\r\n\t\t\tthis.hero.queueAction('look');\r\n\t\t\tthis.advance();\r\n\t\t});\r\n\t\tfor (let i = 0; i < 9; i++) {\r\n\t\t\tconst key = String(i + 1);\r\n\t\t\tthis.keyboard.on(MAIN_GAME_STATE, key, () => {\r\n\t\t\t\tthis.hero.readyAbilityByIndex(i);\r\n\t\t\t\tthis.draw();\r\n\t\t\t});\r\n\t\t}\r\n\t\t// this.keyboard.start();\r\n\t}\r\n\r\n\tsetupMusic() {\r\n\t\tthis.music = new MusicBox(this.data.playlist);\r\n\t}\r\n\r\n\tremoveKeyboard() {\r\n\t\t// TODO: this.keyboard.off() on all listeners\r\n\t}\r\n\r\n\tcreateDisplay(options = {}) {\r\n\t\tthis.display = new Display(options);\r\n\t\tthis.display.setupElements();\r\n\t}\r\n\r\n\t//---- Draw / Render\r\n\r\n\tprint(str, classes = '', wait = 0) {\r\n\t\tif (wait) {\r\n\t\t\tsetTimeout(() => { this.print(str, classes); }, wait);\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tthis.console.print(str, classes);\r\n\t}\r\n\r\n\tshowInventory() {\r\n\t\tconst items = this.hero.inventory.getString();\r\n\t\tthis.print('Inventory: ' + items);\r\n\t}\r\n\r\n\tdraw() {\r\n\t\tthis.display.drawLevel(this, this.getActiveLevel(), this.hero);\r\n\t}\r\n\r\n\r\n\t//---- Generation\r\n\r\n\tcreateLevel(options = {}, seed) {\r\n\t\toptions.seed = seed || options.seed;\r\n\t\tconst levelOptions = {\r\n\t\t\tcustomEffects: this.customEffects,\r\n\t\t\t...options,\r\n\t\t\tlevelIndex: this.levels.length,\r\n\t\t\tgenerators: this.generators,\r\n\t\t};\r\n\t\t// console.warn(this.customEffects, levelOptions);\r\n\t\tconst level = new Level(levelOptions, this.data);\r\n\t\tthis.levels.push(level);\r\n\t\treturn level;\r\n\t}\r\n\r\n\tcreateLevels(arr = [], baseSeed = 1) {\r\n\t\tlet seed = baseSeed;\r\n\t\tarr.forEach((item, i) => {\r\n\t\t\tseed += i;\r\n\t\t\tif (typeof item === 'string') { // level type key\r\n\t\t\t\tthis.createLevel(this.getLevelType(item), seed);\r\n\t\t\t} else if (typeof item === 'object' && item !== null) {\r\n\t\t\t\tconst n = (typeof item.repeat === 'number') ? item.repeat : 1;\r\n\t\t\t\tfor (let r = 0; r < n; r++) {\r\n\t\t\t\t\tseed += r;\r\n\t\t\t\t\tthis.createLevel(this.getLevelType(item.levelTypeKey), seed);\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t});\r\n\t\tthis.connectStairs();\r\n\t\treturn this.levels;\r\n\t}\r\n\r\n\tcreateActor(options = {}, level) {\r\n\t\tconst actor = new Actor(options);\r\n\t\tthis.scheduler.add(actor, true);\r\n\t\tlevel = (level === true) ? this.getActiveLevel() : level;\r\n\t\tif (level) {\r\n\t\t\tlevel.addActor(actor);\r\n\t\t}\r\n\t\treturn actor;\r\n\t}\r\n\r\n\tcreateItem(options = {}, level) {\r\n\t\tconst item = new Item(options);\r\n\t\tlevel = (level === true) ? this.getActiveLevel() : level;\r\n\t\tif (level) {\r\n\t\t\tlevel.addItem(item);\r\n\t\t}\r\n\t\treturn item;\r\n\t}\r\n\r\n\tcreateHero(options = {}) {\r\n\t\tconst heroOptions = { ...options, character: '@', isHero: true };\r\n\t\tthis.hero = this.createActor(heroOptions, true);\r\n\r\n\t\tconst g = this;\r\n\t\t// Setup action stuff ... this needs to be refactored\r\n\t\tthis.hero.act = function () {\r\n\t\t\tg.engine.lock();\r\n\t\t\twindow.addEventListener('keydown', this); // pass the hero; the `handleEvent` will be used\r\n\t\t};\r\n\t\tthis.hero.handleEvent = function (e) { // Leftover from tutorial, part 2\r\n\t\t\twindow.removeEventListener('keydown', this);\r\n\t\t\tg.engine.unlock();\r\n\t\t};\r\n\t\tif (this.display) {\r\n\t\t\tthis.display.setCameraTarget(this.hero);\r\n\t\t}\r\n\t\tthis.discoverAroundHero();\r\n\t\treturn this.hero;\r\n\t}\r\n\r\n\tconnectStairs() {\r\n\t\tconst STAIR_LINK = 'stairLink';\r\n\t\tconst propTypes = this.getDataPropArray();\r\n\t\tconst stairsDownTypes = propTypes.filter((propType) => { return Boolean(propType[STAIR_LINK]); });\r\n\t\tthis.levels.forEach((level, i) => {\r\n\t\t\t// Handle each type of stairs\r\n\t\t\tstairsDownTypes.forEach((stairsDownType) => {\r\n\t\t\t\tconst stairDownTypeKey = stairsDownType.key;\r\n\t\t\t\tconst stairUpTypeKey = stairsDownType[STAIR_LINK];\r\n\t\t\t\tconst levelStairsDown = level.props.filter((prop) => {\r\n\t\t\t\t\treturn prop.type === stairDownTypeKey;\r\n\t\t\t\t});\r\n\t\t\t\tlevelStairsDown.forEach((stair) => {\r\n\t\t\t\t\tconst levelBelow = this.levels[i + 1];\r\n\t\t\t\t\tif (!levelBelow) { return; }\r\n\t\t\t\t\tconst possibleStairsUp = levelBelow.props.filter((prop) => {\r\n\t\t\t\t\t\treturn prop.type === stairUpTypeKey && !Boolean(prop.teleport);\r\n\t\t\t\t\t});\r\n\t\t\t\t\t// TODO: Find stairs to connect to based on proximity\r\n\t\t\t\t\tconst levelBelowStairsUp = possibleStairsUp[0]; // TODO: remove this\r\n\t\t\t\t\tthis.connectTeleportProps(levelBelowStairsUp, stair, i, i + 1, 'ascend', 'descend');\r\n\t\t\t\t});\r\n\t\t\t});\r\n\t\t});\r\n\t}\r\n\r\n\tconnectTeleportProps(prop1, prop2, levelIndex1, levelIndex2, verb1, verb2) {\r\n\t\tif (!prop1 || !prop2) { return; }\r\n\t\tprop1.setTeleport({\r\n\t\t\tlevelIndex: levelIndex1, x: prop2.x, y: prop2.y, verb: verb1\r\n\t\t});\r\n\t\tprop2.setTeleport({\r\n\t\t\tlevelIndex: levelIndex2, x: prop1.x, y: prop1.y, verb: verb2\r\n\t\t});\r\n\t}\r\n\r\n\t//---- Movement, Combat\r\n\r\n\tmoveActor(actor, direction, bumpCombat = false) {\r\n\t\tconst diff = ROT.DIRS[8][direction];\r\n\t\tvar newX = actor.x + diff[0];\r\n\t\tvar newY = actor.y + diff[1];\r\n\t\treturn this.moveActorTo(actor, newX, newY, bumpCombat);\r\n\t}\r\n\r\n\tmoveActorTo(actor, x, y, bumpCombat = false) {\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tconst canMoveToCell = level.getCellPassability(x, y);\r\n\t\t// console.log('considering moving to', x, y, '... free?', canMoveToCell);\r\n\t\tif (!canMoveToCell) {\r\n\t\t\tconst blocker = level.findActor(x, y);\r\n\t\t\tif (blocker) {\r\n\t\t\t\treturn this.bump(actor, blocker, x, y, bumpCombat);\r\n\t\t\t}\r\n\t\t\treturn { x: x, y: y, moved: false };\r\n\t\t}\r\n\t\tactor.moveTo(x, y);\r\n\t\t// TODO: just redraw the space that was under the actor and the actor in the new spot?\r\n\t\tif (actor.isHero) {\r\n\t\t\tthis.discoverAroundHero();\r\n\t\t\tthis.narrateAroundHero();\r\n\t\t}\r\n\t\tthis.draw();\r\n\t\treturn { x, y, moved: true };\r\n\t}\r\n\r\n\tbump(actor, blocker, x, y, bumpCombat) {\r\n\t\tif (bumpCombat && actor.faction !== blocker.faction) {\r\n\t\t\tthis.resolveCombat(actor, blocker, x, y);\r\n\t\t\treturn { x, y, moved: false };\r\n\t\t} else if (Game.canBumpSwitch(actor, blocker)) {\r\n\t\t\tactor.moveTo(x, y);\r\n\t\t\treturn { x, y, moved: true };\r\n\t\t\t// TODO: allow pushes based on authority/size\r\n\t\t} else { // just blocked\r\n\t\t\treturn { x, y, moved: false };\r\n\t\t}\r\n\t}\r\n\r\n\tstatic canBumpSwitch(actor, blocker) {\r\n\t\tif (actor.aggro || blocker.aggro) { // TOOD: make this more nuanced\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tconst blockersNextAction = blocker.getNextAction();\r\n\t\tif (!blockersNextAction) { return true; }\r\n\t\treturn (\r\n\t\t\tblockersNextAction.verb === 'move' &&\r\n\t\t\tblockersNextAction.x === actor.x &&\r\n\t\t\tblockersNextAction.y === actor.y\r\n\t\t);\r\n\t}\r\n\r\n\tteleportActor(actor, teleportParams = {}) {\r\n\t\tconst originalLevelIndex = this.activeLevelIndex;\r\n\t\t// console.warn('teleporting', actor, teleportParams);\r\n\t\tconst { levelIndex, x, y } = teleportParams;\r\n\t\tconst currentLevel = this.getActiveLevel();\r\n\t\tcurrentLevel.removeActor(actor);\r\n\t\tthis.setActiveLevel(levelIndex);\r\n\t\tconst newLevel = this.getActiveLevel();\r\n\t\tnewLevel.addActor(actor);\r\n\t\tactor.setCoordinates(x, y);\r\n\t\tconsole.log('New Level:', newLevel);\r\n\t\tif (actor.isHero) {\r\n\t\t\tthis.discoverAroundHero();\r\n\t\t\tthis.narrateAroundHero();\r\n\t\t}\r\n\t\tif (originalLevelIndex !== levelIndex) {\r\n\t\t\tthis.hook('afterTeleportLevel', { levelIndex, x, y });\r\n\t\t}\r\n\t\t// this.draw();\r\n\t}\r\n\r\n\tresolveCombat(actor, opponent, x, y) {\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tif (!actor || !opponent || actor.faction === opponent.faction) {\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tconst { outcomeAttack } = level.resolveCombatEffects(actor, opponent);\r\n\t\t// TODO: get messages from resolve and effects methods\r\n\t\tg.print(`${actor.name} attacks ${opponent.name} and does ${outcomeAttack.damage} damage!`);\r\n\t\tif (opponent.dead()) {\r\n\t\t\tg.print(`${opponent.name} has been killed.`);\r\n\t\t\tactor.score += (this.activeLevelIndex + 1) * 10;\r\n\t\t}\r\n\t}\r\n\r\n\tactorAddDefaultAction(actor) {\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tconst thing = level.findThingSmart(actor.x, actor.y, 'portable');\r\n\t\t// TODO: Maybe get multiple things and check if they have actions?\r\n\t\tconsole.log(thing, actor.x, actor.y);\r\n\t\tif (!thing) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (thing.portable) {\r\n\t\t\tactor.queueAction('pickup', { target: thing });\r\n\t\t} else if (thing.hasAction('open')) {\r\n\t\t\tactor.queueAction('open', { target: thing });\r\n\t\t} else if (thing.hasAction('use')) {\r\n\t\t\tactor.queueAction('use', { target: thing });\r\n\t\t} else if (thing.hasAction('descend') || thing.hasAction('ascend')) {\r\n\t\t\tactor.queueAction('teleport', { teleport: thing.teleport });\r\n\t\t\tconsole.log('Planning to teleport...', actor.actionQueue);\r\n\t\t}\r\n\t}\r\n\r\n\tadvance() {\r\n\t\tconst startHp = this.hero.hp;\r\n\t\t// TODO: advance time\r\n\t\t// Do actions for all actors\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tlevel.resolveRoundEffects();\r\n\t\tconst actors = level.getActorsInitiativeOrdered();\r\n\t\tlevel.coolOffInitiativeBoosts();\r\n\t\tactors.forEach((actor) => {\r\n\t\t\tactor.planAction(level, this.hero);\r\n\t\t\tthis.advanceActor(actor);\r\n\t\t});\r\n\t\t// this.advanceActor(this.hero);\r\n\t\tconst isDamaged = (startHp > this.hero.hp);\r\n\t\tthis.display.drawDamage(isDamaged);\r\n\t\tif (this.hero.dead()) {\r\n\t\t\tthis.hook('afterHeroDeath', {});\r\n\t\t}\r\n\t\tthis.draw();\r\n\t}\r\n\r\n\tadvanceActor(actor) {\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tconst action = actor.doAction();\r\n\t\tif (!action) { return; }\r\n\t\tconst { verb, target, what, x, y } = action;\r\n\t\tif (actor.isHero) {\r\n\t\t\tif (verb === 'move') { console.log(actor.name + ' ' + verb); }\r\n\t\t\telse { console.log(actor.name, verb, action); }\r\n\t\t}\r\n\t\tlet outcome = {};\r\n\t\tlet message = '';\r\n\t\tswitch (verb) {\r\n\t\t\tcase 'move':\r\n\t\t\t\tconst bumpCombat = (actor.isHero || actor.aggro > 0);\r\n\t\t\t\tif (action.direction === undefined) {\r\n\t\t\t\t\tthis.moveActorTo(actor, action.x, action.y, bumpCombat);\r\n\t\t\t\t} else {\r\n\t\t\t\t\tthis.moveActor(actor, action.direction, bumpCombat);\r\n\t\t\t\t}\r\n\t\t\tbreak;\r\n\t\t\tcase 'use':\r\n\t\t\t\toutcome = level.useThing(actor, 'use', target);\r\n\t\t\t\tmessage = outcome.message;\r\n\t\t\tbreak;\r\n\t\t\tcase 'open':\r\n\t\t\t\toutcome = level.useThing(actor, 'open', target);\r\n\t\t\t\tmessage = outcome.message;\r\n\t\t\tbreak;\r\n\t\t\tcase 'teleport':\r\n\t\t\t\tmessage = `${actor.name} now entering: `;\r\n\t\t\t\tthis.teleportActor(actor, action.teleport);\r\n\t\t\t\t{\r\n\t\t\t\t\tconst newLevel = this.getActiveLevel();\r\n\t\t\t\t\tmessage += newLevel.name;\r\n\t\t\t\t\tif (newLevel.description) {\r\n\t\t\t\t\t\tmessage += ' - ' + newLevel.description;\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\tbreak;\r\n\t\t\tcase 'pickup':\r\n\t\t\t\tconst pickedUp = this.pickupItem(actor, target);\r\n\t\t\t\tif (pickedUp) {\r\n\t\t\t\t\tmessage = `${actor.name} picks up the ${target.name}.`;\r\n\t\t\t\t} else if (target) {\r\n\t\t\t\t\tmessage = `${actor.name} could not pick up the ${target.name}.`;\r\n\t\t\t\t} else {\r\n\t\t\t\t\tmessage = `Nothing to pick up.`;\r\n\t\t\t\t}\r\n\t\t\tbreak;\r\n\t\t\tcase 'throw':\r\n\t\t\t\tmessage = level.throw(actor, what, x, y);\r\n\t\t\tbreak;\r\n\t\t\tcase 'look':\r\n\t\t\t\tconst things = this.getActiveLevel().findEverythingInView({ excludeHero: true });\r\n\t\t\t\tconst names = things.map((thing) => thing.name || '?').join(', ');\r\n\t\t\t\tmessage = `${actor.name} looks around and sees: ${names}`;\r\n\t\t\tbreak;\r\n\t\t\tcase 'wait':\r\n\t\t\t\tactor.wait();\r\n\t\t\t\tif (actor.isHero) {\r\n\t\t\t\t\tmessage = `${actor.name} waits (random recovery of AP, BP, or EP points).`;\r\n\t\t\t\t}\r\n\t\t\tbreak;\r\n\t\t}\r\n\t\tif (typeof message !== 'string') {\r\n\t\t\tconsole.error('Unknown message from doing action', verb);\r\n\t\t\tmessage = 'ERROR';\r\n\t\t}\r\n\t\tthis.print(message);\r\n\t}\r\n\r\n\tpickupItem(actor, thing) {\r\n\t\tif (!thing) { return false; }\r\n\t\tif (!thing.portable) { return false; }\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tconst item = level.removeItem(thing);\r\n\t\tif (!item) { return false; }\r\n\t\tconst added = actor.inventory.add(thing);\r\n\t\tif (!added) {\r\n\t\t\tlevel.addItem(item);\r\n\t\t}\r\n\t\treturn added;\r\n\t}\r\n\r\n\t//---- Exploration\r\n\r\n\tdiscoverAroundHero() {\r\n\t\tconst level = this.getActiveLevel();\r\n\t\tconst illumination = this.hero.inventory.items.reduce((n, item) => {\r\n\t\t\treturn n + item.illumination;\r\n\t\t}, 0);\r\n\t\tlevel.discoverCircle(this.hero.x, this.hero.y, this.hero.sightRange + illumination); // TODO: allow different POV\r\n\t\tlevel.setEye(this.hero);\r\n\t}\r\n\r\n\tnarrateAroundHero() {\r\n\t\tconst allThingsOnHero = this.getThingsOnActor(this.hero);\r\n\t\tif (allThingsOnHero.length === 0) { return; }\r\n\t\tconst namesOnHero = allThingsOnHero.map((thing) => thing.name);\r\n\t\tconst namesString = (namesOnHero.length > 1) ? namesOnHero.join(', ') : 'a ' + namesOnHero[0];\r\n\t\tthis.console.print(`You are on ${namesString}.`);\r\n\t}\r\n\r\n\t//---- System\r\n\r\n\tready(callback, fonts = []) { // TODO: remove fonts param?\r\n\t\tconst fontFamiliesToLoad = [ ...fonts ].concat(this.fontFamilies);\r\n\t\tconsole.log(fontFamiliesToLoad);\r\n\t\tready(() => {\r\n\t\t\tif (this.loadingPromise instanceof Promise) {\r\n\t\t\t\tthis.loadingPromise\r\n\t\t\t\t\t.then(() => { callback(); });\r\n\t\t\t\t\t// .catch((err) => { console.error('Error loading something', err) });\r\n\t\t\t} else {\r\n\t\t\t\tcallback();\r\n\t\t\t}\r\n\t\t}, fontFamiliesToLoad);\r\n\t\t// TODO: return a promise so can be used async\r\n\t}\r\n\r\n\tstart() {\r\n\t\tthis.setupEngine();\r\n\t\tthis.setupKeyboard();\r\n\t\tthis.setupMusic();\r\n\t\tconst startState = (this.haveSplash) ? SPLASH_STATE : MAIN_GAME_STATE;\r\n\t\tthis.setStateDetect(startState);\r\n\t\t// TODO: start graphics loop\r\n\t\tthis.draw();\r\n\t}\r\n\r\n\tstop() {\r\n\t\tthis.setState(OFF_STATE);\r\n\t\tthis.removeKeyboard();\r\n\t\t// TODO: stop graphics loop\r\n\t}\r\n\r\n\tloadData(data) {\r\n\t\tconst promises = [];\r\n\t\tfunction parseJson(response) { return response.json(); }\r\n\t\tfunction fixInnerObject(obj, key) {\r\n\t\t\treturn (typeof obj[key] === 'object') ? obj[key] : obj;\r\n\t\t}\r\n\t\tfor (let key in data) {\r\n\t\t\tif (typeof data[key] === 'string') {\r\n\t\t\t\tconst p = fetch(data[key])\r\n\t\t\t\t\t.then(parseJson)\r\n\t\t\t\t\t.then((obj) => fixInnerObject(obj, key))\r\n\t\t\t\t\t.then((obj) => { this.setData(key, obj); });\r\n\t\t\t\t\t//.catch((err) => { console.error(data, key, err); });\r\n\t\t\t\tpromises.push(p);\r\n\t\t\t} else {\r\n\t\t\t\tthis.setData(key, data[key]);\r\n\t\t\t}\r\n\t\t}\r\n\t\tthis.loadingPromise = Promise.all(promises); // .then((resp) => { console.log(resp); });\r\n\t\treturn this.loadingPromise;\r\n\t}\r\n\r\n\t//---- Hooks\r\n\r\n\taddHook(hookName, fn) {\r\n\t\tif (!this.hooks[hookName]) {\r\n\t\t\tthis.hooks[hookName] = [];\r\n\t\t}\r\n\t\tthis.hooks[hookName].push(fn);\r\n\t}\r\n\r\n\tremoveHook(hookName, fn) {\r\n\t\tif (!this.hooks[hookName]) { return; }\r\n\t\tconst i = this.hooks[hookName].indexOf(fn);\r\n\t\tthis.hooks[hookName].splice(i, 1);\r\n\t}\r\n\r\n\thook(hookName, data = {}) {\r\n\t\tconst hook = this.hooks[hookName];\r\n\t\tif (!hook) { return; }\r\n\t\thook.forEach((fn) => {\r\n\t\t\tfn(data, this, hookName);\r\n\t\t});\r\n\t}\r\n\r\n\t//---- Gets\r\n\r\n\tgetActiveLevel() {\r\n\t\treturn this.levels[this.activeLevelIndex];\r\n\t}\r\n\r\n\tgetLevelType(key) {\r\n\t\tconst lt = this.data.levels[key];\r\n\t\tif (typeof lt !== 'object' || lt === null) {\r\n\t\t\tconsole.error('Cannot find level type ', key);\r\n\t\t}\r\n\t\treturn lt;\r\n\t}\r\n\r\n\tgetDataPropArray() {\r\n\t\tconst propKeys = Object.keys(this.data.props);\r\n\t\tconst arr = [];\r\n\t\tpropKeys.forEach((key) => {\r\n\t\t\tconst prop = { ...this.data.props[key], key };\r\n\t\t\tarr.push(prop);\r\n\t\t});\r\n\t\treturn arr;\r\n\t}\r\n\r\n\tgetThingsOnActor(actor) {\r\n\t\tconst { x, y } = actor;\r\n\t\treturn this.getActiveLevel().findThings(x, y);\r\n\t}\r\n\r\n\t//---- Sets\r\n\r\n\tsetActiveLevel(i) {\r\n\t\tthis.activeLevelIndex = i;\r\n\t\treturn;\r\n\t}\r\n\r\n\tsetData(key, obj) {\r\n\t\tthis.data[key] = Object.freeze(obj);\r\n\t}\r\n\r\n\tsetStateDetect(stateFallback) {\r\n\t\t// TODO: improve... not sure i like how this works\r\n\t\t// const hash = location.hash.substring(1).toUpperCase();\r\n\t\t// if (this.states.includes(hash)) {\r\n\t\t// \treturn this.setState(hash);\r\n\t\t// }\r\n\t\treturn this.setState(stateFallback);\r\n\t}\r\n\r\n\tsetState(state) {\r\n\t\tconst isLegitState = this.states.has(state);\r\n\t\tconst consoleMethod = (isLegitState) ? 'log' : 'warn';\r\n\t\tconsole[consoleMethod]('Setting state:', state);\r\n\t\tconst prefix = 'rote-state-';\r\n\t\tthis.state = state;\r\n\t\t// const body = document.getElementsByClassName('rote-state')[0];\r\n\t\tconst body = document.getElementsByTagName('body')[0];\r\n\t\t// body.className = 'rote-state'; // TODO: make this smarter so it only removes rote states\r\n\t\t// body.classList.add(prefix + this.state.toLowerCase());\r\n\t\tbody.className = 'rote-state ' + prefix + state.toLowerCase();\r\n\t\tlocation.hash = state.toLowerCase();\r\n\t\tthis.keyboard.setState(state);\r\n\t}\r\n\r\n\tsetMainGameState() {\r\n\t\tthis.setState(MAIN_GAME_STATE);\r\n\t}\r\n\r\n}\r\n\r\nmodule.exports = Game;\r\n","/* Font Face Observer v2.1.0 - © Bram Stein. License: BSD-3-Clause */(function(){function l(a,b){document.addEventListener?a.addEventListener(\"scroll\",b,!1):a.attachEvent(\"scroll\",b)}function m(a){document.body?a():document.addEventListener?document.addEventListener(\"DOMContentLoaded\",function c(){document.removeEventListener(\"DOMContentLoaded\",c);a()}):document.attachEvent(\"onreadystatechange\",function k(){if(\"interactive\"==document.readyState||\"complete\"==document.readyState)document.detachEvent(\"onreadystatechange\",k),a()})};function t(a){this.a=document.createElement(\"div\");this.a.setAttribute(\"aria-hidden\",\"true\");this.a.appendChild(document.createTextNode(a));this.b=document.createElement(\"span\");this.c=document.createElement(\"span\");this.h=document.createElement(\"span\");this.f=document.createElement(\"span\");this.g=-1;this.b.style.cssText=\"max-width:none;display:inline-block;position:absolute;height:100%;width:100%;overflow:scroll;font-size:16px;\";this.c.style.cssText=\"max-width:none;display:inline-block;position:absolute;height:100%;width:100%;overflow:scroll;font-size:16px;\";\nthis.f.style.cssText=\"max-width:none;display:inline-block;position:absolute;height:100%;width:100%;overflow:scroll;font-size:16px;\";this.h.style.cssText=\"display:inline-block;width:200%;height:200%;font-size:16px;max-width:none;\";this.b.appendChild(this.h);this.c.appendChild(this.f);this.a.appendChild(this.b);this.a.appendChild(this.c)}\nfunction u(a,b){a.a.style.cssText=\"max-width:none;min-width:20px;min-height:20px;display:inline-block;overflow:hidden;position:absolute;width:auto;margin:0;padding:0;top:-999px;white-space:nowrap;font-synthesis:none;font:\"+b+\";\"}function z(a){var b=a.a.offsetWidth,c=b+100;a.f.style.width=c+\"px\";a.c.scrollLeft=c;a.b.scrollLeft=a.b.scrollWidth+100;return a.g!==b?(a.g=b,!0):!1}function A(a,b){function c(){var a=k;z(a)&&a.a.parentNode&&b(a.g)}var k=a;l(a.b,c);l(a.c,c);z(a)};function B(a,b){var c=b||{};this.family=a;this.style=c.style||\"normal\";this.weight=c.weight||\"normal\";this.stretch=c.stretch||\"normal\"}var C=null,D=null,E=null,F=null;function G(){if(null===D)if(J()&&/Apple/.test(window.navigator.vendor)){var a=/AppleWebKit\\/([0-9]+)(?:\\.([0-9]+))(?:\\.([0-9]+))/.exec(window.navigator.userAgent);D=!!a&&603>parseInt(a[1],10)}else D=!1;return D}function J(){null===F&&(F=!!document.fonts);return F}\nfunction K(){if(null===E){var a=document.createElement(\"div\");try{a.style.font=\"condensed 100px sans-serif\"}catch(b){}E=\"\"!==a.style.font}return E}function L(a,b){return[a.style,a.weight,K()?a.stretch:\"\",\"100px\",b].join(\" \")}\nB.prototype.load=function(a,b){var c=this,k=a||\"BESbswy\",r=0,n=b||3E3,H=(new Date).getTime();return new Promise(function(a,b){if(J()&&!G()){var M=new Promise(function(a,b){function e(){(new Date).getTime()-H>=n?b(Error(\"\"+n+\"ms timeout exceeded\")):document.fonts.load(L(c,'\"'+c.family+'\"'),k).then(function(c){1<=c.length?a():setTimeout(e,25)},b)}e()}),N=new Promise(function(a,c){r=setTimeout(function(){c(Error(\"\"+n+\"ms timeout exceeded\"))},n)});Promise.race([N,M]).then(function(){clearTimeout(r);a(c)},\nb)}else m(function(){function v(){var b;if(b=-1!=f&&-1!=g||-1!=f&&-1!=h||-1!=g&&-1!=h)(b=f!=g&&f!=h&&g!=h)||(null===C&&(b=/AppleWebKit\\/([0-9]+)(?:\\.([0-9]+))/.exec(window.navigator.userAgent),C=!!b&&(536>parseInt(b[1],10)||536===parseInt(b[1],10)&&11>=parseInt(b[2],10))),b=C&&(f==w&&g==w&&h==w||f==x&&g==x&&h==x||f==y&&g==y&&h==y)),b=!b;b&&(d.parentNode&&d.parentNode.removeChild(d),clearTimeout(r),a(c))}function I(){if((new Date).getTime()-H>=n)d.parentNode&&d.parentNode.removeChild(d),b(Error(\"\"+\nn+\"ms timeout exceeded\"));else{var a=document.hidden;if(!0===a||void 0===a)f=e.a.offsetWidth,g=p.a.offsetWidth,h=q.a.offsetWidth,v();r=setTimeout(I,50)}}var e=new t(k),p=new t(k),q=new t(k),f=-1,g=-1,h=-1,w=-1,x=-1,y=-1,d=document.createElement(\"div\");d.dir=\"ltr\";u(e,L(c,\"sans-serif\"));u(p,L(c,\"serif\"));u(q,L(c,\"monospace\"));d.appendChild(e.a);d.appendChild(p.a);d.appendChild(q.a);document.body.appendChild(d);w=e.a.offsetWidth;x=p.a.offsetWidth;y=q.a.offsetWidth;I();A(e,function(a){f=a;v()});u(e,\nL(c,'\"'+c.family+'\",sans-serif'));A(p,function(a){g=a;v()});u(p,L(c,'\"'+c.family+'\",serif'));A(q,function(a){h=a;v()});u(q,L(c,'\"'+c.family+'\",monospace'))})})};\"object\"===typeof module?module.exports=B:(window.FontFaceObserver=B,window.FontFaceObserver.prototype.load=B.prototype.load);}());\n","class Cell {\r\n\tconstructor(options = {}) {\r\n\t\tthis.character = options.character || ' ';\r\n\t\tthis.discovered = false;\r\n\t\tthis.color = options.color || '#777';\r\n\t\tthis.background = options.background || '#222';\r\n\t\tthis.passability = false; // TODO: handle this different?\r\n\t}\r\n\r\n\t// Gets\r\n\r\n\tgetPassability() { // TODO: update this\r\n\t\treturn (this.character === '.');\r\n\t}\r\n\r\n\tgetCharacter() {\r\n\t\treturn this.character;\r\n\t}\r\n\r\n\tgetForegroundColor(inView = true) {\r\n\t\tif (!this.discovered) {\r\n\t\t\treturn '#000';\r\n\t\t}\r\n\t\treturn (inView) ? this.color : '#232120';\r\n\t}\r\n\r\n\tgetBackgroundColor(inView = true) {\r\n\t\tif (!this.discovered) {\r\n\t\t\treturn '#000';\r\n\t\t}\r\n\t\treturn (inView) ? this.background : '#111010';\r\n\t}\r\n\r\n\t// Sets\r\n\r\n\tsetCharacter(char) {\r\n\t\tthis.character = char;\r\n\t}\r\n\r\n\tdiscover() {\r\n\t\tthis.discovered = true;\r\n\t}\r\n}\r\n\r\nmodule.exports = Cell;\r\n","const DIRECTION8 = {\r\n\t'UP': 0, 'UP-RIGHT': 1,\r\n\t'RIGHT': 2, 'DOWN-RIGHT': 3,\r\n\t'DOWN': 4, 'DOWN-LEFT': 5,\r\n\t'LEFT': 6, 'UP-LEFT': 7\r\n};\r\nconst DIRECTION4 = { 'UP': 0, 'RIGHT': 1, 'DOWN': 2, 'LEFT': 3 };\r\nconst DIRECTION4_ARRAY = ['UP', 'RIGHT', 'DOWN', 'LEFT'];\r\n\r\nconst USED_KEYS = ['i', 't', 'o', 'p', '1', '2', '3', '4', '5', '6', '7', '8', '9'];\r\nconst KEY_MAP = {\r\n\t\"9\":\t\"TAB\",\r\n\t\"13\":\t\"ENTER\",\r\n\t\"27\":\t\"ESC\",\r\n\t\"32\":\t\"SPACE\",\r\n};\r\nKEY_MAP[38] = 'UP'; // up\r\nKEY_MAP[33] = 'UP-RIGHT';\r\nKEY_MAP[39] = 'RIGHT'; // right\r\nKEY_MAP[34] = 'DOWN-RIGHT';\r\nKEY_MAP[40] = 'DOWN'; // down\r\nKEY_MAP[35] = 'DOWN-LEFT';\r\nKEY_MAP[37] = 'LEFT'; // left\r\nKEY_MAP[36] = 'UP-LEFT';\r\n\r\nconst WASD_KEYMAP = {\r\n\t87: 'UP', // w\r\n\t65: 'LEFT', // a\r\n\t83: 'DOWN', // s\r\n\t68: 'RIGHT', // d\r\n};\r\nconst WASD_DIAGONAL = {\r\n\t...WASD_KEYMAP,\r\n\t81: 'UP-LEFT', // q\r\n\t69: 'UP-RIGHT', // e\r\n\t90: 'DOWN-LEFT', // z\r\n\t67: 'DOWN-RIGHT', // c\r\n};\r\nconst VI_KEYMAP = {\r\n\t72: 'LEFT', // h\r\n\t74: 'DOWN', // j\r\n\t75: 'UP', // k\r\n\t76: 'RIGHT', // l\r\n};\r\nconst VI_DIAGONAL = {\r\n\t...VI_KEYMAP,\r\n\t89: 'UP-LEFT', // y\r\n\t85: 'UP-RIGHT', // u\r\n\t66: 'DOWN-LEFT', // b\r\n\t78: 'DOWN-RIGHT', // n\r\n};\r\n\r\n\r\nconst UNSPECIFIED_STATE = 'UNSPECIFIED';\r\n\r\nclass KeyboardListener {\r\n\tconstructor(options = {}) {\r\n\t\tthis.callbacks = {};\r\n\t\tthis.isListening = false;\r\n\t\tthis.state = options.state || UNSPECIFIED_STATE;\r\n\t\tthis.autoStart = (options.autoStart === undefined) ? false : Boolean(options.autoStart);\r\n\t}\r\n\r\n\tsetState(state = UNSPECIFIED_STATE) {\r\n\t\tthis.state = state.toString();\r\n\t}\r\n\r\n\ton(state, key, callback) {\r\n\t\t// key can be a keyCode or a keyType like 'DIRECTION'\r\n\t\tthis.callbacks[state + '_' + key] = callback;\r\n\t\tif (this.autoStart) {\r\n\t\t\tthis.start();\r\n\t\t}\r\n\t}\r\n\t\r\n\toff(state, key, callback) {\r\n\t\t// TODO: remove callback\r\n\t\t// TODO: if no more callbacks then stop\r\n\t}\r\n\r\n\tgetKeyMap() {\r\n\t\tlet keyMap = { ...KEY_MAP };\r\n\t\t// TODO: variations based on options selected\r\n\t\tkeyMap = { ...keyMap, ...WASD_DIAGONAL, ...VI_DIAGONAL };\r\n\t\treturn keyMap;\r\n\t}\r\n\r\n\thandleEvent(e) {\r\n\t\tconst keyMap = this.getKeyMap();\r\n\t\tconst { keyCode, key } = e;\r\n\t\tconst isKeyUsed = USED_KEYS.includes(key) || (keyCode in keyMap);\r\n\r\n\t\tif (!isKeyUsed) {\r\n\t\t\tconsole.log('Keyboard handleEvent - unaccounted for key:', key, keyCode);\r\n\t\t\treturn;\r\n\t\t}\r\n\t\te.preventDefault();\r\n\r\n\t\t// Lookup key name and direction\r\n\t\tconst keyName = keyMap[keyCode] || key;\r\n\t\tconst direction = DIRECTION8[keyName];\r\n\t\t// console.log('handleEvent', e, keyName, keyCode, direction);\r\n\r\n\t\t// Callbacks\r\n\t\tif (direction !== undefined) {\r\n\t\t\tconst typeCallback = this.callbacks[this.state + '_DIRECTION'];\r\n\t\t\tif (typeCallback) {\r\n\t\t\t\ttypeCallback(keyName, keyCode, direction);\r\n\t\t\t}\r\n\t\t}\r\n\t\tconst callback = this.callbacks[this.state + '_' + keyName];\r\n\t\t// console.log(this.state + '_' + keyName, callback);\r\n\t\tif (callback) {\r\n\t\t\tcallback(keyName, keyCode, direction);\r\n\t\t}\r\n\t}\r\n\r\n\tstart() {\r\n\t\tif (this.isListening) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\twindow.addEventListener('keydown', this);  // pass this; the `handleEvent` will be used\r\n\t\tthis.isListening = true;\r\n\t}\r\n\r\n\tstop() {\r\n\t\t// TODO: remove event listener\r\n\t}\r\n}\r\n\r\nmodule.exports = KeyboardListener;\r\n","\r\nclass MusicBox {\r\n\tconstructor(playlist) {\r\n\t\tthis.audio = null;\r\n\t\tthis.playlist = [ ...playlist ];\r\n\t}\r\n\r\n\taddToPlaylist(songPath) {\r\n\t\tthis.playlist.push(songPath);\r\n\t}\r\n\r\n\tplay(i = 0) {\r\n\t\tthis.audio = new Audio(this.playlist[i]);\r\n\t\tthis.audio.play();\r\n\t}\r\n}\r\n\r\nmodule.exports = MusicBox;\r\n","class Console {\r\n\tconstructor(options = {}) {\r\n\t\tthis.id = options.id || 'console';\r\n\t\tthis.container = null;\r\n\t\tthis.list = null;\r\n\t\tthis.messages = [];\r\n\t\tthis.writeToConsoleLog = false;\r\n\t}\r\n\r\n\tsetup() {\r\n\t\tthis.container = document.getElementById(this.id);\r\n\t\tthis.clear();\r\n\t}\r\n\r\n\tclear() {\r\n\t\tthis.messages.length = 0;\r\n\t\tthis.container.innerHTML = '<ul></ul>';\r\n\t\tthis.list = this.container.firstChild;\r\n\t}\r\n\r\n\tprint(str, classes = '') {\r\n\t\tif (!str) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (this.writeToConsoleLog) {\r\n\t\t\tconsole.log('%c' + str, 'color: #559955');\r\n\t\t}\r\n\t\tconst safeStr = str.replace('<', '&lt;');\r\n\t\tthis.list.innerHTML += `<li class=\"${classes}\">${safeStr}</li>`;\r\n\t\tthis.container.scrollTop = this.container.scrollHeight;\r\n\t\tthis.trim();\r\n\t}\r\n\r\n\t// aliases\r\n\tlog(str) { return this.print(str);\t}\r\n\tadd(str) { return this.print(str); }\r\n\r\n\ttrim() {\r\n\t\tif (this.list.innerHTML.length > 5000) {\r\n\t\t\tthis.list.removeChild(this.list.firstChild);\r\n\t\t}\r\n\t}\r\n}\r\n\r\nmodule.exports = Console;\r\n"],"sourceRoot":""} \ No newline at end of file