From 177db92ff2165c1fc8d8e0458483566809540727 Mon Sep 17 00:00:00 2001 From: Patrick Brosset Date: Tue, 28 Nov 2023 16:17:06 +0100 Subject: [PATCH] Slow calendar in-progress demo --- slow-calendar/README.md | 24 + slow-calendar/package-lock.json | 1580 ++++ slow-calendar/package.json | 16 + slow-calendar/public/app.css | 251 + slow-calendar/public/bundle.js | 2 + slow-calendar/public/clock.png | Bin 0 -> 1097 bytes slow-calendar/public/data.json | 11446 ++++++++++++++++++++++++++ slow-calendar/public/index.html | 13 + slow-calendar/public/info.png | Bin 0 -> 1117 bytes slow-calendar/public/pin.png | Bin 0 -> 1193 bytes slow-calendar/src/AppUI.ts | 163 + slow-calendar/src/CalendarEvent.ts | 71 + slow-calendar/src/EventPopup.ts | 93 + slow-calendar/src/MonthGrid.ts | 117 + slow-calendar/src/Sidebar.ts | 41 + slow-calendar/src/Store.ts | 24 + slow-calendar/src/Toolbar.ts | 73 + slow-calendar/src/WeekGrid.ts | 93 + slow-calendar/src/app.ts | 35 + slow-calendar/src/events-factory.ts | 99 + slow-calendar/src/utils.ts | 33 + slow-calendar/tsconfig.json | 12 + slow-calendar/webpack.config.js | 22 + 23 files changed, 14208 insertions(+) create mode 100644 slow-calendar/README.md create mode 100644 slow-calendar/package-lock.json create mode 100644 slow-calendar/package.json create mode 100644 slow-calendar/public/app.css create mode 100644 slow-calendar/public/bundle.js create mode 100644 slow-calendar/public/clock.png create mode 100644 slow-calendar/public/data.json create mode 100644 slow-calendar/public/index.html create mode 100644 slow-calendar/public/info.png create mode 100644 slow-calendar/public/pin.png create mode 100644 slow-calendar/src/AppUI.ts create mode 100644 slow-calendar/src/CalendarEvent.ts create mode 100644 slow-calendar/src/EventPopup.ts create mode 100644 slow-calendar/src/MonthGrid.ts create mode 100644 slow-calendar/src/Sidebar.ts create mode 100644 slow-calendar/src/Store.ts create mode 100644 slow-calendar/src/Toolbar.ts create mode 100644 slow-calendar/src/WeekGrid.ts create mode 100644 slow-calendar/src/app.ts create mode 100644 slow-calendar/src/events-factory.ts create mode 100644 slow-calendar/src/utils.ts create mode 100644 slow-calendar/tsconfig.json create mode 100644 slow-calendar/webpack.config.js diff --git a/slow-calendar/README.md b/slow-calendar/README.md new file mode 100644 index 0000000..a4b63d6 --- /dev/null +++ b/slow-calendar/README.md @@ -0,0 +1,24 @@ +# Slow Calendar app + +**IN PROGRESS**: this demo app is still in progress and isn't ready for use yet. + +## Building + +To build the calendar app, you'll need to have [Node.js](https://nodejs.org/) installed. + +Then run: + +1. `cd slow-calendar` +1. `npm install` +1. `npm run build` + +The build app will be in the `public` directory. + +## Running + +To run the app locally, use a local web server like [http-server](https://www.npmjs.com/package/http-server): + +1. `npm install -g http-server` +1. `http-server public` + +Then open [http://localhost:8080](http://localhost:8080) in your browser. diff --git a/slow-calendar/package-lock.json b/slow-calendar/package-lock.json new file mode 100644 index 0000000..c85197b --- /dev/null +++ b/slow-calendar/package-lock.json @@ -0,0 +1,1580 @@ +{ + "name": "slow-calendar", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "slow-calendar", + "dependencies": { + "event-target-shim": "^6.0.2" + }, + "devDependencies": { + "ts-loader": "^9.5.1", + "typescript": "^5.3.2", + "webpack": "^5.89.0", + "webpack-cli": "^5.1.4" + } + }, + "node_modules/@discoveryjs/json-ext": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz", + "integrity": "sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==", + "dev": true, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz", + "integrity": "sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ==", + "dev": true, + "dependencies": { + "@jridgewell/set-array": "^1.0.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.9" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.1.tgz", + "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/set-array": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.2.tgz", + "integrity": "sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/source-map": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/source-map/-/source-map-0.3.5.tgz", + "integrity": "sha512-UTYAUj/wviwdsMfzoSJspJxbkH5o1snzwX0//0ENX1u/55kkZZkcTZP6u9bwKGkv+dkk9at4m1Cpt0uY80kcpQ==", + "dev": true, + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.0", + "@jridgewell/trace-mapping": "^0.3.9" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.4.15", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", + "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", + "dev": true + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.20", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.20.tgz", + "integrity": "sha512-R8LcPeWZol2zR8mmH3JeKQ6QRCFb7XgUhV9ZlGhHLGyg4wpPiPZNQOOWhFZhxKw8u//yTbNGI42Bx/3paXEQ+Q==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@types/eslint": { + "version": "8.44.7", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.44.7.tgz", + "integrity": "sha512-f5ORu2hcBbKei97U73mf+l9t4zTGl74IqZ0GQk4oVea/VS8tQZYkUveSYojk+frraAVYId0V2WC9O4PTNru2FQ==", + "dev": true, + "dependencies": { + "@types/estree": "*", + "@types/json-schema": "*" + } + }, + "node_modules/@types/eslint-scope": { + "version": "3.7.7", + "resolved": "https://registry.npmjs.org/@types/eslint-scope/-/eslint-scope-3.7.7.tgz", + "integrity": "sha512-MzMFlSLBqNF2gcHWO0G1vP/YQyfvrxZ0bF+u7mzUdZ1/xK4A4sru+nraZz5i3iEIk1l1uyicaDVTB4QbbEkAYg==", + "dev": true, + "dependencies": { + "@types/eslint": "*", + "@types/estree": "*" + } + }, + "node_modules/@types/estree": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", + "dev": true + }, + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true + }, + "node_modules/@types/node": { + "version": "20.10.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.10.0.tgz", + "integrity": "sha512-D0WfRmU9TQ8I9PFx9Yc+EBHw+vSpIub4IDvQivcp26PtPrdMGAq5SDcpXEo/epqa/DXotVpekHiLNTg3iaKXBQ==", + "dev": true, + "dependencies": { + "undici-types": "~5.26.4" + } + }, + "node_modules/@webassemblyjs/ast": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.11.6.tgz", + "integrity": "sha512-IN1xI7PwOvLPgjcf180gC1bqn3q/QaOCwYUahIOhbYUu8KA/3tw2RT/T0Gidi1l7Hhj5D/INhJxiICObqpMu4Q==", + "dev": true, + "dependencies": { + "@webassemblyjs/helper-numbers": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6" + } + }, + "node_modules/@webassemblyjs/floating-point-hex-parser": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.6.tgz", + "integrity": "sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-api-error": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.6.tgz", + "integrity": "sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-buffer": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.11.6.tgz", + "integrity": "sha512-z3nFzdcp1mb8nEOFFk8DrYLpHvhKC3grJD2ardfKOzmbmJvEf/tPIqCY+sNcwZIY8ZD7IkB2l7/pqhUhqm7hLA==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-numbers": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.6.tgz", + "integrity": "sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g==", + "dev": true, + "dependencies": { + "@webassemblyjs/floating-point-hex-parser": "1.11.6", + "@webassemblyjs/helper-api-error": "1.11.6", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/helper-wasm-bytecode": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.6.tgz", + "integrity": "sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-wasm-section": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.11.6.tgz", + "integrity": "sha512-LPpZbSOwTpEC2cgn4hTydySy1Ke+XEu+ETXuoyvuyezHO3Kjdu90KK95Sh9xTbmjrCsUwvWwCOQQNta37VrS9g==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-buffer": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/wasm-gen": "1.11.6" + } + }, + "node_modules/@webassemblyjs/ieee754": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.11.6.tgz", + "integrity": "sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg==", + "dev": true, + "dependencies": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "node_modules/@webassemblyjs/leb128": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.11.6.tgz", + "integrity": "sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ==", + "dev": true, + "dependencies": { + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/utf8": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.11.6.tgz", + "integrity": "sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA==", + "dev": true + }, + "node_modules/@webassemblyjs/wasm-edit": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.11.6.tgz", + "integrity": "sha512-Ybn2I6fnfIGuCR+Faaz7YcvtBKxvoLV3Lebn1tM4o/IAJzmi9AWYIPWpyBfU8cC+JxAO57bk4+zdsTjJR+VTOw==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-buffer": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/helper-wasm-section": "1.11.6", + "@webassemblyjs/wasm-gen": "1.11.6", + "@webassemblyjs/wasm-opt": "1.11.6", + "@webassemblyjs/wasm-parser": "1.11.6", + "@webassemblyjs/wast-printer": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wasm-gen": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.11.6.tgz", + "integrity": "sha512-3XOqkZP/y6B4F0PBAXvI1/bky7GryoogUtfwExeP/v7Nzwo1QLcq5oQmpKlftZLbT+ERUOAZVQjuNVak6UXjPA==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/ieee754": "1.11.6", + "@webassemblyjs/leb128": "1.11.6", + "@webassemblyjs/utf8": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wasm-opt": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.11.6.tgz", + "integrity": "sha512-cOrKuLRE7PCe6AsOVl7WasYf3wbSo4CeOk6PkrjS7g57MFfVUF9u6ysQBBODX0LdgSvQqRiGz3CXvIDKcPNy4g==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-buffer": "1.11.6", + "@webassemblyjs/wasm-gen": "1.11.6", + "@webassemblyjs/wasm-parser": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wasm-parser": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.11.6.tgz", + "integrity": "sha512-6ZwPeGzMJM3Dqp3hCsLgESxBGtT/OeCvCZ4TA1JUPYgmhAx38tTPR9JaKy0S5H3evQpO/h2uWs2j6Yc/fjkpTQ==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@webassemblyjs/helper-api-error": "1.11.6", + "@webassemblyjs/helper-wasm-bytecode": "1.11.6", + "@webassemblyjs/ieee754": "1.11.6", + "@webassemblyjs/leb128": "1.11.6", + "@webassemblyjs/utf8": "1.11.6" + } + }, + "node_modules/@webassemblyjs/wast-printer": { + "version": "1.11.6", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.11.6.tgz", + "integrity": "sha512-JM7AhRcE+yW2GWYaKeHL5vt4xqee5N2WcezptmgyhNS+ScggqcT1OtXykhAb13Sn5Yas0j2uv9tHgrjwvzAP4A==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.11.6", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webpack-cli/configtest": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/@webpack-cli/configtest/-/configtest-2.1.1.tgz", + "integrity": "sha512-wy0mglZpDSiSS0XHrVR+BAdId2+yxPSoJW8fsna3ZpYSlufjvxnP4YbKTCBZnNIcGN4r6ZPXV55X4mYExOfLmw==", + "dev": true, + "engines": { + "node": ">=14.15.0" + }, + "peerDependencies": { + "webpack": "5.x.x", + "webpack-cli": "5.x.x" + } + }, + "node_modules/@webpack-cli/info": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/@webpack-cli/info/-/info-2.0.2.tgz", + "integrity": "sha512-zLHQdI/Qs1UyT5UBdWNqsARasIA+AaF8t+4u2aS2nEpBQh2mWIVb8qAklq0eUENnC5mOItrIB4LiS9xMtph18A==", + "dev": true, + "engines": { + "node": ">=14.15.0" + }, + "peerDependencies": { + "webpack": "5.x.x", + "webpack-cli": "5.x.x" + } + }, + "node_modules/@webpack-cli/serve": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@webpack-cli/serve/-/serve-2.0.5.tgz", + "integrity": "sha512-lqaoKnRYBdo1UgDX8uF24AfGMifWK19TxPmM5FHc2vAGxrJ/qtyUyFBWoY1tISZdelsQ5fBcOusifo5o5wSJxQ==", + "dev": true, + "engines": { + "node": ">=14.15.0" + }, + "peerDependencies": { + "webpack": "5.x.x", + "webpack-cli": "5.x.x" + }, + "peerDependenciesMeta": { + "webpack-dev-server": { + "optional": true + } + } + }, + "node_modules/@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 + }, + "node_modules/@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 + }, + "node_modules/acorn": { + "version": "8.11.2", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.11.2.tgz", + "integrity": "sha512-nc0Axzp/0FILLEVsm4fNwLCwMttvhEI263QtVPQcbpfZZ3ts0hLsZGOpE6czNlid7CJ9MlyH8reXkpsf3YUY4w==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-import-assertions": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/acorn-import-assertions/-/acorn-import-assertions-1.9.0.tgz", + "integrity": "sha512-cmMwop9x+8KFhxvKrKfPYmN6/pKTYYHBqLa0DfvVZcKMJWNyWLnaqND7dx/qn66R7ewM1UX5XMaDVP5wlVTaVA==", + "dev": true, + "peerDependencies": { + "acorn": "^8" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ajv-keywords": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "dev": true, + "peerDependencies": { + "ajv": "^6.9.1" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.22.1", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.22.1.tgz", + "integrity": "sha512-FEVc202+2iuClEhZhrWy6ZiAcRLvNMyYcxZ8raemul1DYVOVdFsbqckWLdsixQZCpJlwe77Z3UTalE7jsjnKfQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "caniuse-lite": "^1.0.30001541", + "electron-to-chromium": "^1.4.535", + "node-releases": "^2.0.13", + "update-browserslist-db": "^1.0.13" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001565", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001565.tgz", + "integrity": "sha512-xrE//a3O7TP0vaJ8ikzkD2c2NgcVUvsEe2IvFTntV4Yd1Z9FVzh+gW+enX96L0psrbaFMcVcH2l90xNuGDWc8w==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ] + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/chrome-trace-event": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz", + "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==", + "dev": true, + "engines": { + "node": ">=6.0" + } + }, + "node_modules/clone-deep": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", + "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", + "dev": true, + "dependencies": { + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/colorette": { + "version": "2.0.20", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.20.tgz", + "integrity": "sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==", + "dev": true + }, + "node_modules/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 + }, + "node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.4.594", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.594.tgz", + "integrity": "sha512-xT1HVAu5xFn7bDfkjGQi9dNpMqGchUkebwf1GL7cZN32NSwwlHRPMSDJ1KN6HkS0bWUtndbSQZqvpQftKG2uFQ==", + "dev": true + }, + "node_modules/enhanced-resolve": { + "version": "5.15.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.15.0.tgz", + "integrity": "sha512-LXYT42KJ7lpIKECr2mAXIaMldcNCh/7E0KBKOu4KSfkHmP+mZmSs+8V5gBAqisWBy0OO4W5Oyys0GO1Y8KtdKg==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/envinfo": { + "version": "7.11.0", + "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.11.0.tgz", + "integrity": "sha512-G9/6xF1FPbIw0TtalAMaVPpiq2aDEuKLXM314jPVAO9r2fo2a4BLqMNkmRS7O/xPPZ+COAhGIz3ETvHEV3eUcg==", + "dev": true, + "bin": { + "envinfo": "dist/cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/es-module-lexer": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-1.4.1.tgz", + "integrity": "sha512-cXLGjP0c4T3flZJKQSuziYoq7MlT+rnvfZjfp7h+I7K9BNX54kP9nyWvdbwjQ4u1iWbOL4u96fgeZLToQlZC7w==", + "dev": true + }, + "node_modules/escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esrecurse/node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/event-target-shim": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-6.0.2.tgz", + "integrity": "sha512-8q3LsZjRezbFZ2PN+uP+Q7pnHUMmAOziU2vA2OwoFaKIXxlxl38IylhSSgUorWu/rf4er67w0ikBqjBFk/pomA==", + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "url": "https://github.com/sponsors/mysticatea" + } + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "dev": true, + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "node_modules/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 + }, + "node_modules/fastest-levenshtein": { + "version": "1.0.16", + "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz", + "integrity": "sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg==", + "dev": true, + "engines": { + "node": ">= 4.9.1" + } + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/flat": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", + "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", + "dev": true, + "bin": { + "flat": "cli.js" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/glob-to-regexp": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz", + "integrity": "sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw==", + "dev": true + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/hasown": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.0.tgz", + "integrity": "sha512-vUptKVTpIJhcczKBbgnS+RtcuYMB8+oNzPK2/Hp3hanz8JmpATdmmgLgSaadVREkDm+e2giHwY3ZRkyjSIDDFA==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/import-local": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", + "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", + "dev": true, + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/interpret": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-3.1.1.tgz", + "integrity": "sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ==", + "dev": true, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/is-core-module": { + "version": "2.13.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.1.tgz", + "integrity": "sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==", + "dev": true, + "dependencies": { + "hasown": "^2.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/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, + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true + }, + "node_modules/isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/jest-worker": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", + "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "dev": true, + "dependencies": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true + }, + "node_modules/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 + }, + "node_modules/kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/loader-runner": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-4.3.0.tgz", + "integrity": "sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg==", + "dev": true, + "engines": { + "node": ">=6.11.5" + } + }, + "node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true + }, + "node_modules/micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", + "dev": true, + "dependencies": { + "braces": "^3.0.2", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dev": true, + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true + }, + "node_modules/node-releases": { + "version": "2.0.13", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.13.tgz", + "integrity": "sha512-uYr7J37ae/ORWdZeQ1xxMJe3NtdmqMC/JZK+geofDrkLUApKRHPd18/TxtBOJ4A0/+uUIliorNrfYV6s1b02eQ==", + "dev": true + }, + "node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/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, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dev": true, + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, + "node_modules/rechoir": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/rechoir/-/rechoir-0.8.0.tgz", + "integrity": "sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ==", + "dev": true, + "dependencies": { + "resolve": "^1.20.0" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/resolve": { + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", + "dev": true, + "dependencies": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/schema-utils": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.3.0.tgz", + "integrity": "sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.8", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/semver": { + "version": "7.5.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.5.4.tgz", + "integrity": "sha512-1bCSESV6Pv+i21Hvpxp3Dx+pSD8lIPt8uVjRrxAUt/nbswYc+tK6Y2btiULjd4+fnq15PX+nqQDC7Oft7WkwcA==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/serialize-javascript": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.1.tgz", + "integrity": "sha512-owoXEFjWRllis8/M1Q+Cw5k8ZH40e3zhp/ovX+Xr/vi1qj6QesbyXXViFbpNvWvPNAD62SutwEXavefrLJWj7w==", + "dev": true, + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/shallow-clone": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", + "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.21", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", + "integrity": "sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==", + "dev": true, + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/tapable": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.1.tgz", + "integrity": "sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/terser": { + "version": "5.24.0", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.24.0.tgz", + "integrity": "sha512-ZpGR4Hy3+wBEzVEnHvstMvqpD/nABNelQn/z2r0fjVWGQsN3bpOLzQlqDxmb4CDZnXq5lpjnQ+mHQLAOpfM5iw==", + "dev": true, + "dependencies": { + "@jridgewell/source-map": "^0.3.3", + "acorn": "^8.8.2", + "commander": "^2.20.0", + "source-map-support": "~0.5.20" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/terser-webpack-plugin": { + "version": "5.3.9", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-5.3.9.tgz", + "integrity": "sha512-ZuXsqE07EcggTWQjXUj+Aot/OMcD0bMKGgF63f7UxYcu5/AJF53aIpK1YoP5xR9l6s/Hy2b+t1AM0bLNPRuhwA==", + "dev": true, + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.17", + "jest-worker": "^27.4.5", + "schema-utils": "^3.1.1", + "serialize-javascript": "^6.0.1", + "terser": "^5.16.8" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^5.1.0" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "uglify-js": { + "optional": true + } + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/ts-loader": { + "version": "9.5.1", + "resolved": "https://registry.npmjs.org/ts-loader/-/ts-loader-9.5.1.tgz", + "integrity": "sha512-rNH3sK9kGZcH9dYzC7CewQm4NtxJTjSEVRJ2DyBZR7f8/wcta+iV44UPCXc5+nzDzivKtlzV6c9P4e+oFhDLYg==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0", + "enhanced-resolve": "^5.0.0", + "micromatch": "^4.0.0", + "semver": "^7.3.4", + "source-map": "^0.7.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "typescript": "*", + "webpack": "^5.0.0" + } + }, + "node_modules/ts-loader/node_modules/source-map": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.4.tgz", + "integrity": "sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/typescript": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.3.2.tgz", + "integrity": "sha512-6l+RyNy7oAHDfxC4FzSJcz9vnjTKxrLpDG5M2Vu4SHRVNg6xzqZp6LYSR9zjqQTu8DU/f5xwxUdADOkbrIX2gQ==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/undici-types": { + "version": "5.26.5", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.26.5.tgz", + "integrity": "sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==", + "dev": true + }, + "node_modules/update-browserslist-db": { + "version": "1.0.13", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.13.tgz", + "integrity": "sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "escalade": "^3.1.1", + "picocolors": "^1.0.0" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/watchpack": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-2.4.0.tgz", + "integrity": "sha512-Lcvm7MGST/4fup+ifyKi2hjyIAwcdI4HRgtvTpIUxBRhB+RFtUh8XtDOxUfctVCnhVi+QQj49i91OyvzkJl6cg==", + "dev": true, + "dependencies": { + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.1.2" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/webpack": { + "version": "5.89.0", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-5.89.0.tgz", + "integrity": "sha512-qyfIC10pOr70V+jkmud8tMfajraGCZMBWJtrmuBymQKCrLTRejBI8STDp1MCyZu/QTdZSeacCQYpYNQVOzX5kw==", + "dev": true, + "dependencies": { + "@types/eslint-scope": "^3.7.3", + "@types/estree": "^1.0.0", + "@webassemblyjs/ast": "^1.11.5", + "@webassemblyjs/wasm-edit": "^1.11.5", + "@webassemblyjs/wasm-parser": "^1.11.5", + "acorn": "^8.7.1", + "acorn-import-assertions": "^1.9.0", + "browserslist": "^4.14.5", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^5.15.0", + "es-module-lexer": "^1.2.1", + "eslint-scope": "5.1.1", + "events": "^3.2.0", + "glob-to-regexp": "^0.4.1", + "graceful-fs": "^4.2.9", + "json-parse-even-better-errors": "^2.3.1", + "loader-runner": "^4.2.0", + "mime-types": "^2.1.27", + "neo-async": "^2.6.2", + "schema-utils": "^3.2.0", + "tapable": "^2.1.1", + "terser-webpack-plugin": "^5.3.7", + "watchpack": "^2.4.0", + "webpack-sources": "^3.2.3" + }, + "bin": { + "webpack": "bin/webpack.js" + }, + "engines": { + "node": ">=10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/webpack-cli": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/webpack-cli/-/webpack-cli-5.1.4.tgz", + "integrity": "sha512-pIDJHIEI9LR0yxHXQ+Qh95k2EvXpWzZ5l+d+jIo+RdSm9MiHfzazIxwwni/p7+x4eJZuvG1AJwgC4TNQ7NRgsg==", + "dev": true, + "dependencies": { + "@discoveryjs/json-ext": "^0.5.0", + "@webpack-cli/configtest": "^2.1.1", + "@webpack-cli/info": "^2.0.2", + "@webpack-cli/serve": "^2.0.5", + "colorette": "^2.0.14", + "commander": "^10.0.1", + "cross-spawn": "^7.0.3", + "envinfo": "^7.7.3", + "fastest-levenshtein": "^1.0.12", + "import-local": "^3.0.2", + "interpret": "^3.1.1", + "rechoir": "^0.8.0", + "webpack-merge": "^5.7.3" + }, + "bin": { + "webpack-cli": "bin/cli.js" + }, + "engines": { + "node": ">=14.15.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "5.x.x" + }, + "peerDependenciesMeta": { + "@webpack-cli/generators": { + "optional": true + }, + "webpack-bundle-analyzer": { + "optional": true + }, + "webpack-dev-server": { + "optional": true + } + } + }, + "node_modules/webpack-cli/node_modules/commander": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.1.tgz", + "integrity": "sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==", + "dev": true, + "engines": { + "node": ">=14" + } + }, + "node_modules/webpack-merge": { + "version": "5.10.0", + "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-5.10.0.tgz", + "integrity": "sha512-+4zXKdx7UnO+1jaN4l2lHVD+mFvnlZQP/6ljaJVb4SZiwIKeUnrT5l0gkT8z+n4hKpC+jpOv6O9R+gLtag7pSA==", + "dev": true, + "dependencies": { + "clone-deep": "^4.0.1", + "flat": "^5.0.2", + "wildcard": "^2.0.0" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/webpack-sources": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-3.2.3.tgz", + "integrity": "sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==", + "dev": true, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wildcard": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.1.tgz", + "integrity": "sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ==", + "dev": true + }, + "node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + } + } +} diff --git a/slow-calendar/package.json b/slow-calendar/package.json new file mode 100644 index 0000000..6549b18 --- /dev/null +++ b/slow-calendar/package.json @@ -0,0 +1,16 @@ +{ + "name": "slow-calendar", + "devDependencies": { + "ts-loader": "^9.5.1", + "typescript": "^5.3.2", + "webpack": "^5.89.0", + "webpack-cli": "^5.1.4" + }, + "scripts": { + "build": "webpack --mode=production", + "start": "webpack serve --mode=development" + }, + "dependencies": { + "event-target-shim": "^6.0.2" + } +} diff --git a/slow-calendar/public/app.css b/slow-calendar/public/app.css new file mode 100644 index 0000000..84314be --- /dev/null +++ b/slow-calendar/public/app.css @@ -0,0 +1,251 @@ +:root { + --border-color: #bbb; + --background-color: aliceblue; + --accent-color: #f06; + --today-color: #feffed; +} + +html { + font-family: system-ui; + font-size: .8rem; +} + +html, body { + margin: 0; + padding: 0; + overflow: hidden; + height: 100%; +} + +button { + border: 1px solid var(--border-color); + background: #eee; + font-size: inherit; + font-family: inherit; + border-radius: .25rem; + padding: .5rem 1rem; +} + +button.primary { + background: var(--accent-color); + border-color: #0004; +} + +button:active { + filter: brightness(.9); +} + +#app { + height: calc(100% - 2rem); + padding: 1rem; + display: grid; + gap: 1rem; + grid-template-rows: max-content 1fr; + grid-template-columns: 1fr 200px; + background-color: var(--background-color); +} + +#toolbar { + padding: 1rem; + background: white; + border: 1px solid var(--border-color); + display: flex; + align-items: center; + gap: .5rem; + border-radius: .5rem; + grid-column: span 2; +} + +#toolbar #month-year { + margin: 0 auto; + font-weight: bold; +} + +#month-grid { + display: grid; + grid-template-columns: repeat(7, 1fr); + grid-auto-rows: 1fr; + gap: 1px; + border: 1px solid var(--border-color); + background: var(--border-color); + border-radius: .5rem; + overflow: hidden; +} + +#week-grid { + display: none; + grid-template-columns: repeat(7, 1fr); + grid-template-rows: 1fr; + gap: 1px; + border: 1px solid var(--border-color); + background: var(--border-color); + border-radius: .5rem; + overflow: hidden; +} + +#sidebar { + padding: 1rem; + background: white; + border: 1px solid var(--border-color); + border-radius: .5rem; + overflow-y: auto; +} + +#sidebar h2 { + margin: 0; + text-align: center; +} + +.day { + padding: .5rem; + overflow: hidden; + background-color: white; +} + +.day:hover { + background: #eee; +} + +.day.prev-month, .day.next-month { + opacity: .4; +} + +.day.today { + background: var(--today-color); + box-shadow: inset 0 0 0 1px #aaa; + background-image: linear-gradient(to bottom, + transparent 0 var(--now-marker), + red var(--now-marker) calc(var(--now-marker) + 2px), + transparent 0 + ); +} + +.events, .event { + margin: 0; + padding: 0; + list-style: none; +} + +.event { + background-color: var(--event-color); + border: 1px solid color-mix(in srgb, var(--event-color) 50%, black 50%); +} + +.day .header { + font-size: .8rem; + white-space: nowrap; +} + +.day.today .header { + font-weight: bold; +} + +.event { + margin-block-start: .25rem; + padding: .25rem; + border-radius: .25rem; + overflow: hidden; + cursor: default; +} + +.event * { + pointer-events: none; +} + +.event .time { + font-weight: bold; + font-size: .9em; +} + +.event.multi-day { + box-shadow: -1rem 0 0 0 var(--event-color), 1rem 0 0 0 var(--event-color); + border: 0; +} + +.event.unconfirmed { + border-style: dashed; +} + +.event:hover { + box-shadow: inset 0 0 0 2px #0005; +} + +.popup { + display: none; + position: absolute; + width: calc(100vw / 7 - 1rem - 3px); + background: white; + padding: .5rem; + border-radius: .25rem; + border: 1px solid #aaa; + margin: 0 .5rem; + background: var(--today-color); + box-shadow: 0 0 1rem 0 #0004; +} + +.popup.left { + translate: calc(-1.5rem - 2px); + margin: 0; +} + +.popup.visible { + display: block; +} + +.popup::before { + content: ""; + position: absolute; + width: 1rem; + height: 1rem; + background: var(--event-color); + border-radius: 50%; + top: 1rem; + right: 1rem; + border: 1px solid color-mix(in srgb, var(--event-color) 50%, black 50%); +} + +.popup.unconfirmed::before { + border-style: dashed; +} + +.popup h2 { + margin: 0; + padding-inline-end: 2rem; +} + +.popup p { + margin: 1rem 0 0 0; +} + +.when::before { + content: 'When: '; + font-weight: bold; + padding-inline-start: 1.25rem; + background: url(clock.png); + background-repeat: no-repeat; + background-size: 1rem; + background-position: 0 .25rem; + background-blend-mode: darken; +} + +.where::before { + content: 'Where: '; + font-weight: bold; + padding-inline-start: 1.25rem; + background: url(pin.png); + background-repeat: no-repeat; + background-size: 1rem; + background-position: 0 .2rem; + background-blend-mode: darken; +} + +.what::before { + content: 'What: '; + font-weight: bold; + padding-inline-start: 1.25rem; + background: url(info.png); + background-repeat: no-repeat; + background-size: 1rem; + background-position: 0 .2rem; + background-blend-mode: darken; +} \ No newline at end of file diff --git a/slow-calendar/public/bundle.js b/slow-calendar/public/bundle.js new file mode 100644 index 0000000..c2f4329 --- /dev/null +++ b/slow-calendar/public/bundle.js @@ -0,0 +1,2 @@ +(()=>{"use strict";function e(e,n,...s){if(!e)throw new TypeError(t(n,s))}function t(e,t){let s=0;return e.replace(/%[os]/gu,(()=>n(t[s++])))}function n(e){return"object"!=typeof e||null===e?String(e):Object.prototype.toString.call(e)}const s="undefined"!=typeof window?window:"undefined"!=typeof self?self:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:void 0;let i;class o{constructor(e,t){this.code=e,this.message=t}warn(...e){var t;try{i;const n=(null!==(t=(new Error).stack)&&void 0!==t?t:"").replace(/^(?:.+?\n){2}/gu,"\n");console.warn(this.message,...e,n)}catch(e){}}}const a=new o("W01","Unable to initialize event under dispatching."),r=new o("W02","Assigning any falsy value to 'cancelBubble' property has no effect."),l=new o("W03","Assigning any truthy value to 'returnValue' property has no effect."),c=new o("W04","Unable to preventDefault on non-cancelable events."),d=new o("W05","Unable to preventDefault inside passive event listener invocation."),h=new o("W06","An event listener wasn't added because it has been added already: %o, %o"),u=new o("W07","The %o option value was abandoned because the event listener wasn't added as duplicated."),p=new o("W08","The 'callback' argument must be a function or an object that has 'handleEvent' method: %o");new o("W09","Event attribute handler must be a function: %o");class g{static get NONE(){return v}static get CAPTURING_PHASE(){return m}static get AT_TARGET(){return E}static get BUBBLING_PHASE(){return y}constructor(e,t){Object.defineProperty(this,"isTrusted",{value:!1,enumerable:!0});const n=null!=t?t:{};f.set(this,{type:String(e),bubbles:Boolean(n.bubbles),cancelable:Boolean(n.cancelable),composed:Boolean(n.composed),target:null,currentTarget:null,stopPropagationFlag:!1,stopImmediatePropagationFlag:!1,canceledFlag:!1,inPassiveListenerFlag:!1,dispatchFlag:!1,timeStamp:Date.now()})}get type(){return b(this).type}get target(){return b(this).target}get srcElement(){return b(this).target}get currentTarget(){return b(this).currentTarget}composedPath(){const e=b(this).currentTarget;return e?[e]:[]}get NONE(){return v}get CAPTURING_PHASE(){return m}get AT_TARGET(){return E}get BUBBLING_PHASE(){return y}get eventPhase(){return b(this).dispatchFlag?2:0}stopPropagation(){b(this).stopPropagationFlag=!0}get cancelBubble(){return b(this).stopPropagationFlag}set cancelBubble(e){e?b(this).stopPropagationFlag=!0:r.warn()}stopImmediatePropagation(){const e=b(this);e.stopPropagationFlag=e.stopImmediatePropagationFlag=!0}get bubbles(){return b(this).bubbles}get cancelable(){return b(this).cancelable}get returnValue(){return!b(this).canceledFlag}set returnValue(e){e?l.warn():w(b(this))}preventDefault(){w(b(this))}get defaultPrevented(){return b(this).canceledFlag}get composed(){return b(this).composed}get isTrusted(){return!1}get timeStamp(){return b(this).timeStamp}initEvent(e,t=!1,n=!1){const s=b(this);s.dispatchFlag?a.warn():f.set(this,{...s,type:String(e),bubbles:Boolean(t),cancelable:Boolean(n),target:null,currentTarget:null,stopPropagationFlag:!1,stopImmediatePropagationFlag:!1,canceledFlag:!1})}}const v=0,m=1,E=2,y=3,f=new WeakMap;function b(t,n="this"){const s=f.get(t);return e(null!=s,"'%s' must be an object that Event constructor created, but got another one: %o",n,t),s}function w(e){e.inPassiveListenerFlag?d.warn():e.cancelable?e.canceledFlag=!0:c.warn()}Object.defineProperty(g,"NONE",{enumerable:!0}),Object.defineProperty(g,"CAPTURING_PHASE",{enumerable:!0}),Object.defineProperty(g,"AT_TARGET",{enumerable:!0}),Object.defineProperty(g,"BUBBLING_PHASE",{enumerable:!0});const D=Object.getOwnPropertyNames(g.prototype);for(let e=0;ei,configurable:!0,enumerable:!0})}}class P extends g{static wrap(e){return new(L(e))(e)}constructor(e){super(e.type,{bubbles:e.bubbles,cancelable:e.cancelable,composed:e.composed}),e.cancelBubble&&super.stopPropagation(),e.defaultPrevented&&super.preventDefault(),M.set(this,{original:e});const t=Object.keys(e);for(let n=0;nn!==t)),!1):(e.listeners.splice(t,1),!0)}R.set(Object.prototype,P),void 0!==s&&void 0!==s.Event&&R.set(s.Event.prototype,P);class G{constructor(){B.set(this,Object.create(null))}addEventListener(e,t,n){const s=H(this),{callback:i,capture:o,once:a,passive:r,signal:l,type:c}=function(e,t,n){var s;return W(t),"object"==typeof n&&null!==n?{type:String(e),callback:null!=t?t:void 0,capture:Boolean(n.capture),passive:Boolean(n.passive),once:Boolean(n.once),signal:null!==(s=n.signal)&&void 0!==s?s:void 0}:{type:String(e),callback:null!=t?t:void 0,capture:Boolean(n),passive:!1,once:!1,signal:void 0}}(e,t,n);if(null==i||(null==l?void 0:l.aborted))return;const d=function(e,t){var n;return null!==(n=e[t])&&void 0!==n?n:e[t]={attrCallback:void 0,attrListener:void 0,cow:!1,listeners:[]}}(s,c),p=j(d,i,o);-1===p?function(e,t,n,s,i,o){let a;o&&(a=x.bind(null,e,t,n),o.addEventListener("abort",a));const r=function(e,t,n,s,i,o){return{callback:e,flags:(t?1:0)|(n?2:0)|(s?4:0),signal:i,signalListener:o}}(t,n,s,i,o,a);e.cow?(e.cow=!1,e.listeners=[...e.listeners,r]):e.listeners.push(r)}(d,i,o,r,a,l):function(e,t,n,s){h.warn($(e)?"capture":"bubble",e.callback),I(e)!==t&&u.warn("passive"),F(e)!==n&&u.warn("once"),e.signal!==s&&u.warn("signal")}(d.listeners[p],r,a,l)}removeEventListener(e,t,n){const s=H(this),{callback:i,capture:o,type:a}=function(e,t,n){return W(t),"object"==typeof n&&null!==n?{type:String(e),callback:null!=t?t:void 0,capture:Boolean(n.capture)}:{type:String(e),callback:null!=t?t:void 0,capture:Boolean(n)}}(e,t,n),r=s[a];null!=i&&r&&x(r,i,o)}dispatchEvent(e){const t=H(this)[String(e.type)];if(null==t)return!0;const n=e instanceof g?e:P.wrap(e),i=b(n,"event");if(i.dispatchFlag)throw o="This event has been in dispatching.",s.DOMException?new s.DOMException(o,"InvalidStateError"):(null==_&&(_=class e extends Error{constructor(t){super(t),Error.captureStackTrace&&Error.captureStackTrace(this,e)}get code(){return 11}get name(){return"InvalidStateError"}},Object.defineProperties(_.prototype,{code:{enumerable:!0},name:{enumerable:!0}}),k(_),k(_.prototype)),new _(o));var o;if(i.dispatchFlag=!0,i.target=i.currentTarget=this,!i.stopPropagationFlag){const{cow:e,listeners:s}=t;t.cow=!0;for(let o=0;oe.date.getFullYear()===t.getFullYear()&&e.date.getMonth()===t.getMonth()&&e.date.getDate()===t.getDate())).sort(((e,t)=>!e.time&&t.time?-1:e.time&&!t.time?1:e.time||t.time?parseInt(e.time.replace(/^0/,"").replace(":",""))-parseInt(t.time.replace(/^0/,"").replace(":","")):0)).sort(((e,t)=>e.multiDays&&!t.multiDays?-1:!e.multiDays&&t.multiDays?1:void 0))}void 0!==s&&void 0!==s.EventTarget&&Object.setPrototypeOf(G.prototype,s.EventTarget.prototype);const V=["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"];class q extends G{constructor(e,t,n){super(),this.el=e,this._date=new Date,this.rootEl=e,this._date=t,this._events=n,this.render()}render(){this.rootEl.innerHTML="";const e=this._date.getMonth(),t=this._date.getFullYear(),n=new Date(t,e,1),s=new Date(t,e+1,0),i=n.getDay(),o=s.getDay(),a=s.getDate(),r=new Date(t,e,0).getDate(),l=[];let c=0;for(let n=r-i+1;n<=r;n++){const s=new Date(t,e-1,n);l.push({date:s,html:this.renderDay(s,n,c,"prev-month")}),c++}for(let n=1;n<=a;n++){const s=new Date(t,e,n);l.push({date:s,html:this.renderDay(s,n,c)}),c++}for(let n=1;n<=7-o-1;n++){const s=new Date(t,e+1,n);l.push({date:s,html:this.renderDay(s,n,c,"next-month")}),c++}for(const{date:e,html:t}of l)this.rootEl.insertAdjacentHTML("beforeend",t),this.rootEl.lastElementChild.addEventListener("click",(t=>{this.dispatchEvent(new CustomEvent("day-clicked",{detail:e}))}));this.updateNowMarker()}renderDay(e,t,n,s=""){const i=U(this._events,e);return`\n
\n
\n ${V[e.getDay()]}\n ${t}\n
\n
    \n ${i.map((e=>e.asOneLineHTML(n))).join("")}\n
\n
\n `}updateNowMarker(){const e=new Date,t=100*(100*e.getHours()+e.getMinutes())/2400,n=this.rootEl.querySelector(".day.today");if(n){const e=n.offsetHeight*t/100;this.rootEl.style.setProperty("--now-marker",`${e}px`)}setTimeout((()=>this.updateNowMarker()),1e3)}set date(e){this._date=e,this.render()}set events(e){this._events=e,this.render()}}const J=["January","February","March","April","May","June","July","August","September","October","November","December"];class X extends G{constructor(e,t,n){super(),this.el=e,this.selectedMode="month",this.rootEl=e,this._date=t,this.selectedMode=n,this.render()}formatMonthYear(){const e=this._date.getMonth(),t=this._date.getFullYear();return`${J[e]} ${t}`}render(){this.rootEl.innerHTML=`\n \n \n \n ${this.formatMonthYear()}\n \n \n \n \n `,this.rootEl.querySelector("#prev-month").addEventListener("click",(()=>{console.log(`Prev ${this.selectedMode} clicked`),this.dispatchEvent(new g("prev"))})),this.rootEl.querySelector("#next-month").addEventListener("click",(()=>{console.log(`Next ${this.selectedMode} clicked`),this.dispatchEvent(new g("next"))})),this.rootEl.querySelector("#today").addEventListener("click",(()=>{console.log("Today clicked"),this.dispatchEvent(new g("today"))})),this.rootEl.querySelector("#month-view").addEventListener("click",(()=>{console.log("Month view clicked"),this.dispatchEvent(new g("month-view")),this.selectedMode="month"})),this.rootEl.querySelector("#week-view").addEventListener("click",(()=>{console.log("Week view clicked"),this.dispatchEvent(new g("week-view")),this.selectedMode="week"}))}set date(e){this._date=e,this.render()}}class z{constructor(e,t,n){this.el=e,this._events=[],this._date=new Date,this.rootEl=e,this._date=t,this._events=n,this.render()}render(){if(this.rootEl.innerHTML="\n

Click a day to view events

\n ",!this._events.length)return;this.rootEl.innerHTML="";const e=document.createElement("ul");e.className="events",this.rootEl.appendChild(e);for(const t of this._events)e.innerHTML+=t.asFullHTML()}set events(e){this._events=e,this.render()}}const Q=["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"];class Z extends G{constructor(e,t,n){super(),this.el=e,this._date=new Date,this.rootEl=e,this._date=t,this._events=n,this.render()}render(){this.rootEl.innerHTML="";const e=this._date.getMonth(),t=this._date.getFullYear(),n=this._date.getDate(),s=this._date.getDay(),i=new Date(t,e,n-s);for(let n=0;n<7;n++){const s=new Date(t,e,i.getDate()+n);this.rootEl.insertAdjacentHTML("beforeend",this.renderDay(s,s.getDate())),this.rootEl.lastElementChild.addEventListener("click",(e=>{this.dispatchEvent(new CustomEvent("day-clicked",{detail:s}))}))}this.updateNowMarker()}renderDay(e,t,n=""){const s=U(this._events,e);return`\n
\n
\n ${Q[e.getDay()]}\n ${t}\n
\n
    \n ${s.map((e=>e.asMediumLengthHTML())).join("")}\n
\n
\n `}updateNowMarker(){const e=new Date,t=100*(100*e.getHours()+e.getMinutes())/2400,n=this.rootEl.querySelector(".day.today");if(n){const e=n.offsetHeight*t/100;this.rootEl.style.setProperty("--now-marker",`${e}px`)}setTimeout((()=>this.updateNowMarker()),1e3)}set date(e){this._date=e,this.render()}set events(e){this._events=e,this.render()}}class K extends G{constructor(e,t,n,s){super(),this.mode="month",this.rootEl=e,this._date=t,this._events=s,this.mode=n,this.render()}render(){this.rootEl.innerHTML="",this.toolbarEl=document.createElement("div"),this.toolbarEl.id="toolbar",this.rootEl.appendChild(this.toolbarEl),this.toolbar=new X(this.toolbarEl,this._date,this.mode),this.toolbar.addEventListener("prev",(()=>{const e="month"===this.mode?new Date(this._date.getFullYear(),this._date.getMonth()-1,1):new Date(this._date.getFullYear(),this._date.getMonth(),this._date.getDate()-7);this.date=e,this.dispatchEvent(new CustomEvent("date-changed",{detail:this._date}))})),this.toolbar.addEventListener("next",(()=>{const e="month"===this.mode?new Date(this._date.getFullYear(),this._date.getMonth()+1,1):new Date(this._date.getFullYear(),this._date.getMonth(),this._date.getDate()+7);this.date=e,this.dispatchEvent(new CustomEvent("date-changed",{detail:this._date}))})),this.toolbar.addEventListener("today",(()=>{this.date=new Date,this.dispatchEvent(new CustomEvent("date-changed",{detail:this._date}))})),this.toolbar.addEventListener("month-view",(()=>{this.mode="month",this.monthGridEl.style.display="grid",this.weekGridEl.style.display="none",this.dispatchEvent(new CustomEvent("mode-changed",{detail:this.mode}))})),this.toolbar.addEventListener("week-view",(()=>{this.mode="week",this.monthGridEl.style.display="none",this.weekGridEl.style.display="grid",this.dispatchEvent(new CustomEvent("mode-changed",{detail:this.mode}))})),this.monthGridEl=document.createElement("div"),this.monthGridEl.id="month-grid",this.rootEl.appendChild(this.monthGridEl),this.monthGrid=new q(this.monthGridEl,this._date,this._events),this.weekGridEl=document.createElement("div"),this.weekGridEl.id="week-grid",this.rootEl.appendChild(this.weekGridEl),this.weekGrid=new Z(this.weekGridEl,this._date,this._events),this.sidebarEl=document.createElement("div"),this.sidebarEl.id="sidebar",this.rootEl.appendChild(this.sidebarEl),this.sidebar=new z(this.sidebarEl,this._date,this._events),this.monthGrid.addEventListener("day-clicked",(e=>{const t=e.detail,n=U(this._events,t);this.sidebar.events=n})),this.weekGrid.addEventListener("day-clicked",(e=>{const t=e.detail,n=U(this._events,t);this.sidebar.events=n})),this.sidebar.events=U(this._events,new Date),"week"===this.mode?(this.monthGridEl.style.display="none",this.weekGridEl.style.display="grid"):(this.monthGridEl.style.display="grid",this.weekGridEl.style.display="none")}set date(e){console.log("Setting date to",e),this._date=e,this.monthGrid.date=e,this.weekGrid.date=e,this.toolbar.date=e,this.sidebar.events=U(this._events,this._date)}set events(e){this._events=e,this.monthGrid.events=e,this.weekGrid.events=e,this.sidebar.events=U(this._events,this._date)}}class ee{constructor(e,t,n){this.rsvp=!1,this.title=e,this.id=t,this.date=n}asOneLineHTML(e=void 0){const t=this.time?`${this.time}`:"";return`\n
  • \n
    ${t} ${this.title}
    \n
  • \n `}asMediumLengthHTML(){const e=this.time?`${this.time}`:"";return`\n
  • \n
    ${e}
    \n

    ${this.title}

    \n

    ${this.description}

    \n
  • \n `}asFullHTML(){let e="";return this.multiDays?e=`From ${this.multiDays.start.toDateString()} to ${this.multiDays.end.toDateString()}`:(e=this.date.toDateString(),this.time&&(e+=`, at ${this.time}`),this.repeat&&(e+=`, ${this.repeat}`),this.duration&&(e+=` (${this.duration} minutes)`)),`\n
  • \n
    \n

    ${this.title}

    \n

    ${e}

    \n ${this.location?`

    ${this.location}

    `:""}\n

    ${this.description}

    \n
    \n
  • \n `}}var te=function(e,t,n,s){return new(n||(n=Promise))((function(i,o){function a(e){try{l(s.next(e))}catch(e){o(e)}}function r(e){try{l(s.throw(e))}catch(e){o(e)}}function l(e){var t;e.done?i(e.value):(t=e.value,t instanceof n?t:new n((function(e){e(t)}))).then(a,r)}l((s=s.apply(e,t||[])).next())}))};class ne{constructor(e){this.events=e}start(){this.popupEl||(this.popupEl=document.createElement("div"),this.popupEl.className="popup"),addEventListener("mousemove",(e=>{this.popupEl.classList.remove("visible"),this.popupEl.remove();const t=e.target.closest("#month-grid .event, #week-grid .event");if(!t)return;const n=t.id;if(!n||!n.startsWith("id-"))return;let s=n.replace("id-","");const i=this.events.find((e=>e.id===s));if(!i)return void console.log("No event found for id",s);const o=parseInt(t.dataset.indexInGrid);this.popupEl.style.setProperty("--event-color",i.color),this.popupEl.classList.toggle("visible",!0),this.popupEl.classList.toggle("unconfirmed",!i.rsvp);let a="";i.multiDays?a=`From ${i.multiDays.start.toDateString()} to ${i.multiDays.end.toDateString()}`:(a=i.date.toDateString(),i.time&&(a+=`, at ${i.time}`),i.repeat&&(a+=`, ${i.repeat}`),i.duration&&(a+=` (${i.duration} minutes)`)),this.popupEl.innerHTML=`\n

    ${i.title}

    \n

    ${a}

    \n ${i.location?`

    ${i.location}

    `:""}\n

    ${i.description}

    \n `,document.body.appendChild(this.popupEl);const r=t.getBoundingClientRect(),l=r.top,c=r.right,d=r.left,h=r.width;this.popupEl.style.top=o>=28?l-this.popupEl.clientHeight+"px":`${l}px`,6===o||13===o||20===o||27===o||34===o?(this.popupEl.style.left=d-h+"px",this.popupEl.classList.toggle("left",!0)):(this.popupEl.style.left=`${c}px`,this.popupEl.classList.toggle("left",!1))}))}}class se{getStoredPrefs(){return e=this,t=void 0,s=function*(){yield new Promise((e=>setTimeout(e,250*Math.random())));const e=localStorage.getItem("slow-cal-mode"),t=localStorage.getItem("slow-cal-initDate");return{mode:e,initDate:t?new Date(t):void 0}},new((n=void 0)||(n=Promise))((function(i,o){function a(e){try{l(s.next(e))}catch(e){o(e)}}function r(e){try{l(s.throw(e))}catch(e){o(e)}}function l(e){var t;e.done?i(e.value):(t=e.value,t instanceof n?t:new n((function(e){e(t)}))).then(a,r)}l((s=s.apply(e,t||[])).next())}));var e,t,n,s}set mode(e){console.log("Storing mode prefs"),localStorage.setItem("slow-cal-mode",e)}set initDate(e){console.log("Storing date prefs"),localStorage.setItem("slow-cal-initDate",e.toISOString())}}addEventListener("DOMContentLoaded",(()=>{return e=void 0,t=void 0,s=function*(){const e=new se,t=yield e.getStoredPrefs(),n=t.initDate||new Date,s=t.mode||"month",i=document.getElementById("app"),o=new K(i,n,s,[]),a=yield function(){return te(this,void 0,void 0,(function*(){const e=yield function(){return te(this,void 0,void 0,(function*(){console.log("Fetching calendar data ..."),yield new Promise((e=>setTimeout(e,500*Math.random())));const e=yield fetch("./data.json");return yield e.json()}))}();return console.log("Processing events ..."),e.events.map((e=>function(e){const t=[new Date(e.startDate)];let n=null;if(e.repeat){const n=new Date(e.startDate),s=e.endDate?new Date(e.endDate):new Date(n.getTime()+15768e7);if("daily"===e.repeat){console.log("Expanding daily event ...");const e=(s.getTime()-n.getTime())/864e5;for(let s=1;s<=e;s++){const e=new Date(n.getTime()+24*s*60*60*1e3);0!==e.getDay()&&6!==e.getDay()&&t.push(e)}}else if("weekly"===e.repeat){console.log("Expanding weekly event ...");const e=(s.getTime()-n.getTime())/6048e5;for(let s=1;s<=e;s++){const e=new Date(n.getTime()+7*s*24*60*60*1e3);t.push(e)}}else if("monthly"===e.repeat){console.log("Expanding monthly event ...");const e=12*(s.getFullYear()-n.getFullYear())+(s.getMonth()-n.getMonth());for(let s=1;s<=e;s++){const e=new Date(n.getTime());e.setMonth(e.getMonth()+s),t.push(e)}}else if("yearly"===e.repeat){console.log("Expanding yearly event ...");const e=s.getFullYear()-n.getFullYear();for(let s=1;s<=e;s++){const e=new Date(n.getTime());e.setFullYear(e.getFullYear()+s),t.push(e)}}}else if(e.endDate){const s=new Date(e.startDate),i=new Date(e.endDate),o=(i.getTime()-s.getTime())/864e5;for(let e=1;e<=o;e++){const n=new Date(s.getTime()+24*e*60*60*1e3);t.push(n)}n={start:s,end:i}}return t.map(((t,s)=>{const i=e.id+(s>0?`-${s}`:""),o=new ee(e.title,i,t);return o.multiDays=n,o.color=e.color,o.time=e.startTime,o.duration=e.duration,o.repeat=e.repeat,o.location=e.location,o.description=e.description,o.rsvp=e.rsvp,o}))}(e))).flat()}))}();console.log("Refreshing calendar with events"),o.events=a,console.log("Initializing the popup util"),new ne(a).start(),o.addEventListener("mode-changed",(t=>{e.mode=t.detail})),o.addEventListener("date-changed",(t=>{e.initDate=t.detail}))},new((n=void 0)||(n=Promise))((function(i,o){function a(e){try{l(s.next(e))}catch(e){o(e)}}function r(e){try{l(s.throw(e))}catch(e){o(e)}}function l(e){var t;e.done?i(e.value):(t=e.value,t instanceof n?t:new n((function(e){e(t)}))).then(a,r)}l((s=s.apply(e,t||[])).next())}));var e,t,n,s}))})(); +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"bundle.js","mappings":"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","sources":["webpack://slow-calendar/./node_modules/event-target-shim/index.mjs","webpack://slow-calendar/./src/utils.ts","webpack://slow-calendar/./src/MonthGrid.ts","webpack://slow-calendar/./src/Toolbar.ts","webpack://slow-calendar/./src/Sidebar.ts","webpack://slow-calendar/./src/WeekGrid.ts","webpack://slow-calendar/./src/AppUI.ts","webpack://slow-calendar/./src/CalendarEvent.ts","webpack://slow-calendar/./src/EventPopup.ts","webpack://slow-calendar/./src/Store.ts","webpack://slow-calendar/./src/app.ts","webpack://slow-calendar/./src/events-factory.ts"],"sourcesContent":["/**\n * Assert a condition.\n * @param condition The condition that it should satisfy.\n * @param message The error message.\n * @param args The arguments for replacing placeholders in the message.\n */\nfunction assertType(condition, message, ...args) {\n    if (!condition) {\n        throw new TypeError(format(message, args));\n    }\n}\n/**\n * Convert a text and arguments to one string.\n * @param message The formating text\n * @param args The arguments.\n */\nfunction format(message, args) {\n    let i = 0;\n    return message.replace(/%[os]/gu, () => anyToString(args[i++]));\n}\n/**\n * Convert a value to a string representation.\n * @param x The value to get the string representation.\n */\nfunction anyToString(x) {\n    if (typeof x !== \"object\" || x === null) {\n        return String(x);\n    }\n    return Object.prototype.toString.call(x);\n}\n\nlet currentErrorHandler;\n/**\n * Set the error handler.\n * @param value The error handler to set.\n */\nfunction setErrorHandler(value) {\n    assertType(typeof value === \"function\" || value === undefined, \"The error handler must be a function or undefined, but got %o.\", value);\n    currentErrorHandler = value;\n}\n/**\n * Print a error message.\n * @param maybeError The error object.\n */\nfunction reportError(maybeError) {\n    try {\n        const error = maybeError instanceof Error\n            ? maybeError\n            : new Error(anyToString(maybeError));\n        // Call the user-defined error handler if exists.\n        if (currentErrorHandler) {\n            currentErrorHandler(error);\n            return;\n        }\n        // Dispatch an `error` event if this is on a browser.\n        if (typeof dispatchEvent === \"function\" &&\n            typeof ErrorEvent === \"function\") {\n            dispatchEvent(new ErrorEvent(\"error\", { error, message: error.message }));\n        }\n        // Emit an `uncaughtException` event if this is on Node.js.\n        //istanbul ignore else\n        else if (typeof process !== \"undefined\" &&\n            typeof process.emit === \"function\") {\n            process.emit(\"uncaughtException\", error);\n            return;\n        }\n        // Otherwise, print the error.\n        console.error(error);\n    }\n    catch (_a) {\n        // ignore.\n    }\n}\n\n/**\n * The global object.\n */\n//istanbul ignore next\nconst Global = typeof window !== \"undefined\"\n    ? window\n    : typeof self !== \"undefined\"\n        ? self\n        : typeof global !== \"undefined\"\n            ? global\n            : typeof globalThis !== \"undefined\"\n                ? globalThis\n                : undefined;\n\nlet currentWarnHandler;\n/**\n * Set the warning handler.\n * @param value The warning handler to set.\n */\nfunction setWarningHandler(value) {\n    assertType(typeof value === \"function\" || value === undefined, \"The warning handler must be a function or undefined, but got %o.\", value);\n    currentWarnHandler = value;\n}\n/**\n * The warning information.\n */\nclass Warning {\n    constructor(code, message) {\n        this.code = code;\n        this.message = message;\n    }\n    /**\n     * Report this warning.\n     * @param args The arguments of the warning.\n     */\n    warn(...args) {\n        var _a;\n        try {\n            // Call the user-defined warning handler if exists.\n            if (currentWarnHandler) {\n                currentWarnHandler({ ...this, args });\n                return;\n            }\n            // Otherwise, print the warning.\n            const stack = ((_a = new Error().stack) !== null && _a !== void 0 ? _a : \"\").replace(/^(?:.+?\\n){2}/gu, \"\\n\");\n            console.warn(this.message, ...args, stack);\n        }\n        catch (_b) {\n            // Ignore.\n        }\n    }\n}\n\nconst InitEventWasCalledWhileDispatching = new Warning(\"W01\", \"Unable to initialize event under dispatching.\");\nconst FalsyWasAssignedToCancelBubble = new Warning(\"W02\", \"Assigning any falsy value to 'cancelBubble' property has no effect.\");\nconst TruthyWasAssignedToReturnValue = new Warning(\"W03\", \"Assigning any truthy value to 'returnValue' property has no effect.\");\nconst NonCancelableEventWasCanceled = new Warning(\"W04\", \"Unable to preventDefault on non-cancelable events.\");\nconst CanceledInPassiveListener = new Warning(\"W05\", \"Unable to preventDefault inside passive event listener invocation.\");\nconst EventListenerWasDuplicated = new Warning(\"W06\", \"An event listener wasn't added because it has been added already: %o, %o\");\nconst OptionWasIgnored = new Warning(\"W07\", \"The %o option value was abandoned because the event listener wasn't added as duplicated.\");\nconst InvalidEventListener = new Warning(\"W08\", \"The 'callback' argument must be a function or an object that has 'handleEvent' method: %o\");\nconst InvalidAttributeHandler = new Warning(\"W09\", \"Event attribute handler must be a function: %o\");\n\n/*eslint-disable class-methods-use-this */\n/**\n * An implementation of `Event` interface, that wraps a given event object.\n * `EventTarget` shim can control the internal state of this `Event` objects.\n * @see https://dom.spec.whatwg.org/#event\n */\nclass Event {\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-none\n     */\n    static get NONE() {\n        return NONE;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-capturing_phase\n     */\n    static get CAPTURING_PHASE() {\n        return CAPTURING_PHASE;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-at_target\n     */\n    static get AT_TARGET() {\n        return AT_TARGET;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-bubbling_phase\n     */\n    static get BUBBLING_PHASE() {\n        return BUBBLING_PHASE;\n    }\n    /**\n     * Initialize this event instance.\n     * @param type The type of this event.\n     * @param eventInitDict Options to initialize.\n     * @see https://dom.spec.whatwg.org/#dom-event-event\n     */\n    constructor(type, eventInitDict) {\n        Object.defineProperty(this, \"isTrusted\", {\n            value: false,\n            enumerable: true,\n        });\n        const opts = eventInitDict !== null && eventInitDict !== void 0 ? eventInitDict : {};\n        internalDataMap.set(this, {\n            type: String(type),\n            bubbles: Boolean(opts.bubbles),\n            cancelable: Boolean(opts.cancelable),\n            composed: Boolean(opts.composed),\n            target: null,\n            currentTarget: null,\n            stopPropagationFlag: false,\n            stopImmediatePropagationFlag: false,\n            canceledFlag: false,\n            inPassiveListenerFlag: false,\n            dispatchFlag: false,\n            timeStamp: Date.now(),\n        });\n    }\n    /**\n     * The type of this event.\n     * @see https://dom.spec.whatwg.org/#dom-event-type\n     */\n    get type() {\n        return $(this).type;\n    }\n    /**\n     * The event target of the current dispatching.\n     * @see https://dom.spec.whatwg.org/#dom-event-target\n     */\n    get target() {\n        return $(this).target;\n    }\n    /**\n     * The event target of the current dispatching.\n     * @deprecated Use the `target` property instead.\n     * @see https://dom.spec.whatwg.org/#dom-event-srcelement\n     */\n    get srcElement() {\n        return $(this).target;\n    }\n    /**\n     * The event target of the current dispatching.\n     * @see https://dom.spec.whatwg.org/#dom-event-currenttarget\n     */\n    get currentTarget() {\n        return $(this).currentTarget;\n    }\n    /**\n     * The event target of the current dispatching.\n     * This doesn't support node tree.\n     * @see https://dom.spec.whatwg.org/#dom-event-composedpath\n     */\n    composedPath() {\n        const currentTarget = $(this).currentTarget;\n        if (currentTarget) {\n            return [currentTarget];\n        }\n        return [];\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-none\n     */\n    get NONE() {\n        return NONE;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-capturing_phase\n     */\n    get CAPTURING_PHASE() {\n        return CAPTURING_PHASE;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-at_target\n     */\n    get AT_TARGET() {\n        return AT_TARGET;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-bubbling_phase\n     */\n    get BUBBLING_PHASE() {\n        return BUBBLING_PHASE;\n    }\n    /**\n     * The current event phase.\n     * @see https://dom.spec.whatwg.org/#dom-event-eventphase\n     */\n    get eventPhase() {\n        return $(this).dispatchFlag ? 2 : 0;\n    }\n    /**\n     * Stop event bubbling.\n     * Because this shim doesn't support node tree, this merely changes the `cancelBubble` property value.\n     * @see https://dom.spec.whatwg.org/#dom-event-stoppropagation\n     */\n    stopPropagation() {\n        $(this).stopPropagationFlag = true;\n    }\n    /**\n     * `true` if event bubbling was stopped.\n     * @deprecated\n     * @see https://dom.spec.whatwg.org/#dom-event-cancelbubble\n     */\n    get cancelBubble() {\n        return $(this).stopPropagationFlag;\n    }\n    /**\n     * Stop event bubbling if `true` is set.\n     * @deprecated Use the `stopPropagation()` method instead.\n     * @see https://dom.spec.whatwg.org/#dom-event-cancelbubble\n     */\n    set cancelBubble(value) {\n        if (value) {\n            $(this).stopPropagationFlag = true;\n        }\n        else {\n            FalsyWasAssignedToCancelBubble.warn();\n        }\n    }\n    /**\n     * Stop event bubbling and subsequent event listener callings.\n     * @see https://dom.spec.whatwg.org/#dom-event-stopimmediatepropagation\n     */\n    stopImmediatePropagation() {\n        const data = $(this);\n        data.stopPropagationFlag = data.stopImmediatePropagationFlag = true;\n    }\n    /**\n     * `true` if this event will bubble.\n     * @see https://dom.spec.whatwg.org/#dom-event-bubbles\n     */\n    get bubbles() {\n        return $(this).bubbles;\n    }\n    /**\n     * `true` if this event can be canceled by the `preventDefault()` method.\n     * @see https://dom.spec.whatwg.org/#dom-event-cancelable\n     */\n    get cancelable() {\n        return $(this).cancelable;\n    }\n    /**\n     * `true` if the default behavior will act.\n     * @deprecated Use the `defaultPrevented` proeprty instead.\n     * @see https://dom.spec.whatwg.org/#dom-event-returnvalue\n     */\n    get returnValue() {\n        return !$(this).canceledFlag;\n    }\n    /**\n     * Cancel the default behavior if `false` is set.\n     * @deprecated Use the `preventDefault()` method instead.\n     * @see https://dom.spec.whatwg.org/#dom-event-returnvalue\n     */\n    set returnValue(value) {\n        if (!value) {\n            setCancelFlag($(this));\n        }\n        else {\n            TruthyWasAssignedToReturnValue.warn();\n        }\n    }\n    /**\n     * Cancel the default behavior.\n     * @see https://dom.spec.whatwg.org/#dom-event-preventdefault\n     */\n    preventDefault() {\n        setCancelFlag($(this));\n    }\n    /**\n     * `true` if the default behavior was canceled.\n     * @see https://dom.spec.whatwg.org/#dom-event-defaultprevented\n     */\n    get defaultPrevented() {\n        return $(this).canceledFlag;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-composed\n     */\n    get composed() {\n        return $(this).composed;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-istrusted\n     */\n    //istanbul ignore next\n    get isTrusted() {\n        return false;\n    }\n    /**\n     * @see https://dom.spec.whatwg.org/#dom-event-timestamp\n     */\n    get timeStamp() {\n        return $(this).timeStamp;\n    }\n    /**\n     * @deprecated Don't use this method. The constructor did initialization.\n     */\n    initEvent(type, bubbles = false, cancelable = false) {\n        const data = $(this);\n        if (data.dispatchFlag) {\n            InitEventWasCalledWhileDispatching.warn();\n            return;\n        }\n        internalDataMap.set(this, {\n            ...data,\n            type: String(type),\n            bubbles: Boolean(bubbles),\n            cancelable: Boolean(cancelable),\n            target: null,\n            currentTarget: null,\n            stopPropagationFlag: false,\n            stopImmediatePropagationFlag: false,\n            canceledFlag: false,\n        });\n    }\n}\n//------------------------------------------------------------------------------\n// Helpers\n//------------------------------------------------------------------------------\nconst NONE = 0;\nconst CAPTURING_PHASE = 1;\nconst AT_TARGET = 2;\nconst BUBBLING_PHASE = 3;\n/**\n * Private data for event wrappers.\n */\nconst internalDataMap = new WeakMap();\n/**\n * Get private data.\n * @param event The event object to get private data.\n * @param name The variable name to report.\n * @returns The private data of the event.\n */\nfunction $(event, name = \"this\") {\n    const retv = internalDataMap.get(event);\n    assertType(retv != null, \"'%s' must be an object that Event constructor created, but got another one: %o\", name, event);\n    return retv;\n}\n/**\n * https://dom.spec.whatwg.org/#set-the-canceled-flag\n * @param data private data.\n */\nfunction setCancelFlag(data) {\n    if (data.inPassiveListenerFlag) {\n        CanceledInPassiveListener.warn();\n        return;\n    }\n    if (!data.cancelable) {\n        NonCancelableEventWasCanceled.warn();\n        return;\n    }\n    data.canceledFlag = true;\n}\n// Set enumerable\nObject.defineProperty(Event, \"NONE\", { enumerable: true });\nObject.defineProperty(Event, \"CAPTURING_PHASE\", { enumerable: true });\nObject.defineProperty(Event, \"AT_TARGET\", { enumerable: true });\nObject.defineProperty(Event, \"BUBBLING_PHASE\", { enumerable: true });\nconst keys = Object.getOwnPropertyNames(Event.prototype);\nfor (let i = 0; i < keys.length; ++i) {\n    if (keys[i] === \"constructor\") {\n        continue;\n    }\n    Object.defineProperty(Event.prototype, keys[i], { enumerable: true });\n}\n// Ensure `event instanceof window.Event` is `true`.\nif (typeof Global !== \"undefined\" && typeof Global.Event !== \"undefined\") {\n    Object.setPrototypeOf(Event.prototype, Global.Event.prototype);\n}\n\n/**\n * Create a new InvalidStateError instance.\n * @param message The error message.\n */\nfunction createInvalidStateError(message) {\n    if (Global.DOMException) {\n        return new Global.DOMException(message, \"InvalidStateError\");\n    }\n    if (DOMException == null) {\n        DOMException = class DOMException extends Error {\n            constructor(msg) {\n                super(msg);\n                if (Error.captureStackTrace) {\n                    Error.captureStackTrace(this, DOMException);\n                }\n            }\n            // eslint-disable-next-line class-methods-use-this\n            get code() {\n                return 11;\n            }\n            // eslint-disable-next-line class-methods-use-this\n            get name() {\n                return \"InvalidStateError\";\n            }\n        };\n        Object.defineProperties(DOMException.prototype, {\n            code: { enumerable: true },\n            name: { enumerable: true },\n        });\n        defineErrorCodeProperties(DOMException);\n        defineErrorCodeProperties(DOMException.prototype);\n    }\n    return new DOMException(message);\n}\n//------------------------------------------------------------------------------\n// Helpers\n//------------------------------------------------------------------------------\nlet DOMException;\nconst ErrorCodeMap = {\n    INDEX_SIZE_ERR: 1,\n    DOMSTRING_SIZE_ERR: 2,\n    HIERARCHY_REQUEST_ERR: 3,\n    WRONG_DOCUMENT_ERR: 4,\n    INVALID_CHARACTER_ERR: 5,\n    NO_DATA_ALLOWED_ERR: 6,\n    NO_MODIFICATION_ALLOWED_ERR: 7,\n    NOT_FOUND_ERR: 8,\n    NOT_SUPPORTED_ERR: 9,\n    INUSE_ATTRIBUTE_ERR: 10,\n    INVALID_STATE_ERR: 11,\n    SYNTAX_ERR: 12,\n    INVALID_MODIFICATION_ERR: 13,\n    NAMESPACE_ERR: 14,\n    INVALID_ACCESS_ERR: 15,\n    VALIDATION_ERR: 16,\n    TYPE_MISMATCH_ERR: 17,\n    SECURITY_ERR: 18,\n    NETWORK_ERR: 19,\n    ABORT_ERR: 20,\n    URL_MISMATCH_ERR: 21,\n    QUOTA_EXCEEDED_ERR: 22,\n    TIMEOUT_ERR: 23,\n    INVALID_NODE_TYPE_ERR: 24,\n    DATA_CLONE_ERR: 25,\n};\nfunction defineErrorCodeProperties(obj) {\n    const keys = Object.keys(ErrorCodeMap);\n    for (let i = 0; i < keys.length; ++i) {\n        const key = keys[i];\n        const value = ErrorCodeMap[key];\n        Object.defineProperty(obj, key, {\n            get() {\n                return value;\n            },\n            configurable: true,\n            enumerable: true,\n        });\n    }\n}\n\n/**\n * An implementation of `Event` interface, that wraps a given event object.\n * This class controls the internal state of `Event`.\n * @see https://dom.spec.whatwg.org/#interface-event\n */\nclass EventWrapper extends Event {\n    /**\n     * Wrap a given event object to control states.\n     * @param event The event-like object to wrap.\n     */\n    static wrap(event) {\n        return new (getWrapperClassOf(event))(event);\n    }\n    constructor(event) {\n        super(event.type, {\n            bubbles: event.bubbles,\n            cancelable: event.cancelable,\n            composed: event.composed,\n        });\n        if (event.cancelBubble) {\n            super.stopPropagation();\n        }\n        if (event.defaultPrevented) {\n            super.preventDefault();\n        }\n        internalDataMap$1.set(this, { original: event });\n        // Define accessors\n        const keys = Object.keys(event);\n        for (let i = 0; i < keys.length; ++i) {\n            const key = keys[i];\n            if (!(key in this)) {\n                Object.defineProperty(this, key, defineRedirectDescriptor(event, key));\n            }\n        }\n    }\n    stopPropagation() {\n        super.stopPropagation();\n        const { original } = $$1(this);\n        if (\"stopPropagation\" in original) {\n            original.stopPropagation();\n        }\n    }\n    get cancelBubble() {\n        return super.cancelBubble;\n    }\n    set cancelBubble(value) {\n        super.cancelBubble = value;\n        const { original } = $$1(this);\n        if (\"cancelBubble\" in original) {\n            original.cancelBubble = value;\n        }\n    }\n    stopImmediatePropagation() {\n        super.stopImmediatePropagation();\n        const { original } = $$1(this);\n        if (\"stopImmediatePropagation\" in original) {\n            original.stopImmediatePropagation();\n        }\n    }\n    get returnValue() {\n        return super.returnValue;\n    }\n    set returnValue(value) {\n        super.returnValue = value;\n        const { original } = $$1(this);\n        if (\"returnValue\" in original) {\n            original.returnValue = value;\n        }\n    }\n    preventDefault() {\n        super.preventDefault();\n        const { original } = $$1(this);\n        if (\"preventDefault\" in original) {\n            original.preventDefault();\n        }\n    }\n    get timeStamp() {\n        const { original } = $$1(this);\n        if (\"timeStamp\" in original) {\n            return original.timeStamp;\n        }\n        return super.timeStamp;\n    }\n}\n/**\n * Private data for event wrappers.\n */\nconst internalDataMap$1 = new WeakMap();\n/**\n * Get private data.\n * @param event The event object to get private data.\n * @returns The private data of the event.\n */\nfunction $$1(event) {\n    const retv = internalDataMap$1.get(event);\n    assertType(retv != null, \"'this' is expected an Event object, but got\", event);\n    return retv;\n}\n/**\n * Cache for wrapper classes.\n * @type {WeakMap<Object, Function>}\n * @private\n */\nconst wrapperClassCache = new WeakMap();\n// Make association for wrappers.\nwrapperClassCache.set(Object.prototype, EventWrapper);\nif (typeof Global !== \"undefined\" && typeof Global.Event !== \"undefined\") {\n    wrapperClassCache.set(Global.Event.prototype, EventWrapper);\n}\n/**\n * Get the wrapper class of a given prototype.\n * @param originalEvent The event object to wrap.\n */\nfunction getWrapperClassOf(originalEvent) {\n    const prototype = Object.getPrototypeOf(originalEvent);\n    if (prototype == null) {\n        return EventWrapper;\n    }\n    let wrapper = wrapperClassCache.get(prototype);\n    if (wrapper == null) {\n        wrapper = defineWrapper(getWrapperClassOf(prototype), prototype);\n        wrapperClassCache.set(prototype, wrapper);\n    }\n    return wrapper;\n}\n/**\n * Define new wrapper class.\n * @param BaseEventWrapper The base wrapper class.\n * @param originalPrototype The prototype of the original event.\n */\nfunction defineWrapper(BaseEventWrapper, originalPrototype) {\n    class CustomEventWrapper extends BaseEventWrapper {\n    }\n    const keys = Object.keys(originalPrototype);\n    for (let i = 0; i < keys.length; ++i) {\n        Object.defineProperty(CustomEventWrapper.prototype, keys[i], defineRedirectDescriptor(originalPrototype, keys[i]));\n    }\n    return CustomEventWrapper;\n}\n/**\n * Get the property descriptor to redirect a given property.\n */\nfunction defineRedirectDescriptor(obj, key) {\n    const d = Object.getOwnPropertyDescriptor(obj, key);\n    return {\n        get() {\n            const original = $$1(this).original;\n            const value = original[key];\n            if (typeof value === \"function\") {\n                return value.bind(original);\n            }\n            return value;\n        },\n        set(value) {\n            const original = $$1(this).original;\n            original[key] = value;\n        },\n        configurable: d.configurable,\n        enumerable: d.enumerable,\n    };\n}\n\n/**\n * Create a new listener.\n * @param callback The callback function.\n * @param capture The capture flag.\n * @param passive The passive flag.\n * @param once The once flag.\n * @param signal The abort signal.\n * @param signalListener The abort event listener for the abort signal.\n */\nfunction createListener(callback, capture, passive, once, signal, signalListener) {\n    return {\n        callback,\n        flags: (capture ? 1 /* Capture */ : 0) |\n            (passive ? 2 /* Passive */ : 0) |\n            (once ? 4 /* Once */ : 0),\n        signal,\n        signalListener,\n    };\n}\n/**\n * Set the `removed` flag to the given listener.\n * @param listener The listener to check.\n */\nfunction setRemoved(listener) {\n    listener.flags |= 8 /* Removed */;\n}\n/**\n * Check if the given listener has the `capture` flag or not.\n * @param listener The listener to check.\n */\nfunction isCapture(listener) {\n    return (listener.flags & 1 /* Capture */) === 1 /* Capture */;\n}\n/**\n * Check if the given listener has the `passive` flag or not.\n * @param listener The listener to check.\n */\nfunction isPassive(listener) {\n    return (listener.flags & 2 /* Passive */) === 2 /* Passive */;\n}\n/**\n * Check if the given listener has the `once` flag or not.\n * @param listener The listener to check.\n */\nfunction isOnce(listener) {\n    return (listener.flags & 4 /* Once */) === 4 /* Once */;\n}\n/**\n * Check if the given listener has the `removed` flag or not.\n * @param listener The listener to check.\n */\nfunction isRemoved(listener) {\n    return (listener.flags & 8 /* Removed */) === 8 /* Removed */;\n}\n/**\n * Call an event listener.\n * @param listener The listener to call.\n * @param target The event target object for `thisArg`.\n * @param event The event object for the first argument.\n * @param attribute `true` if this callback is an event attribute handler.\n */\nfunction invokeCallback({ callback }, target, event) {\n    try {\n        if (typeof callback === \"function\") {\n            callback.call(target, event);\n        }\n        else if (typeof callback.handleEvent === \"function\") {\n            callback.handleEvent(event);\n        }\n    }\n    catch (thrownError) {\n        reportError(thrownError);\n    }\n}\n\n/**\n * Find the index of given listener.\n * This returns `-1` if not found.\n * @param list The listener list.\n * @param callback The callback function to find.\n * @param capture The capture flag to find.\n */\nfunction findIndexOfListener({ listeners }, callback, capture) {\n    for (let i = 0; i < listeners.length; ++i) {\n        if (listeners[i].callback === callback &&\n            isCapture(listeners[i]) === capture) {\n            return i;\n        }\n    }\n    return -1;\n}\n/**\n * Add the given listener.\n * Does copy-on-write if needed.\n * @param list The listener list.\n * @param callback The callback function.\n * @param capture The capture flag.\n * @param passive The passive flag.\n * @param once The once flag.\n * @param signal The abort signal.\n */\nfunction addListener(list, callback, capture, passive, once, signal) {\n    let signalListener;\n    if (signal) {\n        signalListener = removeListener.bind(null, list, callback, capture);\n        signal.addEventListener(\"abort\", signalListener);\n    }\n    const listener = createListener(callback, capture, passive, once, signal, signalListener);\n    if (list.cow) {\n        list.cow = false;\n        list.listeners = [...list.listeners, listener];\n    }\n    else {\n        list.listeners.push(listener);\n    }\n    return listener;\n}\n/**\n * Remove a listener.\n * @param list The listener list.\n * @param callback The callback function to find.\n * @param capture The capture flag to find.\n * @returns `true` if it mutated the list directly.\n */\nfunction removeListener(list, callback, capture) {\n    const index = findIndexOfListener(list, callback, capture);\n    if (index !== -1) {\n        return removeListenerAt(list, index);\n    }\n    return false;\n}\n/**\n * Remove a listener.\n * @param list The listener list.\n * @param index The index of the target listener.\n * @param disableCow Disable copy-on-write if true.\n * @returns `true` if it mutated the `listeners` array directly.\n */\nfunction removeListenerAt(list, index, disableCow = false) {\n    const listener = list.listeners[index];\n    // Set the removed flag.\n    setRemoved(listener);\n    // Dispose the abort signal listener if exists.\n    if (listener.signal) {\n        listener.signal.removeEventListener(\"abort\", listener.signalListener);\n    }\n    // Remove it from the array.\n    if (list.cow && !disableCow) {\n        list.cow = false;\n        list.listeners = list.listeners.filter((_, i) => i !== index);\n        return false;\n    }\n    list.listeners.splice(index, 1);\n    return true;\n}\n\n/**\n * Create a new `ListenerListMap` object.\n */\nfunction createListenerListMap() {\n    return Object.create(null);\n}\n/**\n * Get the listener list of the given type.\n * If the listener list has not been initialized, initialize and return it.\n * @param listenerMap The listener list map.\n * @param type The event type to get.\n */\nfunction ensureListenerList(listenerMap, type) {\n    var _a;\n    return ((_a = listenerMap[type]) !== null && _a !== void 0 ? _a : (listenerMap[type] = {\n        attrCallback: undefined,\n        attrListener: undefined,\n        cow: false,\n        listeners: [],\n    }));\n}\n\n/**\n * An implementation of the `EventTarget` interface.\n * @see https://dom.spec.whatwg.org/#eventtarget\n */\nclass EventTarget {\n    /**\n     * Initialize this instance.\n     */\n    constructor() {\n        internalDataMap$2.set(this, createListenerListMap());\n    }\n    // Implementation\n    addEventListener(type0, callback0, options0) {\n        const listenerMap = $$2(this);\n        const { callback, capture, once, passive, signal, type, } = normalizeAddOptions(type0, callback0, options0);\n        if (callback == null || (signal === null || signal === void 0 ? void 0 : signal.aborted)) {\n            return;\n        }\n        const list = ensureListenerList(listenerMap, type);\n        // Find existing listener.\n        const i = findIndexOfListener(list, callback, capture);\n        if (i !== -1) {\n            warnDuplicate(list.listeners[i], passive, once, signal);\n            return;\n        }\n        // Add the new listener.\n        addListener(list, callback, capture, passive, once, signal);\n    }\n    // Implementation\n    removeEventListener(type0, callback0, options0) {\n        const listenerMap = $$2(this);\n        const { callback, capture, type } = normalizeOptions(type0, callback0, options0);\n        const list = listenerMap[type];\n        if (callback != null && list) {\n            removeListener(list, callback, capture);\n        }\n    }\n    // Implementation\n    dispatchEvent(e) {\n        const list = $$2(this)[String(e.type)];\n        if (list == null) {\n            return true;\n        }\n        const event = e instanceof Event ? e : EventWrapper.wrap(e);\n        const eventData = $(event, \"event\");\n        if (eventData.dispatchFlag) {\n            throw createInvalidStateError(\"This event has been in dispatching.\");\n        }\n        eventData.dispatchFlag = true;\n        eventData.target = eventData.currentTarget = this;\n        if (!eventData.stopPropagationFlag) {\n            const { cow, listeners } = list;\n            // Set copy-on-write flag.\n            list.cow = true;\n            // Call listeners.\n            for (let i = 0; i < listeners.length; ++i) {\n                const listener = listeners[i];\n                // Skip if removed.\n                if (isRemoved(listener)) {\n                    continue;\n                }\n                // Remove this listener if has the `once` flag.\n                if (isOnce(listener) && removeListenerAt(list, i, !cow)) {\n                    // Because this listener was removed, the next index is the\n                    // same as the current value.\n                    i -= 1;\n                }\n                // Call this listener with the `passive` flag.\n                eventData.inPassiveListenerFlag = isPassive(listener);\n                invokeCallback(listener, this, event);\n                eventData.inPassiveListenerFlag = false;\n                // Stop if the `event.stopImmediatePropagation()` method was called.\n                if (eventData.stopImmediatePropagationFlag) {\n                    break;\n                }\n            }\n            // Restore copy-on-write flag.\n            if (!cow) {\n                list.cow = false;\n            }\n        }\n        eventData.target = null;\n        eventData.currentTarget = null;\n        eventData.stopImmediatePropagationFlag = false;\n        eventData.stopPropagationFlag = false;\n        eventData.dispatchFlag = false;\n        return !eventData.canceledFlag;\n    }\n}\n/**\n * Internal data.\n */\nconst internalDataMap$2 = new WeakMap();\n/**\n * Get private data.\n * @param target The event target object to get private data.\n * @param name The variable name to report.\n * @returns The private data of the event.\n */\nfunction $$2(target, name = \"this\") {\n    const retv = internalDataMap$2.get(target);\n    assertType(retv != null, \"'%s' must be an object that EventTarget constructor created, but got another one: %o\", name, target);\n    return retv;\n}\n/**\n * Normalize options.\n * @param options The options to normalize.\n */\nfunction normalizeAddOptions(type, callback, options) {\n    var _a;\n    assertCallback(callback);\n    if (typeof options === \"object\" && options !== null) {\n        return {\n            type: String(type),\n            callback: callback !== null && callback !== void 0 ? callback : undefined,\n            capture: Boolean(options.capture),\n            passive: Boolean(options.passive),\n            once: Boolean(options.once),\n            signal: (_a = options.signal) !== null && _a !== void 0 ? _a : undefined,\n        };\n    }\n    return {\n        type: String(type),\n        callback: callback !== null && callback !== void 0 ? callback : undefined,\n        capture: Boolean(options),\n        passive: false,\n        once: false,\n        signal: undefined,\n    };\n}\n/**\n * Normalize options.\n * @param options The options to normalize.\n */\nfunction normalizeOptions(type, callback, options) {\n    assertCallback(callback);\n    if (typeof options === \"object\" && options !== null) {\n        return {\n            type: String(type),\n            callback: callback !== null && callback !== void 0 ? callback : undefined,\n            capture: Boolean(options.capture),\n        };\n    }\n    return {\n        type: String(type),\n        callback: callback !== null && callback !== void 0 ? callback : undefined,\n        capture: Boolean(options),\n    };\n}\n/**\n * Assert the type of 'callback' argument.\n * @param callback The callback to check.\n */\nfunction assertCallback(callback) {\n    if (typeof callback === \"function\" ||\n        (typeof callback === \"object\" &&\n            callback !== null &&\n            typeof callback.handleEvent === \"function\")) {\n        return;\n    }\n    if (callback == null || typeof callback === \"object\") {\n        InvalidEventListener.warn(callback);\n        return;\n    }\n    throw new TypeError(format(InvalidEventListener.message, [callback]));\n}\n/**\n * Print warning for duplicated.\n * @param listener The current listener that is duplicated.\n * @param passive The passive flag of the new duplicated listener.\n * @param once The once flag of the new duplicated listener.\n * @param signal The signal object of the new duplicated listener.\n */\nfunction warnDuplicate(listener, passive, once, signal) {\n    EventListenerWasDuplicated.warn(isCapture(listener) ? \"capture\" : \"bubble\", listener.callback);\n    if (isPassive(listener) !== passive) {\n        OptionWasIgnored.warn(\"passive\");\n    }\n    if (isOnce(listener) !== once) {\n        OptionWasIgnored.warn(\"once\");\n    }\n    if (listener.signal !== signal) {\n        OptionWasIgnored.warn(\"signal\");\n    }\n}\n// Set enumerable\nconst keys$1 = Object.getOwnPropertyNames(EventTarget.prototype);\nfor (let i = 0; i < keys$1.length; ++i) {\n    if (keys$1[i] === \"constructor\") {\n        continue;\n    }\n    Object.defineProperty(EventTarget.prototype, keys$1[i], { enumerable: true });\n}\n// Ensure `eventTarget instanceof window.EventTarget` is `true`.\nif (typeof Global !== \"undefined\" &&\n    typeof Global.EventTarget !== \"undefined\") {\n    Object.setPrototypeOf(EventTarget.prototype, Global.EventTarget.prototype);\n}\n\n/**\n * Get the current value of a given event attribute.\n * @param target The `EventTarget` object to get.\n * @param type The event type.\n */\nfunction getEventAttributeValue(target, type) {\n    var _a, _b;\n    const listMap = $$2(target, \"target\");\n    return (_b = (_a = listMap[type]) === null || _a === void 0 ? void 0 : _a.attrCallback) !== null && _b !== void 0 ? _b : null;\n}\n/**\n * Set an event listener to a given event attribute.\n * @param target The `EventTarget` object to set.\n * @param type The event type.\n * @param callback The event listener.\n */\nfunction setEventAttributeValue(target, type, callback) {\n    if (callback != null && typeof callback !== \"function\") {\n        InvalidAttributeHandler.warn(callback);\n    }\n    if (typeof callback === \"function\" ||\n        (typeof callback === \"object\" && callback !== null)) {\n        upsertEventAttributeListener(target, type, callback);\n    }\n    else {\n        removeEventAttributeListener(target, type);\n    }\n}\n//------------------------------------------------------------------------------\n// Helpers\n//------------------------------------------------------------------------------\n/**\n * Update or insert the given event attribute handler.\n * @param target The `EventTarget` object to set.\n * @param type The event type.\n * @param callback The event listener.\n */\nfunction upsertEventAttributeListener(target, type, callback) {\n    const list = ensureListenerList($$2(target, \"target\"), String(type));\n    list.attrCallback = callback;\n    if (list.attrListener == null) {\n        list.attrListener = addListener(list, defineEventAttributeCallback(list), false, false, false, undefined);\n    }\n}\n/**\n * Remove the given event attribute handler.\n * @param target The `EventTarget` object to remove.\n * @param type The event type.\n * @param callback The event listener.\n */\nfunction removeEventAttributeListener(target, type) {\n    const listMap = $$2(target, \"target\");\n    const list = listMap[String(type)];\n    if (list && list.attrListener) {\n        removeListener(list, list.attrListener.callback, false);\n        list.attrCallback = list.attrListener = undefined;\n    }\n}\n/**\n * Define the callback function for the given listener list object.\n * It calls `attrCallback` property if the property value is a function.\n * @param list The `ListenerList` object.\n */\nfunction defineEventAttributeCallback(list) {\n    return function (event) {\n        const callback = list.attrCallback;\n        if (typeof callback === \"function\") {\n            callback.call(this, event);\n        }\n    };\n}\n\n/**\n * Define an `EventTarget` class that has event attibutes.\n * @param types The types to define event attributes.\n * @deprecated Use `getEventAttributeValue`/`setEventAttributeValue` pair on your derived class instead because of static analysis friendly.\n */\nfunction defineCustomEventTarget(...types) {\n    class CustomEventTarget extends EventTarget {\n    }\n    for (let i = 0; i < types.length; ++i) {\n        defineEventAttribute(CustomEventTarget.prototype, types[i]);\n    }\n    return CustomEventTarget;\n}\n/**\n * Define an event attribute.\n * @param target The `EventTarget` object to define an event attribute.\n * @param type The event type to define.\n * @param _eventClass Unused, but to infer `Event` class type.\n * @deprecated Use `getEventAttributeValue`/`setEventAttributeValue` pair on your derived class instead because of static analysis friendly.\n */\nfunction defineEventAttribute(target, type, _eventClass) {\n    Object.defineProperty(target, `on${type}`, {\n        get() {\n            return getEventAttributeValue(this, type);\n        },\n        set(value) {\n            setEventAttributeValue(this, type, value);\n        },\n        configurable: true,\n        enumerable: true,\n    });\n}\n\nexport default EventTarget;\nexport { Event, EventTarget, defineCustomEventTarget, defineEventAttribute, getEventAttributeValue, setErrorHandler, setEventAttributeValue, setWarningHandler };\n//# sourceMappingURL=index.mjs.map\n","import { CalendarEvent } from \"./CalendarEvent\";\r\n\r\nexport function getEventsForDay(events: CalendarEvent[], date: Date): CalendarEvent[] {\r\n  return events.filter(event => {\r\n    return event.date.getFullYear() === date.getFullYear() &&\r\n      event.date.getMonth() === date.getMonth() &&\r\n      event.date.getDate() === date.getDate();\r\n  }).sort((a, b) => {\r\n    // If one has no time property, it should be first.\r\n    // If both have no time, no sorting is needed.\r\n    if (!a.time && !!b.time) {\r\n      return -1;\r\n    } else if (!!a.time && !b.time) {\r\n      return 1;\r\n    } else if (!a.time && !b.time) {\r\n      return 0;\r\n    }\r\n\r\n    // Sort the events by time. Times are strings in 24\r\n    // formats, so just remove the leading 0, if any, and\r\n    // the colon, and compare the numbers.\r\n    const aTime = parseInt(a.time.replace(/^0/, '').replace(':', ''));\r\n    const bTime = parseInt(b.time.replace(/^0/, '').replace(':', ''));\r\n    return aTime - bTime;\r\n  }).sort((a, b) => {\r\n    // Sort the events that are multi-days first.\r\n    if (!!a.multiDays && !b.multiDays) {\r\n      return -1;\r\n    } else if (!a.multiDays && !!b.multiDays) {\r\n      return 1;\r\n    }\r\n  });\r\n}\r\n","import { EventTarget, Event } from \"event-target-shim\";\r\nimport { CalendarEvent } from \"./CalendarEvent\";\r\nimport { getEventsForDay } from \"./utils\";\r\n\r\nconst DAYS = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];\r\n\r\nexport class MonthGrid extends EventTarget {\r\n  private rootEl: HTMLElement;\r\n  private _date: Date = new Date();\r\n  private _events: CalendarEvent[];\r\n\r\n  constructor(private el: HTMLElement, date: Date, events: CalendarEvent[]) {\r\n    super();\r\n\r\n    this.rootEl = el;\r\n    this._date = date;\r\n    this._events = events;\r\n\r\n    this.render();\r\n  }\r\n  \r\n  render() {\r\n    this.rootEl.innerHTML = '';\r\n\r\n    const month = this._date.getMonth();\r\n    const year = this._date.getFullYear();\r\n  \r\n    const firstDay = new Date(year, month, 1);\r\n    const lastDay = new Date(year, month + 1, 0);\r\n  \r\n    const firstWeekday = firstDay.getDay();\r\n    const lastWeekday = lastDay.getDay();\r\n    const daysInMonth = lastDay.getDate();\r\n    const daysInLastMonth = new Date(year, month, 0).getDate();\r\n    const days: {date: Date, html: string}[] = [];\r\n  \r\n    let indexInGrid = 0;\r\n\r\n    // Days from last month\r\n    for (let i = daysInLastMonth - firstWeekday + 1; i <= daysInLastMonth; i++) {\r\n      const date = new Date(year, month - 1, i);\r\n      days.push({date, html: this.renderDay(date, i, indexInGrid, 'prev-month')});\r\n      indexInGrid++;\r\n    }\r\n  \r\n    // Days from this month\r\n    for (let i = 1; i <= daysInMonth; i++) {\r\n      const date = new Date(year, month, i);\r\n      days.push({date, html: this.renderDay(date, i, indexInGrid)});\r\n      indexInGrid++;\r\n    }\r\n  \r\n    // Days from next month\r\n    for (let i = 1; i <= 7 - lastWeekday - 1; i++) {\r\n      const date = new Date(year, month + 1, i);\r\n      days.push({date, html: this.renderDay(date, i, indexInGrid, 'next-month')});\r\n      indexInGrid++;\r\n    }\r\n\r\n    for (const {date, html} of days) {\r\n      this.rootEl.insertAdjacentHTML('beforeend', html);\r\n      const dayEl = this.rootEl.lastElementChild as HTMLElement;\r\n\r\n      dayEl.addEventListener('click', (e: MouseEvent) => {\r\n        // @ts-ignore\r\n        this.dispatchEvent(new CustomEvent('day-clicked', {detail: date}));\r\n      });\r\n    }\r\n\r\n    this.updateNowMarker();\r\n  }\r\n\r\n  renderDay(date: Date, dayNumber: number, indexInGrid: number, className: string = ''): string {\r\n    const events = getEventsForDay(this._events, date);\r\n    const isToday = date.toDateString() === new Date().toDateString();\r\n\r\n    return `\r\n      <div class=\"day ${isToday ? 'today' : ''} ${className}\">\r\n        <div class=\"header\">\r\n          <span class=\"day-name\">${DAYS[date.getDay()]}</span>\r\n          <span class=\"day-number\">${dayNumber}</span>\r\n        </div>\r\n        <ul class=\"events\">\r\n          ${events.map(event => event.asOneLineHTML(indexInGrid)).join('')}\r\n        </ul>\r\n      </div>\r\n    `;\r\n  }\r\n\r\n  updateNowMarker() {\r\n    const now = new Date();\r\n    const time = now.getHours() * 100 + now.getMinutes();\r\n    const ratio = time * 100 / 2400;\r\n\r\n    // Forcing a sync reflow here by querying the offsetHeight of the day cell.\r\n    // Could just set the % value directly instead.\r\n    const dayEl = this.rootEl.querySelector('.day.today') as HTMLElement;\r\n    if (dayEl) {\r\n      const pxHeight = dayEl.offsetHeight;\r\n      const markerHeight = pxHeight * ratio / 100;\r\n\r\n      this.rootEl.style.setProperty('--now-marker', `${markerHeight}px`);\r\n    }\r\n\r\n    setTimeout(() => this.updateNowMarker(), 1000);\r\n  }\r\n\r\n  set date(date: Date) {\r\n    this._date = date;\r\n    this.render();\r\n  }\r\n\r\n  set events(events: CalendarEvent[]) {\r\n    this._events = events;\r\n    this.render();\r\n  }\r\n}\r\n","import { EventTarget, Event } from \"event-target-shim\";\r\n\r\nconst MONTHS = [\r\n  'January', 'February', 'March', 'April', 'May', 'June', 'July', 'August',\r\n  'September', 'October', 'November','December'\r\n];\r\n\r\nexport class Toolbar extends EventTarget {\r\n  private rootEl: HTMLElement;\r\n  private _date: Date;\r\n  private selectedMode: string = \"month\";\r\n\r\n  constructor(private el: HTMLElement, date: Date, mode: string) {\r\n    super();\r\n\r\n    this.rootEl = el;\r\n    this._date = date;\r\n    this.selectedMode = mode;\r\n\r\n    this.render();\r\n  }\r\n\r\n  formatMonthYear() {\r\n    const month = this._date.getMonth();\r\n    const year = this._date.getFullYear();\r\n    return `${MONTHS[month]} ${year}`;\r\n  }\r\n  \r\n  render() {\r\n    this.rootEl.innerHTML = `\r\n      <button id=\"prev-month\">Prev</button>\r\n      <button id=\"today\" class=\"primary\">Today</button>\r\n      <button id=\"next-month\">Next</button>\r\n      <span id=\"month-year\">${this.formatMonthYear()}</span>\r\n      <input type=\"radio\" name=\"view\" id=\"month-view\" ${this.selectedMode === \"month\" ? \"checked\" : \"\"}>\r\n      <label for=\"month-view\">Month</label>\r\n      <input type=\"radio\" name=\"view\" id=\"week-view\" ${this.selectedMode === \"week\" ? \"checked\" : \"\"}>\r\n      <label for=\"week-view\">Week</label>      \r\n    `;\r\n\r\n    this.rootEl.querySelector('#prev-month').addEventListener('click', () => {\r\n      console.log(`Prev ${this.selectedMode} clicked`);\r\n      this.dispatchEvent(new Event('prev'));\r\n    });\r\n\r\n    this.rootEl.querySelector('#next-month').addEventListener('click', () => {\r\n      console.log(`Next ${this.selectedMode} clicked`);\r\n      this.dispatchEvent(new Event('next'));\r\n    });\r\n\r\n    this.rootEl.querySelector('#today').addEventListener('click', () => {\r\n      console.log(\"Today clicked\");\r\n      this.dispatchEvent(new Event('today'));\r\n    });\r\n\r\n    this.rootEl.querySelector('#month-view').addEventListener('click', () => {\r\n      console.log(\"Month view clicked\");\r\n      this.dispatchEvent(new Event('month-view'));\r\n      this.selectedMode = \"month\";\r\n    });\r\n    \r\n    this.rootEl.querySelector('#week-view').addEventListener('click', () => {\r\n      console.log(\"Week view clicked\");\r\n      this.dispatchEvent(new Event('week-view'));\r\n      this.selectedMode = \"week\";\r\n    });\r\n  }\r\n\r\n  set date(date: Date) {\r\n    this._date = date;\r\n    this.render();\r\n  }\r\n}\r\n","import { CalendarEvent } from \"./CalendarEvent\";\r\n\r\nexport class Sidebar {\r\n  private rootEl: HTMLElement;\r\n  private _events: CalendarEvent[] = [];\r\n  private _date: Date = new Date();\r\n\r\n  constructor(private el: HTMLElement, date: Date, events: CalendarEvent[]) {\r\n    this.rootEl = el;\r\n    this._date = date;\r\n    this._events = events;\r\n\r\n    this.render();\r\n  }\r\n  \r\n  render() {\r\n    this.rootEl.innerHTML = `\r\n      <h2>Click a day to view events</h2>\r\n    `;\r\n\r\n    if (!this._events.length) {\r\n      return;\r\n    }\r\n\r\n    this.rootEl.innerHTML = '';\r\n\r\n    const ul = document.createElement('ul');\r\n    ul.className = 'events';\r\n    this.rootEl.appendChild(ul);\r\n\r\n    for (const event of this._events) {\r\n      ul.innerHTML += event.asFullHTML();\r\n    }\r\n  }\r\n\r\n  set events(events: CalendarEvent[]) {\r\n    this._events = events;\r\n    this.render();\r\n  }\r\n}\r\n``","import { EventTarget, Event } from \"event-target-shim\";\r\nimport { CalendarEvent } from \"./CalendarEvent\";\r\nimport { getEventsForDay } from \"./utils\";\r\n\r\nconst DAYS = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'];\r\n\r\nexport class WeekGrid extends EventTarget {\r\n  private rootEl: HTMLElement;\r\n  private _date: Date = new Date();\r\n  private _events: CalendarEvent[];\r\n\r\n  constructor(private el: HTMLElement, date: Date, events: CalendarEvent[]) {\r\n    super();\r\n\r\n    this.rootEl = el;\r\n    this._date = date;\r\n    this._events = events;\r\n\r\n    this.render();\r\n  }\r\n  \r\n  render() {\r\n    this.rootEl.innerHTML = '';\r\n\r\n    // Find the dates that make the week around this._date.\r\n    const month = this._date.getMonth();\r\n    const year = this._date.getFullYear();\r\n    const date = this._date.getDate();\r\n    const day = this._date.getDay();\r\n\r\n    const firstDay = new Date(year, month, date - day);\r\n\r\n    // Iterate over the dates and render the days.\r\n    const days: {date: Date, html: string}[] = [];\r\n    for (let i = 0; i < 7; i++) {\r\n      const date = new Date(year, month, firstDay.getDate() + i);\r\n      this.rootEl.insertAdjacentHTML('beforeend', this.renderDay(date, date.getDate()));\r\n\r\n      const dayEl = this.rootEl.lastElementChild as HTMLElement;\r\n      dayEl.addEventListener('click', (e: MouseEvent) => {\r\n        // @ts-ignore\r\n        this.dispatchEvent(new CustomEvent('day-clicked', {detail: date}));\r\n      });\r\n    }\r\n\r\n    this.updateNowMarker();\r\n  }\r\n\r\n  renderDay(date: Date, dayNumber: number, className: string = ''): string {\r\n    const events = getEventsForDay(this._events, date);\r\n    const isToday = date.toDateString() === new Date().toDateString();\r\n\r\n    return `\r\n      <div class=\"day ${isToday ? 'today' : ''} ${className}\">\r\n        <div class=\"header\">\r\n          <span class=\"day-name\">${DAYS[date.getDay()]}</span>\r\n          <span class=\"day-number\">${dayNumber}</span>\r\n        </div>\r\n        <ul class=\"events\">\r\n          ${events.map(event => event.asMediumLengthHTML()).join('')}\r\n        </ul>\r\n      </div>\r\n    `;\r\n  }\r\n\r\n  updateNowMarker() {\r\n    const now = new Date();\r\n    const time = now.getHours() * 100 + now.getMinutes();\r\n    const ratio = time * 100 / 2400;\r\n\r\n    // Forcing a sync reflow here by querying the offsetHeight of the day cell.\r\n    // Could just set the % value directly instead.\r\n    const dayEl = this.rootEl.querySelector('.day.today') as HTMLElement;\r\n    if (dayEl) {\r\n      const pxHeight = dayEl.offsetHeight;\r\n      const markerHeight = pxHeight * ratio / 100;\r\n\r\n      this.rootEl.style.setProperty('--now-marker', `${markerHeight}px`);\r\n    }\r\n\r\n    setTimeout(() => this.updateNowMarker(), 1000);\r\n  }\r\n\r\n  set date(date: Date) {\r\n    this._date = date;\r\n    this.render();\r\n  }\r\n\r\n  set events(events: CalendarEvent[]) {\r\n    this._events = events;\r\n    this.render();\r\n  }\r\n}\r\n","import { EventTarget, Event } from \"event-target-shim\";\r\nimport { CalendarEvent } from \"./CalendarEvent\";\r\nimport { MonthGrid } from \"./MonthGrid\";\r\nimport { Toolbar } from \"./Toolbar\";\r\nimport { Sidebar } from \"./Sidebar\";\r\nimport { getEventsForDay } from \"./utils\";\r\nimport { WeekGrid } from \"./WeekGrid\";\r\n\r\nexport class AppUI extends EventTarget {\r\n  private rootEl: HTMLElement;\r\n\r\n  private monthGridEl: HTMLElement;\r\n  public monthGrid: MonthGrid;\r\n\r\n  private weekGridEl: HTMLElement;\r\n  public weekGrid: WeekGrid;\r\n\r\n  private toolbarEl: HTMLElement;\r\n  private toolbar: Toolbar;\r\n\r\n  private sidebarEl: HTMLElement;\r\n  private sidebar: Sidebar;\r\n\r\n  private _date: Date;\r\n  private _events: CalendarEvent[];\r\n\r\n  private mode: string = \"month\";\r\n\r\n  constructor(el: HTMLElement, date: Date, mode: string, events: CalendarEvent[]) {\r\n    super();\r\n\r\n    this.rootEl = el;\r\n    this._date = date;\r\n    this._events = events;\r\n    this.mode = mode;\r\n\r\n    this.render();\r\n  }\r\n\r\n  render() {\r\n    this.rootEl.innerHTML = '';\r\n\r\n    // Create an element for the toolbar.\r\n    this.toolbarEl = document.createElement('div');\r\n    this.toolbarEl.id = 'toolbar';\r\n    this.rootEl.appendChild(this.toolbarEl);\r\n\r\n    // Init the toolbar component in it.\r\n    this.toolbar = new Toolbar(this.toolbarEl, this._date, this.mode);\r\n\r\n    this.toolbar.addEventListener('prev', () => {\r\n      const newDate = this.mode === \"month\"\r\n        ? new Date(this._date.getFullYear(), this._date.getMonth() - 1, 1)\r\n        : new Date(this._date.getFullYear(), this._date.getMonth(), this._date.getDate() - 7);\r\n      this.date = newDate;\r\n\r\n      // @ts-ignore\r\n      this.dispatchEvent(new CustomEvent('date-changed', {detail: this._date}));\r\n    });\r\n    \r\n    this.toolbar.addEventListener('next', () => {\r\n      const newDate = this.mode === \"month\"\r\n        ? new Date(this._date.getFullYear(), this._date.getMonth() + 1, 1)\r\n        : new Date(this._date.getFullYear(), this._date.getMonth(), this._date.getDate() + 7);\r\n      this.date = newDate;\r\n\r\n      // @ts-ignore\r\n      this.dispatchEvent(new CustomEvent('date-changed', {detail: this._date}));\r\n    });\r\n    \r\n    this.toolbar.addEventListener('today', () => {\r\n      this.date = new Date();\r\n      \r\n      // @ts-ignore\r\n      this.dispatchEvent(new CustomEvent('date-changed', {detail: this._date}));\r\n    });\r\n\r\n    this.toolbar.addEventListener('month-view', () => {\r\n      this.mode = \"month\";\r\n      this.monthGridEl.style.display = 'grid';\r\n      this.weekGridEl.style.display = 'none';\r\n\r\n      // @ts-ignore\r\n      this.dispatchEvent(new CustomEvent('mode-changed', {detail: this.mode}));\r\n    });\r\n    \r\n    this.toolbar.addEventListener('week-view', () => {\r\n      this.mode = \"week\";\r\n      this.monthGridEl.style.display = 'none';\r\n      this.weekGridEl.style.display = 'grid';\r\n\r\n      // @ts-ignore\r\n      this.dispatchEvent(new CustomEvent('mode-changed', {detail: this.mode}));\r\n    });\r\n\r\n    // Create an element for the month grid.\r\n    this.monthGridEl = document.createElement('div');\r\n    this.monthGridEl.id = 'month-grid';\r\n    this.rootEl.appendChild(this.monthGridEl);\r\n\r\n    // Init the month grid component in it.\r\n    this.monthGrid = new MonthGrid(this.monthGridEl, this._date, this._events);\r\n\r\n    // Create an element for the week grid.\r\n    this.weekGridEl = document.createElement('div');\r\n    this.weekGridEl.id = 'week-grid';\r\n    this.rootEl.appendChild(this.weekGridEl);\r\n\r\n    // Init the week grid component in it.\r\n    this.weekGrid = new WeekGrid(this.weekGridEl, this._date, this._events);\r\n\r\n    // Create an element for the sidebar.\r\n    this.sidebarEl = document.createElement('div');\r\n    this.sidebarEl.id = 'sidebar';\r\n    this.rootEl.appendChild(this.sidebarEl);\r\n\r\n    // Init the sidebar component in it.\r\n    this.sidebar = new Sidebar(this.sidebarEl, this._date, this._events);\r\n\r\n    // Listen for day-clicked events from the month and week grids to show the full\r\n    // list in the sidebar.\r\n    // @ts-ignore\r\n    this.monthGrid.addEventListener('day-clicked', (e: CustomEvent) => {\r\n      const date = e.detail;\r\n      const events = getEventsForDay(this._events, date);\r\n      this.sidebar.events = events;\r\n    });\r\n    // @ts-ignore\r\n    this.weekGrid.addEventListener('day-clicked', (e: CustomEvent) => {\r\n      const date = e.detail;\r\n      const events = getEventsForDay(this._events, date);\r\n      this.sidebar.events = events;\r\n    });\r\n\r\n    // Start with today.\r\n    this.sidebar.events = getEventsForDay(this._events, new Date());\r\n\r\n    // Show/hide the right mode\r\n    if (this.mode === \"week\") {\r\n      this.monthGridEl.style.display = 'none';\r\n      this.weekGridEl.style.display = 'grid';\r\n    } else {\r\n      this.monthGridEl.style.display = 'grid';\r\n      this.weekGridEl.style.display = 'none';\r\n    }\r\n  }\r\n\r\n  set date(date: Date) {\r\n    console.log(\"Setting date to\", date);\r\n    this._date = date;\r\n    this.monthGrid.date = date;\r\n    this.weekGrid.date = date;\r\n    this.toolbar.date = date;\r\n    this.sidebar.events = getEventsForDay(this._events, this._date);\r\n  }\r\n\r\n  set events(events: CalendarEvent[]) {\r\n    this._events = events;\r\n    this.monthGrid.events = events;\r\n    this.weekGrid.events = events;\r\n    this.sidebar.events = getEventsForDay(this._events, this._date);\r\n  }\r\n}\r\n","export class CalendarEvent {\r\n  title: string;\r\n  id: string;\r\n  date: Date;\r\n  time: string|undefined;\r\n  duration: number|undefined;\r\n  repeat: string|undefined;\r\n  description: string|undefined;\r\n  color: string|undefined;\r\n  location: string|undefined;\r\n  multiDays: {start: Date, end: Date}|null;\r\n  rsvp: boolean = false;\r\n\r\n  constructor(title: string, id: string, date: Date) {\r\n    this.title = title;\r\n    this.id = id;\r\n    this.date = date;\r\n  }\r\n\r\n  asOneLineHTML(indexInGrid: number|undefined = undefined): string {\r\n    const time = this.time ? `<span class=\"time\">${this.time}</span>` : '';\r\n\r\n    return `\r\n      <li id=\"id-${this.id}\" data-index-in-grid=\"${indexInGrid}\" class=\"event${!!this.multiDays ? \" multi-day\" : \"\"}${this.rsvp ? \"\" : \" unconfirmed\"}\" style=\"--event-color: ${this.color};\">\r\n        <div>${time} <span class=\"title\">${this.title}</span></div>\r\n      </li>\r\n    `;\r\n  }\r\n\r\n  asMediumLengthHTML(): string {\r\n    const time = this.time ? `<span class=\"time\">${this.time}</span>` : '';\r\n\r\n    return `\r\n      <li id=\"id-${this.id}\" class=\"event${!!this.multiDays ? \" multi-day\" : \"\"}${this.rsvp ? \"\" : \" unconfirmed\"}\" style=\"--event-color: ${this.color};\">\r\n        <div>${time}</div>\r\n        <p class=\"title\">${this.title}</p>\r\n        <p>${this.description}</p>\r\n      </li>\r\n    `;\r\n  }\r\n\r\n  asFullHTML(): string {\r\n    let when = \"\";\r\n      \r\n    if (!this.multiDays) {\r\n      when = this.date.toDateString();\r\n      if (this.time) {\r\n        when += `, at ${this.time}`;\r\n      }\r\n      if (this.repeat) {\r\n        when += `, ${this.repeat}`;\r\n      }\r\n      if (this.duration) {\r\n        when += ` (${this.duration} minutes)`;\r\n      }\r\n    } else {\r\n      when = `From ${this.multiDays.start.toDateString()} to ${this.multiDays.end.toDateString()}`;\r\n    }\r\n\r\n    return `\r\n      <li class=\"event${this.rsvp ? \"\" : \" unconfirmed\"}\" style=\"--event-color: ${this.color};\">\r\n        <div>\r\n          <h2>${this.title}</h2>\r\n          <p class=\"when\">${when}</p>\r\n          ${this.location ? `<p class=\"where\">${this.location}</p>` : \"\"}\r\n          <p class=\"what\">${this.description}</p>\r\n        </div>\r\n      </li>\r\n    `;\r\n  }\r\n}\r\n","import { CalendarEvent } from \"./CalendarEvent\";\r\n\r\nexport class EventPopup {\r\n  private popupEl: HTMLElement|undefined;\r\n\r\n  constructor(private events: CalendarEvent[]) {}\r\n\r\n  start() {\r\n    if (!this.popupEl) {\r\n      this.popupEl = document.createElement(\"div\");\r\n      this.popupEl.className = \"popup\";\r\n    }\r\n    \r\n    addEventListener(\"mousemove\", (e: MouseEvent) => {\r\n      this.popupEl.classList.remove(\"visible\");\r\n      this.popupEl.remove();\r\n\r\n      const eventEl = (e.target as HTMLElement).closest(\"#month-grid .event, #week-grid .event\") as HTMLElement;\r\n      if (!eventEl) {\r\n        return;\r\n      }\r\n\r\n      const id = eventEl.id;\r\n\r\n      if (!id || !id.startsWith(\"id-\")) {\r\n        return;\r\n      }\r\n\r\n      let eventId = id.replace(\"id-\", \"\");\r\n\r\n      const event = this.events.find(event => event.id === eventId);      \r\n      if (!event) {\r\n        console.log(\"No event found for id\", eventId);\r\n        return;\r\n      }\r\n\r\n      const eventIndexInGrid = parseInt(eventEl.dataset.indexInGrid!);\r\n\r\n      this.popupEl.style.setProperty(\"--event-color\", event.color);\r\n      this.popupEl.classList.toggle(\"visible\", true);\r\n      this.popupEl.classList.toggle(\"unconfirmed\", !event.rsvp);\r\n\r\n      let when = \"\";\r\n      \r\n      if (!event.multiDays) {\r\n        when = event.date.toDateString();\r\n        if (event.time) {\r\n          when += `, at ${event.time}`;\r\n        }\r\n        if (event.repeat) {\r\n          when += `, ${event.repeat}`;\r\n        }\r\n        if (event.duration) {\r\n          when += ` (${event.duration} minutes)`;\r\n        }\r\n      } else {\r\n        when = `From ${event.multiDays.start.toDateString()} to ${event.multiDays.end.toDateString()}`;\r\n      }\r\n\r\n      this.popupEl.innerHTML = `\r\n        <h2>${event.title}</h2>\r\n        <p class=\"when\">${when}</p>\r\n        ${event.location ? `<p class=\"where\">${event.location}</p>` : \"\"}\r\n        <p class=\"what\">${event.description}</p>\r\n      `;\r\n\r\n      document.body.appendChild(this.popupEl);\r\n\r\n      // Get the coordinates of the event element.\r\n      const rect = eventEl.getBoundingClientRect();\r\n      const top = rect.top;\r\n      const right = rect.right;\r\n      const left = rect.left;\r\n      const width = rect.width;\r\n\r\n      if (eventIndexInGrid >= 28) {\r\n        // The event is in the last 2 rows. Show the popup above it.\r\n        this.popupEl.style.top = `${top - this.popupEl.clientHeight}px`;\r\n      } else {\r\n        this.popupEl.style.top = `${top}px`;\r\n      }\r\n\r\n      if (eventIndexInGrid === 6 || eventIndexInGrid === 13 || eventIndexInGrid === 20 || eventIndexInGrid === 27 || eventIndexInGrid === 34) {\r\n        // The event is last in a row, so we want to show the popup to the left of it.\r\n        this.popupEl.style.left = `${left - width}px`;\r\n        this.popupEl.classList.toggle(\"left\", true);\r\n      } else {\r\n        this.popupEl.style.left = `${right}px`;\r\n        this.popupEl.classList.toggle(\"left\", false);\r\n      }\r\n    });\r\n  }\r\n}\r\n","export class Store {\r\n  async getStoredPrefs(): Promise<{mode: string, initDate: Date|undefined}> {\r\n    // Fake some random time (up to 250ms) to simulate an API call.\r\n    await new Promise(resolve => setTimeout(resolve, Math.random() * 250));\r\n\r\n    const mode = localStorage.getItem('slow-cal-mode');\r\n    const initDate = localStorage.getItem('slow-cal-initDate');\r\n\r\n    return {\r\n      mode,\r\n      initDate: initDate ? new Date(initDate) : undefined\r\n    };\r\n  }\r\n\r\n  set mode(mode: string) {\r\n    console.log(\"Storing mode prefs\");\r\n    localStorage.setItem('slow-cal-mode', mode);\r\n  }\r\n\r\n  set initDate(date: Date) {\r\n    console.log(\"Storing date prefs\");\r\n    localStorage.setItem('slow-cal-initDate', date.toISOString());\r\n  }\r\n}\r\n","import { AppUI } from \"./AppUI\";\r\nimport { getAllEvents } from \"./events-factory\";\r\nimport { EventPopup } from \"./EventPopup\";\r\nimport { Store } from \"./Store\";\r\n\r\naddEventListener(\"DOMContentLoaded\", async () => {\r\n  const store = new Store();\r\n  const prefs = await store.getStoredPrefs();\r\n\r\n  // This is what the calendar will be centered around on load.\r\n  const initDate = prefs.initDate || new Date();\r\n  const initMode = prefs.mode || \"month\";\r\n\r\n  const appEl = document.getElementById('app');\r\n  const appUI = new AppUI(appEl!, initDate, initMode, []);\r\n\r\n  const events = await getAllEvents()\r\n\r\n  console.log(\"Refreshing calendar with events\");\r\n  appUI.events = events;\r\n\r\n  console.log(\"Initializing the popup util\");\r\n  const popup = new EventPopup(events);\r\n  popup.start();\r\n\r\n  // @ts-ignore\r\n  appUI.addEventListener(\"mode-changed\", (e: CustomEvent) => {\r\n    store.mode = e.detail;\r\n  });\r\n\r\n  // @ts-ignore\r\n  appUI.addEventListener(\"date-changed\", (e: CustomEvent) => {\r\n    store.initDate = e.detail;\r\n  });\r\n});\r\n","import { CalendarEvent } from \"./CalendarEvent\";\r\n\r\nasync function fetchData() {\r\n  // Simulate a slower API endpoint call.\r\n  console.log(\"Fetching calendar data ...\");\r\n  await new Promise(resolve => setTimeout(resolve, Math.random() * 500));\r\n  const response = await fetch('./data.json');\r\n  return await response.json();\r\n}\r\n\r\nfunction createOneOrMultipleEventsFromData(eventData: any): CalendarEvent[] {  \r\n  const dates: Date[] = [new Date(eventData.startDate)];\r\n  let multiDays: {start: Date, end: Date}|null = null;\r\n\r\n  if (eventData.repeat) {\r\n    const startDate = new Date(eventData.startDate);\r\n\r\n    // If this event is repeated, it must have an end date.\r\n    const endDate = eventData.endDate\r\n      // Either the end date is specified.\r\n      ? new Date(eventData.endDate)\r\n      // Or it's 5 year after the start date.\r\n      : new Date(startDate.getTime() + 5 * 365 * 24 * 60 * 60 * 1000);\r\n\r\n    if (eventData.repeat === \"daily\") {\r\n      console.log(\"Expanding daily event ...\");\r\n      const days = (endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24);\r\n      for (let i = 1; i <= days; i++) {\r\n        const date = new Date(startDate.getTime() + i * 24 * 60 * 60 * 1000);\r\n        // If date is on a week end, skip it.\r\n        if (date.getDay() === 0 || date.getDay() === 6) {\r\n          continue;\r\n        }\r\n\r\n        dates.push(date);\r\n      }\r\n    } else if (eventData.repeat === \"weekly\") {\r\n      console.log(\"Expanding weekly event ...\");\r\n      const weeks = (endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24 * 7);\r\n      for (let i = 1; i <= weeks; i++) {\r\n        const date = new Date(startDate.getTime() + i * 7 * 24 * 60 * 60 * 1000);\r\n        dates.push(date);\r\n      }\r\n    } else if (eventData.repeat === \"monthly\") {\r\n      console.log(\"Expanding monthly event ...\");\r\n      const months = (endDate.getFullYear() - startDate.getFullYear()) * 12 + (endDate.getMonth() - startDate.getMonth());\r\n      for (let i = 1; i <= months; i++) {\r\n        const date = new Date(startDate.getTime());\r\n        date.setMonth(date.getMonth() + i);\r\n        dates.push(date);\r\n      }\r\n    } else if (eventData.repeat === \"yearly\") {\r\n      console.log(\"Expanding yearly event ...\");\r\n      const years = endDate.getFullYear() - startDate.getFullYear();\r\n      for (let i = 1; i <= years; i++) {\r\n        const date = new Date(startDate.getTime());\r\n        date.setFullYear(date.getFullYear() + i);\r\n        dates.push(date);\r\n      }\r\n    }\r\n  } else if (eventData.endDate) {\r\n    // If this event has a start and end date, but isn't repeated.\r\n    const startDate = new Date(eventData.startDate);\r\n    const endDate = new Date(eventData.endDate);\r\n    const days = (endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24);\r\n    for (let i = 1; i <= days; i++) {\r\n      const date = new Date(startDate.getTime() + i * 24 * 60 * 60 * 1000);\r\n      dates.push(date);\r\n    }\r\n    multiDays = {start: startDate, end: endDate};\r\n  }\r\n\r\n\r\n  return dates.map((date, index) => {\r\n    const id = eventData.id + (index > 0 ? `-${index}` : \"\");\r\n    const event = new CalendarEvent(eventData.title, id, date);\r\n\r\n    event.multiDays = multiDays;\r\n\r\n    event.color = eventData.color;\r\n    event.time = eventData.startTime;\r\n    event.duration = eventData.duration;\r\n    event.repeat = eventData.repeat;\r\n    event.location = eventData.location;\r\n    event.description = eventData.description;\r\n    event.rsvp = eventData.rsvp;\r\n\r\n    return event;\r\n  });\r\n}\r\n\r\nexport async function getAllEvents(): Promise<CalendarEvent[]> {\r\n  const data = await fetchData();\r\n\r\n  console.log(\"Processing events ...\");\r\n  return data.events.map((event: any) => {\r\n    return createOneOrMultipleEventsFromData(event);\r\n  }).flat();\r\n}\r\n"],"names":["assertType","condition","message","args","TypeError","format","i","replace","anyToString","x","String","Object","prototype","toString","call","Global","window","self","global","globalThis","undefined","currentWarnHandler","Warning","constructor","code","this","warn","_a","stack","Error","console","_b","InitEventWasCalledWhileDispatching","FalsyWasAssignedToCancelBubble","TruthyWasAssignedToReturnValue","NonCancelableEventWasCanceled","CanceledInPassiveListener","EventListenerWasDuplicated","OptionWasIgnored","InvalidEventListener","Event","NONE","CAPTURING_PHASE","AT_TARGET","BUBBLING_PHASE","type","eventInitDict","defineProperty","value","enumerable","opts","internalDataMap","set","bubbles","Boolean","cancelable","composed","target","currentTarget","stopPropagationFlag","stopImmediatePropagationFlag","canceledFlag","inPassiveListenerFlag","dispatchFlag","timeStamp","Date","now","$","srcElement","composedPath","eventPhase","stopPropagation","cancelBubble","stopImmediatePropagation","data","returnValue","setCancelFlag","preventDefault","defaultPrevented","isTrusted","initEvent","WeakMap","event","name","retv","get","keys","getOwnPropertyNames","length","DOMException","setPrototypeOf","ErrorCodeMap","INDEX_SIZE_ERR","DOMSTRING_SIZE_ERR","HIERARCHY_REQUEST_ERR","WRONG_DOCUMENT_ERR","INVALID_CHARACTER_ERR","NO_DATA_ALLOWED_ERR","NO_MODIFICATION_ALLOWED_ERR","NOT_FOUND_ERR","NOT_SUPPORTED_ERR","INUSE_ATTRIBUTE_ERR","INVALID_STATE_ERR","SYNTAX_ERR","INVALID_MODIFICATION_ERR","NAMESPACE_ERR","INVALID_ACCESS_ERR","VALIDATION_ERR","TYPE_MISMATCH_ERR","SECURITY_ERR","NETWORK_ERR","ABORT_ERR","URL_MISMATCH_ERR","QUOTA_EXCEEDED_ERR","TIMEOUT_ERR","INVALID_NODE_TYPE_ERR","DATA_CLONE_ERR","defineErrorCodeProperties","obj","key","configurable","EventWrapper","wrap","getWrapperClassOf","super","internalDataMap$1","original","defineRedirectDescriptor","$$1","wrapperClassCache","originalEvent","getPrototypeOf","wrapper","BaseEventWrapper","originalPrototype","CustomEventWrapper","defineWrapper","d","getOwnPropertyDescriptor","bind","isCapture","listener","flags","isPassive","isOnce","isRemoved","invokeCallback","callback","handleEvent","thrownError","maybeError","error","dispatchEvent","ErrorEvent","process","emit","reportError","findIndexOfListener","listeners","capture","removeListener","list","index","removeListenerAt","disableCow","setRemoved","signal","removeEventListener","signalListener","cow","filter","_","splice","EventTarget","internalDataMap$2","create","addEventListener","type0","callback0","options0","listenerMap","$$2","once","passive","options","assertCallback","normalizeAddOptions","aborted","attrCallback","attrListener","ensureListenerList","createListener","push","addListener","warnDuplicate","normalizeOptions","e","eventData","msg","captureStackTrace","defineProperties","keys$1","getEventsForDay","events","date","getFullYear","getMonth","getDate","sort","a","b","time","parseInt","multiDays","DAYS","MonthGrid","el","_date","rootEl","_events","render","innerHTML","month","year","firstDay","lastDay","firstWeekday","getDay","lastWeekday","daysInMonth","daysInLastMonth","days","indexInGrid","html","renderDay","insertAdjacentHTML","lastElementChild","CustomEvent","detail","updateNowMarker","dayNumber","className","toDateString","map","asOneLineHTML","join","ratio","getHours","getMinutes","dayEl","querySelector","markerHeight","offsetHeight","style","setProperty","setTimeout","MONTHS","Toolbar","mode","selectedMode","formatMonthYear","log","Sidebar","ul","document","createElement","appendChild","asFullHTML","WeekGrid","day","asMediumLengthHTML","AppUI","toolbarEl","id","toolbar","newDate","monthGridEl","display","weekGridEl","monthGrid","weekGrid","sidebarEl","sidebar","CalendarEvent","title","rsvp","color","description","when","start","end","repeat","duration","location","EventPopup","popupEl","classList","remove","eventEl","closest","startsWith","eventId","find","eventIndexInGrid","dataset","toggle","body","rect","getBoundingClientRect","top","right","left","width","clientHeight","Store","getStoredPrefs","Promise","resolve","Math","random","localStorage","getItem","initDate","setItem","toISOString","store","prefs","initMode","appEl","getElementById","appUI","response","fetch","json","fetchData","dates","startDate","endDate","getTime","weeks","months","setMonth","years","setFullYear","startTime","createOneOrMultipleEventsFromData","flat","getAllEvents"],"sourceRoot":""} \ No newline at end of file diff --git a/slow-calendar/public/clock.png b/slow-calendar/public/clock.png new file mode 100644 index 0000000000000000000000000000000000000000..b3ecbcd505a646c94715259fd0e0778579facf3e GIT binary patch literal 1097 zcmV-P1h)H$P)Px#1ZP1_K>z@;j|==^1poj532;bRa{vGxhX4Q_hXIe}@nrx21KvqQK~zXfomERr zRA&^vY;1s`;S~trKtc!u3mOPZSqKT?(abO|Y+^TR6N|t?stF6bk~Bi)RcTCZfZAwc zE#PBepc1>Vv4S-<-B@v_U{r@@VeD|LJobziN z8$SFOx3)r1;QH08MJz1*h2`bde|MwsdN;TEDsB_-czoF042Yx4<;Kv^SJ=P57FMej z+1c61%F04+ZZ0Y+K0rrD2WDsI#5NLv?&~$gA`JpGpkXi=!kIJO$jHb*Nl6JhJ3qt3 z#3bhCe#7I(f8frY8T9q_qqepVsi~=GXlTIGr}H8dACHU)FgUTe=t6aMHS+TEaP8WU z2!+D3PlQ5|*w-mMdGaS5ju!0M^B%5V{XzD&2{1SVx!tez!1<`FtHb*G-(u|Z`Nc4z zf7V;4#&xufWO!=EP%gD^k)SK`jG^Qm#zX0Ro69&UX62rq^OO-er4m3A^sIwVOO-(p? z@|2{OU&VMVZL75qw{K5D;~xbD1$tABOAxc*WFWAmck45pKK(h4A8$i@d%MmKw6(QM zSNr_|gTUUsR!mJzGetmoc{%#~2X+)-WyKG_L?}DjxLmK1o}P|JkA9CH^1*{2!Rd4w z4F4~QyLW$v#gc}Vm9?k437PqJ2N zHe5KIq7-pD59k9c1!vFp==bl6IG7=<-$gSsGcx9La&mC*-Y>GRVj2X<_*%bw`3j1Q zi}l0LMoUYJ6acI3`t=*Qc=20lHk+*sX=(4HwY3#XOUvSySRe)gMRp{?uDp5k77iai zBB?7XDw4~;tgKAVi?iwF%LTFJi + + + + Slow Calendar + + + + +
    + + + \ No newline at end of file diff --git a/slow-calendar/public/info.png b/slow-calendar/public/info.png new file mode 100644 index 0000000000000000000000000000000000000000..3f51295c17f5793ab8f8efda91a2c27b0259cbd6 GIT binary patch literal 1117 zcmV-j1fu(iP)Px#1ZP1_K>z@;j|==^1poj532;bRa{vGxhX4Q_hXIe}@nrx21M*2kK~zXfl~qkl zlxGm0UG~pi8oCz_1Qtt?;<6mJ2Z)4_*n>@=;UXMtIM{@1J=w&bAh9;GtqCWUl1Oc0 zsT$pal#tk)@n_XWixSO7pd3W}SQEp-{_jrTdDwT|rK_<|@@Br7dFT1&{h4`9o0~DF zx4>jF(XL$%CG&9& zx$S{iJ{ugCSAhVrFrP6%y;w7%D^AuLGu@Qw`bAo(p zD~^D}hrhx^l_f7P{{ve*p2Bj~;XzL%O7#~n*0{aB9dQQ__Q9sk;0?Jz_E=}Uyu1RN z-b0ye7!mC6>gpO+BX;o%UcNNAzP=vulas%}2673^X0yC+Uk9A$&P~8sd^m!!v7g{S zd2$50_URv7xe|om=kuv8PGMz~&F1*QgTLX}z563%B}>=AGDHuDpYhPpFh6>=_aVP`Z#&-W^+B4|DN{#BhuTLw&CSix+M0py{{z{!*RMBd zYHA8U@9OGO-IB?iOeV?WsiVHWessM+V1z6dSx{+)sZeDZ{I;h7DGCpWT+dRjAO^Xg-}8Rf43J70f+x&gTN@(^pt jgRjm($!y+I4n+R~TVT(HCW$G|00000NkvXXu0mjf+&m!X literal 0 HcmV?d00001 diff --git a/slow-calendar/public/pin.png b/slow-calendar/public/pin.png new file mode 100644 index 0000000000000000000000000000000000000000..4e72af1cee372dfb4d40ad51b13a9352bd51adba GIT binary patch literal 1193 zcmV;a1XlZrP)X1^@s6-qmI800001b5ch_0Itp) z=>Px#1ZP1_K>z@;j|==^1poj532;bRa{vGxhX4Q_hXIe}@nrx21U^YbK~zXfeN|gb z(^nWjv=m0$QEs7xfHDFLagUA=h+~onju+g6W0JT)vzA%~I6yflSh{a;b=p>kw#per>bRfp%%fnP!TFQgHQc6nRBCGXVip7XlS7S0h z03IZ%e!}4h#p6U>U0rK|EiEl{_3BULaLmx7M{_haHBH0AKTvaX^O{dnQxkbS-VMYc z4Fp2Su(LBkxN~O-jgS8U9fJppM&)N6rlJbpY+27>{zwDgb6I7)&Iyx>gO*473`9xWP;k<+|6a#@E?cV(^^n3QaM;42P_U)@jzr4Jh zT&`t`L=vZEa`G1Rd-v91Flc64-tl&>sfWaMX(Q-e*3tPOPO(g5trudnYb*v_2! z0)4?*J$?ENS;?iKxw!?5t^zoV)2Ba|SMy^!apDx&?NiY626L=B}AtINcEL^gwC)YazMaoo|P9q6+Sq+FeRKEFJFZ^L15aA?h8J5jS39sLFV za?|Mf^M8>!+d-;s1{V=_;$`5CVXQA{>wRZt?o&=q4*W+(Mxj%_AXUP+aN!#`$|EHM zXQ5Nhig0mpiOR~#;MddB1064i;@Lo|D$>c5rx3t5?e5*bp<_E^VZlii6&0`_InoZD zs$5Dokg~7?qr1Bsf$|BVv9aHvyM22~mc|{>x3;!o6Zx8M=Ga2YqMntrXTJmi%Yg$Q zlDsYWKX&X>*qLzinzL+6%AztJ7#KvMU^0WjNaxQd%3I>ZY#;l|Cz_#N)+bM%38&L3 z3JVKGNy!fRvPMKao+v!q;kBd+X=!Ofr_;)JK91t?dI<49e7_;TPqO`a00000NkvXX Hu0mjfbz4SQ literal 0 HcmV?d00001 diff --git a/slow-calendar/src/AppUI.ts b/slow-calendar/src/AppUI.ts new file mode 100644 index 0000000..9eb83f9 --- /dev/null +++ b/slow-calendar/src/AppUI.ts @@ -0,0 +1,163 @@ +import { EventTarget, Event } from "event-target-shim"; +import { CalendarEvent } from "./CalendarEvent"; +import { MonthGrid } from "./MonthGrid"; +import { Toolbar } from "./Toolbar"; +import { Sidebar } from "./Sidebar"; +import { getEventsForDay } from "./utils"; +import { WeekGrid } from "./WeekGrid"; + +export class AppUI extends EventTarget { + private rootEl: HTMLElement; + + private monthGridEl: HTMLElement; + public monthGrid: MonthGrid; + + private weekGridEl: HTMLElement; + public weekGrid: WeekGrid; + + private toolbarEl: HTMLElement; + private toolbar: Toolbar; + + private sidebarEl: HTMLElement; + private sidebar: Sidebar; + + private _date: Date; + private _events: CalendarEvent[]; + + private mode: string = "month"; + + constructor(el: HTMLElement, date: Date, mode: string, events: CalendarEvent[]) { + super(); + + this.rootEl = el; + this._date = date; + this._events = events; + this.mode = mode; + + this.render(); + } + + render() { + this.rootEl.innerHTML = ''; + + // Create an element for the toolbar. + this.toolbarEl = document.createElement('div'); + this.toolbarEl.id = 'toolbar'; + this.rootEl.appendChild(this.toolbarEl); + + // Init the toolbar component in it. + this.toolbar = new Toolbar(this.toolbarEl, this._date, this.mode); + + this.toolbar.addEventListener('prev', () => { + const newDate = this.mode === "month" + ? new Date(this._date.getFullYear(), this._date.getMonth() - 1, 1) + : new Date(this._date.getFullYear(), this._date.getMonth(), this._date.getDate() - 7); + this.date = newDate; + + // @ts-ignore + this.dispatchEvent(new CustomEvent('date-changed', {detail: this._date})); + }); + + this.toolbar.addEventListener('next', () => { + const newDate = this.mode === "month" + ? new Date(this._date.getFullYear(), this._date.getMonth() + 1, 1) + : new Date(this._date.getFullYear(), this._date.getMonth(), this._date.getDate() + 7); + this.date = newDate; + + // @ts-ignore + this.dispatchEvent(new CustomEvent('date-changed', {detail: this._date})); + }); + + this.toolbar.addEventListener('today', () => { + this.date = new Date(); + + // @ts-ignore + this.dispatchEvent(new CustomEvent('date-changed', {detail: this._date})); + }); + + this.toolbar.addEventListener('month-view', () => { + this.mode = "month"; + this.monthGridEl.style.display = 'grid'; + this.weekGridEl.style.display = 'none'; + + // @ts-ignore + this.dispatchEvent(new CustomEvent('mode-changed', {detail: this.mode})); + }); + + this.toolbar.addEventListener('week-view', () => { + this.mode = "week"; + this.monthGridEl.style.display = 'none'; + this.weekGridEl.style.display = 'grid'; + + // @ts-ignore + this.dispatchEvent(new CustomEvent('mode-changed', {detail: this.mode})); + }); + + // Create an element for the month grid. + this.monthGridEl = document.createElement('div'); + this.monthGridEl.id = 'month-grid'; + this.rootEl.appendChild(this.monthGridEl); + + // Init the month grid component in it. + this.monthGrid = new MonthGrid(this.monthGridEl, this._date, this._events); + + // Create an element for the week grid. + this.weekGridEl = document.createElement('div'); + this.weekGridEl.id = 'week-grid'; + this.rootEl.appendChild(this.weekGridEl); + + // Init the week grid component in it. + this.weekGrid = new WeekGrid(this.weekGridEl, this._date, this._events); + + // Create an element for the sidebar. + this.sidebarEl = document.createElement('div'); + this.sidebarEl.id = 'sidebar'; + this.rootEl.appendChild(this.sidebarEl); + + // Init the sidebar component in it. + this.sidebar = new Sidebar(this.sidebarEl, this._date, this._events); + + // Listen for day-clicked events from the month and week grids to show the full + // list in the sidebar. + // @ts-ignore + this.monthGrid.addEventListener('day-clicked', (e: CustomEvent) => { + const date = e.detail; + const events = getEventsForDay(this._events, date); + this.sidebar.events = events; + }); + // @ts-ignore + this.weekGrid.addEventListener('day-clicked', (e: CustomEvent) => { + const date = e.detail; + const events = getEventsForDay(this._events, date); + this.sidebar.events = events; + }); + + // Start with today. + this.sidebar.events = getEventsForDay(this._events, new Date()); + + // Show/hide the right mode + if (this.mode === "week") { + this.monthGridEl.style.display = 'none'; + this.weekGridEl.style.display = 'grid'; + } else { + this.monthGridEl.style.display = 'grid'; + this.weekGridEl.style.display = 'none'; + } + } + + set date(date: Date) { + console.log("Setting date to", date); + this._date = date; + this.monthGrid.date = date; + this.weekGrid.date = date; + this.toolbar.date = date; + this.sidebar.events = getEventsForDay(this._events, this._date); + } + + set events(events: CalendarEvent[]) { + this._events = events; + this.monthGrid.events = events; + this.weekGrid.events = events; + this.sidebar.events = getEventsForDay(this._events, this._date); + } +} diff --git a/slow-calendar/src/CalendarEvent.ts b/slow-calendar/src/CalendarEvent.ts new file mode 100644 index 0000000..0b66fb5 --- /dev/null +++ b/slow-calendar/src/CalendarEvent.ts @@ -0,0 +1,71 @@ +export class CalendarEvent { + title: string; + id: string; + date: Date; + time: string|undefined; + duration: number|undefined; + repeat: string|undefined; + description: string|undefined; + color: string|undefined; + location: string|undefined; + multiDays: {start: Date, end: Date}|null; + rsvp: boolean = false; + + constructor(title: string, id: string, date: Date) { + this.title = title; + this.id = id; + this.date = date; + } + + asOneLineHTML(indexInGrid: number|undefined = undefined): string { + const time = this.time ? `${this.time}` : ''; + + return ` +
  • +
    ${time} ${this.title}
    +
  • + `; + } + + asMediumLengthHTML(): string { + const time = this.time ? `${this.time}` : ''; + + return ` +
  • +
    ${time}
    +

    ${this.title}

    +

    ${this.description}

    +
  • + `; + } + + asFullHTML(): string { + let when = ""; + + if (!this.multiDays) { + when = this.date.toDateString(); + if (this.time) { + when += `, at ${this.time}`; + } + if (this.repeat) { + when += `, ${this.repeat}`; + } + if (this.duration) { + when += ` (${this.duration} minutes)`; + } + } else { + when = `From ${this.multiDays.start.toDateString()} to ${this.multiDays.end.toDateString()}`; + } + + return ` +
  • +
    +

    ${this.title}

    +

    ${when}

    + ${this.location ? `

    ${this.location}

    ` : ""} +

    ${this.description}

    +
    +
  • + `; + } +} diff --git a/slow-calendar/src/EventPopup.ts b/slow-calendar/src/EventPopup.ts new file mode 100644 index 0000000..1e4b60c --- /dev/null +++ b/slow-calendar/src/EventPopup.ts @@ -0,0 +1,93 @@ +import { CalendarEvent } from "./CalendarEvent"; + +export class EventPopup { + private popupEl: HTMLElement|undefined; + + constructor(private events: CalendarEvent[]) {} + + start() { + if (!this.popupEl) { + this.popupEl = document.createElement("div"); + this.popupEl.className = "popup"; + } + + addEventListener("mousemove", (e: MouseEvent) => { + this.popupEl.classList.remove("visible"); + this.popupEl.remove(); + + const eventEl = (e.target as HTMLElement).closest("#month-grid .event, #week-grid .event") as HTMLElement; + if (!eventEl) { + return; + } + + const id = eventEl.id; + + if (!id || !id.startsWith("id-")) { + return; + } + + let eventId = id.replace("id-", ""); + + const event = this.events.find(event => event.id === eventId); + if (!event) { + console.log("No event found for id", eventId); + return; + } + + const eventIndexInGrid = parseInt(eventEl.dataset.indexInGrid!); + + this.popupEl.style.setProperty("--event-color", event.color); + this.popupEl.classList.toggle("visible", true); + this.popupEl.classList.toggle("unconfirmed", !event.rsvp); + + let when = ""; + + if (!event.multiDays) { + when = event.date.toDateString(); + if (event.time) { + when += `, at ${event.time}`; + } + if (event.repeat) { + when += `, ${event.repeat}`; + } + if (event.duration) { + when += ` (${event.duration} minutes)`; + } + } else { + when = `From ${event.multiDays.start.toDateString()} to ${event.multiDays.end.toDateString()}`; + } + + this.popupEl.innerHTML = ` +

    ${event.title}

    +

    ${when}

    + ${event.location ? `

    ${event.location}

    ` : ""} +

    ${event.description}

    + `; + + document.body.appendChild(this.popupEl); + + // Get the coordinates of the event element. + const rect = eventEl.getBoundingClientRect(); + const top = rect.top; + const right = rect.right; + const left = rect.left; + const width = rect.width; + + if (eventIndexInGrid >= 28) { + // The event is in the last 2 rows. Show the popup above it. + this.popupEl.style.top = `${top - this.popupEl.clientHeight}px`; + } else { + this.popupEl.style.top = `${top}px`; + } + + if (eventIndexInGrid === 6 || eventIndexInGrid === 13 || eventIndexInGrid === 20 || eventIndexInGrid === 27 || eventIndexInGrid === 34) { + // The event is last in a row, so we want to show the popup to the left of it. + this.popupEl.style.left = `${left - width}px`; + this.popupEl.classList.toggle("left", true); + } else { + this.popupEl.style.left = `${right}px`; + this.popupEl.classList.toggle("left", false); + } + }); + } +} diff --git a/slow-calendar/src/MonthGrid.ts b/slow-calendar/src/MonthGrid.ts new file mode 100644 index 0000000..e118a8a --- /dev/null +++ b/slow-calendar/src/MonthGrid.ts @@ -0,0 +1,117 @@ +import { EventTarget, Event } from "event-target-shim"; +import { CalendarEvent } from "./CalendarEvent"; +import { getEventsForDay } from "./utils"; + +const DAYS = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday']; + +export class MonthGrid extends EventTarget { + private rootEl: HTMLElement; + private _date: Date = new Date(); + private _events: CalendarEvent[]; + + constructor(private el: HTMLElement, date: Date, events: CalendarEvent[]) { + super(); + + this.rootEl = el; + this._date = date; + this._events = events; + + this.render(); + } + + render() { + this.rootEl.innerHTML = ''; + + const month = this._date.getMonth(); + const year = this._date.getFullYear(); + + const firstDay = new Date(year, month, 1); + const lastDay = new Date(year, month + 1, 0); + + const firstWeekday = firstDay.getDay(); + const lastWeekday = lastDay.getDay(); + const daysInMonth = lastDay.getDate(); + const daysInLastMonth = new Date(year, month, 0).getDate(); + const days: {date: Date, html: string}[] = []; + + let indexInGrid = 0; + + // Days from last month + for (let i = daysInLastMonth - firstWeekday + 1; i <= daysInLastMonth; i++) { + const date = new Date(year, month - 1, i); + days.push({date, html: this.renderDay(date, i, indexInGrid, 'prev-month')}); + indexInGrid++; + } + + // Days from this month + for (let i = 1; i <= daysInMonth; i++) { + const date = new Date(year, month, i); + days.push({date, html: this.renderDay(date, i, indexInGrid)}); + indexInGrid++; + } + + // Days from next month + for (let i = 1; i <= 7 - lastWeekday - 1; i++) { + const date = new Date(year, month + 1, i); + days.push({date, html: this.renderDay(date, i, indexInGrid, 'next-month')}); + indexInGrid++; + } + + for (const {date, html} of days) { + this.rootEl.insertAdjacentHTML('beforeend', html); + const dayEl = this.rootEl.lastElementChild as HTMLElement; + + dayEl.addEventListener('click', (e: MouseEvent) => { + // @ts-ignore + this.dispatchEvent(new CustomEvent('day-clicked', {detail: date})); + }); + } + + this.updateNowMarker(); + } + + renderDay(date: Date, dayNumber: number, indexInGrid: number, className: string = ''): string { + const events = getEventsForDay(this._events, date); + const isToday = date.toDateString() === new Date().toDateString(); + + return ` +
    +
    + ${DAYS[date.getDay()]} + ${dayNumber} +
    +
      + ${events.map(event => event.asOneLineHTML(indexInGrid)).join('')} +
    +
    + `; + } + + updateNowMarker() { + const now = new Date(); + const time = now.getHours() * 100 + now.getMinutes(); + const ratio = time * 100 / 2400; + + // Forcing a sync reflow here by querying the offsetHeight of the day cell. + // Could just set the % value directly instead. + const dayEl = this.rootEl.querySelector('.day.today') as HTMLElement; + if (dayEl) { + const pxHeight = dayEl.offsetHeight; + const markerHeight = pxHeight * ratio / 100; + + this.rootEl.style.setProperty('--now-marker', `${markerHeight}px`); + } + + setTimeout(() => this.updateNowMarker(), 1000); + } + + set date(date: Date) { + this._date = date; + this.render(); + } + + set events(events: CalendarEvent[]) { + this._events = events; + this.render(); + } +} diff --git a/slow-calendar/src/Sidebar.ts b/slow-calendar/src/Sidebar.ts new file mode 100644 index 0000000..b33a5e6 --- /dev/null +++ b/slow-calendar/src/Sidebar.ts @@ -0,0 +1,41 @@ +import { CalendarEvent } from "./CalendarEvent"; + +export class Sidebar { + private rootEl: HTMLElement; + private _events: CalendarEvent[] = []; + private _date: Date = new Date(); + + constructor(private el: HTMLElement, date: Date, events: CalendarEvent[]) { + this.rootEl = el; + this._date = date; + this._events = events; + + this.render(); + } + + render() { + this.rootEl.innerHTML = ` +

    Click a day to view events

    + `; + + if (!this._events.length) { + return; + } + + this.rootEl.innerHTML = ''; + + const ul = document.createElement('ul'); + ul.className = 'events'; + this.rootEl.appendChild(ul); + + for (const event of this._events) { + ul.innerHTML += event.asFullHTML(); + } + } + + set events(events: CalendarEvent[]) { + this._events = events; + this.render(); + } +} +`` \ No newline at end of file diff --git a/slow-calendar/src/Store.ts b/slow-calendar/src/Store.ts new file mode 100644 index 0000000..155a1cd --- /dev/null +++ b/slow-calendar/src/Store.ts @@ -0,0 +1,24 @@ +export class Store { + async getStoredPrefs(): Promise<{mode: string, initDate: Date|undefined}> { + // Fake some random time (up to 250ms) to simulate an API call. + await new Promise(resolve => setTimeout(resolve, Math.random() * 250)); + + const mode = localStorage.getItem('slow-cal-mode'); + const initDate = localStorage.getItem('slow-cal-initDate'); + + return { + mode, + initDate: initDate ? new Date(initDate) : undefined + }; + } + + set mode(mode: string) { + console.log("Storing mode prefs"); + localStorage.setItem('slow-cal-mode', mode); + } + + set initDate(date: Date) { + console.log("Storing date prefs"); + localStorage.setItem('slow-cal-initDate', date.toISOString()); + } +} diff --git a/slow-calendar/src/Toolbar.ts b/slow-calendar/src/Toolbar.ts new file mode 100644 index 0000000..191f134 --- /dev/null +++ b/slow-calendar/src/Toolbar.ts @@ -0,0 +1,73 @@ +import { EventTarget, Event } from "event-target-shim"; + +const MONTHS = [ + 'January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', + 'September', 'October', 'November','December' +]; + +export class Toolbar extends EventTarget { + private rootEl: HTMLElement; + private _date: Date; + private selectedMode: string = "month"; + + constructor(private el: HTMLElement, date: Date, mode: string) { + super(); + + this.rootEl = el; + this._date = date; + this.selectedMode = mode; + + this.render(); + } + + formatMonthYear() { + const month = this._date.getMonth(); + const year = this._date.getFullYear(); + return `${MONTHS[month]} ${year}`; + } + + render() { + this.rootEl.innerHTML = ` + + + + ${this.formatMonthYear()} + + + + + `; + + this.rootEl.querySelector('#prev-month').addEventListener('click', () => { + console.log(`Prev ${this.selectedMode} clicked`); + this.dispatchEvent(new Event('prev')); + }); + + this.rootEl.querySelector('#next-month').addEventListener('click', () => { + console.log(`Next ${this.selectedMode} clicked`); + this.dispatchEvent(new Event('next')); + }); + + this.rootEl.querySelector('#today').addEventListener('click', () => { + console.log("Today clicked"); + this.dispatchEvent(new Event('today')); + }); + + this.rootEl.querySelector('#month-view').addEventListener('click', () => { + console.log("Month view clicked"); + this.dispatchEvent(new Event('month-view')); + this.selectedMode = "month"; + }); + + this.rootEl.querySelector('#week-view').addEventListener('click', () => { + console.log("Week view clicked"); + this.dispatchEvent(new Event('week-view')); + this.selectedMode = "week"; + }); + } + + set date(date: Date) { + this._date = date; + this.render(); + } +} diff --git a/slow-calendar/src/WeekGrid.ts b/slow-calendar/src/WeekGrid.ts new file mode 100644 index 0000000..9838c83 --- /dev/null +++ b/slow-calendar/src/WeekGrid.ts @@ -0,0 +1,93 @@ +import { EventTarget, Event } from "event-target-shim"; +import { CalendarEvent } from "./CalendarEvent"; +import { getEventsForDay } from "./utils"; + +const DAYS = ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday']; + +export class WeekGrid extends EventTarget { + private rootEl: HTMLElement; + private _date: Date = new Date(); + private _events: CalendarEvent[]; + + constructor(private el: HTMLElement, date: Date, events: CalendarEvent[]) { + super(); + + this.rootEl = el; + this._date = date; + this._events = events; + + this.render(); + } + + render() { + this.rootEl.innerHTML = ''; + + // Find the dates that make the week around this._date. + const month = this._date.getMonth(); + const year = this._date.getFullYear(); + const date = this._date.getDate(); + const day = this._date.getDay(); + + const firstDay = new Date(year, month, date - day); + + // Iterate over the dates and render the days. + const days: {date: Date, html: string}[] = []; + for (let i = 0; i < 7; i++) { + const date = new Date(year, month, firstDay.getDate() + i); + this.rootEl.insertAdjacentHTML('beforeend', this.renderDay(date, date.getDate())); + + const dayEl = this.rootEl.lastElementChild as HTMLElement; + dayEl.addEventListener('click', (e: MouseEvent) => { + // @ts-ignore + this.dispatchEvent(new CustomEvent('day-clicked', {detail: date})); + }); + } + + this.updateNowMarker(); + } + + renderDay(date: Date, dayNumber: number, className: string = ''): string { + const events = getEventsForDay(this._events, date); + const isToday = date.toDateString() === new Date().toDateString(); + + return ` +
    +
    + ${DAYS[date.getDay()]} + ${dayNumber} +
    +
      + ${events.map(event => event.asMediumLengthHTML()).join('')} +
    +
    + `; + } + + updateNowMarker() { + const now = new Date(); + const time = now.getHours() * 100 + now.getMinutes(); + const ratio = time * 100 / 2400; + + // Forcing a sync reflow here by querying the offsetHeight of the day cell. + // Could just set the % value directly instead. + const dayEl = this.rootEl.querySelector('.day.today') as HTMLElement; + if (dayEl) { + const pxHeight = dayEl.offsetHeight; + const markerHeight = pxHeight * ratio / 100; + + this.rootEl.style.setProperty('--now-marker', `${markerHeight}px`); + } + + setTimeout(() => this.updateNowMarker(), 1000); + } + + set date(date: Date) { + this._date = date; + this.render(); + } + + set events(events: CalendarEvent[]) { + this._events = events; + this.render(); + } +} diff --git a/slow-calendar/src/app.ts b/slow-calendar/src/app.ts new file mode 100644 index 0000000..8ced173 --- /dev/null +++ b/slow-calendar/src/app.ts @@ -0,0 +1,35 @@ +import { AppUI } from "./AppUI"; +import { getAllEvents } from "./events-factory"; +import { EventPopup } from "./EventPopup"; +import { Store } from "./Store"; + +addEventListener("DOMContentLoaded", async () => { + const store = new Store(); + const prefs = await store.getStoredPrefs(); + + // This is what the calendar will be centered around on load. + const initDate = prefs.initDate || new Date(); + const initMode = prefs.mode || "month"; + + const appEl = document.getElementById('app'); + const appUI = new AppUI(appEl!, initDate, initMode, []); + + const events = await getAllEvents() + + console.log("Refreshing calendar with events"); + appUI.events = events; + + console.log("Initializing the popup util"); + const popup = new EventPopup(events); + popup.start(); + + // @ts-ignore + appUI.addEventListener("mode-changed", (e: CustomEvent) => { + store.mode = e.detail; + }); + + // @ts-ignore + appUI.addEventListener("date-changed", (e: CustomEvent) => { + store.initDate = e.detail; + }); +}); diff --git a/slow-calendar/src/events-factory.ts b/slow-calendar/src/events-factory.ts new file mode 100644 index 0000000..24f2b9f --- /dev/null +++ b/slow-calendar/src/events-factory.ts @@ -0,0 +1,99 @@ +import { CalendarEvent } from "./CalendarEvent"; + +async function fetchData() { + // Simulate a slower API endpoint call. + console.log("Fetching calendar data ..."); + await new Promise(resolve => setTimeout(resolve, Math.random() * 500)); + const response = await fetch('./data.json'); + return await response.json(); +} + +function createOneOrMultipleEventsFromData(eventData: any): CalendarEvent[] { + const dates: Date[] = [new Date(eventData.startDate)]; + let multiDays: {start: Date, end: Date}|null = null; + + if (eventData.repeat) { + const startDate = new Date(eventData.startDate); + + // If this event is repeated, it must have an end date. + const endDate = eventData.endDate + // Either the end date is specified. + ? new Date(eventData.endDate) + // Or it's 5 year after the start date. + : new Date(startDate.getTime() + 5 * 365 * 24 * 60 * 60 * 1000); + + if (eventData.repeat === "daily") { + console.log("Expanding daily event ..."); + const days = (endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24); + for (let i = 1; i <= days; i++) { + const date = new Date(startDate.getTime() + i * 24 * 60 * 60 * 1000); + // If date is on a week end, skip it. + if (date.getDay() === 0 || date.getDay() === 6) { + continue; + } + + dates.push(date); + } + } else if (eventData.repeat === "weekly") { + console.log("Expanding weekly event ..."); + const weeks = (endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24 * 7); + for (let i = 1; i <= weeks; i++) { + const date = new Date(startDate.getTime() + i * 7 * 24 * 60 * 60 * 1000); + dates.push(date); + } + } else if (eventData.repeat === "monthly") { + console.log("Expanding monthly event ..."); + const months = (endDate.getFullYear() - startDate.getFullYear()) * 12 + (endDate.getMonth() - startDate.getMonth()); + for (let i = 1; i <= months; i++) { + const date = new Date(startDate.getTime()); + date.setMonth(date.getMonth() + i); + dates.push(date); + } + } else if (eventData.repeat === "yearly") { + console.log("Expanding yearly event ..."); + const years = endDate.getFullYear() - startDate.getFullYear(); + for (let i = 1; i <= years; i++) { + const date = new Date(startDate.getTime()); + date.setFullYear(date.getFullYear() + i); + dates.push(date); + } + } + } else if (eventData.endDate) { + // If this event has a start and end date, but isn't repeated. + const startDate = new Date(eventData.startDate); + const endDate = new Date(eventData.endDate); + const days = (endDate.getTime() - startDate.getTime()) / (1000 * 60 * 60 * 24); + for (let i = 1; i <= days; i++) { + const date = new Date(startDate.getTime() + i * 24 * 60 * 60 * 1000); + dates.push(date); + } + multiDays = {start: startDate, end: endDate}; + } + + + return dates.map((date, index) => { + const id = eventData.id + (index > 0 ? `-${index}` : ""); + const event = new CalendarEvent(eventData.title, id, date); + + event.multiDays = multiDays; + + event.color = eventData.color; + event.time = eventData.startTime; + event.duration = eventData.duration; + event.repeat = eventData.repeat; + event.location = eventData.location; + event.description = eventData.description; + event.rsvp = eventData.rsvp; + + return event; + }); +} + +export async function getAllEvents(): Promise { + const data = await fetchData(); + + console.log("Processing events ..."); + return data.events.map((event: any) => { + return createOneOrMultipleEventsFromData(event); + }).flat(); +} diff --git a/slow-calendar/src/utils.ts b/slow-calendar/src/utils.ts new file mode 100644 index 0000000..7718807 --- /dev/null +++ b/slow-calendar/src/utils.ts @@ -0,0 +1,33 @@ +import { CalendarEvent } from "./CalendarEvent"; + +export function getEventsForDay(events: CalendarEvent[], date: Date): CalendarEvent[] { + return events.filter(event => { + return event.date.getFullYear() === date.getFullYear() && + event.date.getMonth() === date.getMonth() && + event.date.getDate() === date.getDate(); + }).sort((a, b) => { + // If one has no time property, it should be first. + // If both have no time, no sorting is needed. + if (!a.time && !!b.time) { + return -1; + } else if (!!a.time && !b.time) { + return 1; + } else if (!a.time && !b.time) { + return 0; + } + + // Sort the events by time. Times are strings in 24 + // formats, so just remove the leading 0, if any, and + // the colon, and compare the numbers. + const aTime = parseInt(a.time.replace(/^0/, '').replace(':', '')); + const bTime = parseInt(b.time.replace(/^0/, '').replace(':', '')); + return aTime - bTime; + }).sort((a, b) => { + // Sort the events that are multi-days first. + if (!!a.multiDays && !b.multiDays) { + return -1; + } else if (!a.multiDays && !!b.multiDays) { + return 1; + } + }); +} diff --git a/slow-calendar/tsconfig.json b/slow-calendar/tsconfig.json new file mode 100644 index 0000000..658c4c2 --- /dev/null +++ b/slow-calendar/tsconfig.json @@ -0,0 +1,12 @@ +{ + "compilerOptions": { + "outDir": "./dist/", + "sourceMap": true, + "noImplicitAny": true, + "module": "es6", + "target": "es6", + "jsx": "react", + "allowJs": true, + "moduleResolution": "node" + } +} \ No newline at end of file diff --git a/slow-calendar/webpack.config.js b/slow-calendar/webpack.config.js new file mode 100644 index 0000000..93adeb2 --- /dev/null +++ b/slow-calendar/webpack.config.js @@ -0,0 +1,22 @@ +const path = require('path'); + +module.exports = { + entry: './src/app.ts', + devtool: 'inline-source-map', + module: { + rules: [ + { + test: /\.tsx?$/, + use: 'ts-loader', + exclude: /node_modules/, + }, + ], + }, + resolve: { + extensions: ['.tsx', '.ts', '.js'], + }, + output: { + filename: 'bundle.js', + path: path.resolve(__dirname, 'public'), + }, +};