diff --git a/.github/workflows/pr.yml b/.github/workflows/pr.yml deleted file mode 100644 index b63383d9..00000000 --- a/.github/workflows/pr.yml +++ /dev/null @@ -1,24 +0,0 @@ -name: Pull Requests -on: - pull_request: - types: - - opened - - reopened - - synchronize - -jobs: - size: - name: Check PR size - runs-on: ubuntu-latest - steps: - - uses: ookami-kb/gh-pr-size-watcher@v1 - with: - githubToken: ${{ secrets.GITHUB_TOKEN }} - errorSize: 500 - errorMessage: ':no_entry: PR has more than **{allowed} additions**. Split it into smaller PRs.' - warningSize: 300 - warningMessage: ':warning: PR has more than **{allowed} additions**. Consider splitting it into smaller PRs.' - excludePaths: | - pnpm-lock.yaml - **/*.spec.ts - **/*.spec.tsx diff --git a/astro.config.mjs b/astro.config.mjs index a2523c5a..24b34409 100644 --- a/astro.config.mjs +++ b/astro.config.mjs @@ -1,13 +1,25 @@ import { defineConfig } from 'astro/config'; +import Icons from 'unplugin-icons/vite'; import react from '@astrojs/react'; import { visualizer } from 'rollup-plugin-visualizer'; +import svelte from '@astrojs/svelte'; // https://astro.build/config export default defineConfig({ - integrations: [react()], + integrations: [ + react(), + svelte({ + extensions: ['.svelte'], + }), + ], vite: { - plugins: [visualizer({ template: 'raw-data' })], + plugins: [ + visualizer({ template: 'raw-data' }), + Icons({ + compiler: 'svelte', + }), + ], build: { rollupOptions: { output: { diff --git a/eslint.config.mjs b/eslint.config.mjs index c767890d..da6ca4f2 100644 --- a/eslint.config.mjs +++ b/eslint.config.mjs @@ -10,6 +10,7 @@ import { fileURLToPath } from 'node:url'; import js from '@eslint/js'; import { FlatCompat } from '@eslint/eslintrc'; import testingLibrary from 'eslint-plugin-testing-library'; +import eslintPluginSvelte from 'eslint-plugin-svelte'; const __filename = fileURLToPath(import.meta.url); const __dirname = path.dirname(__filename); @@ -29,6 +30,7 @@ export default [ 'src/env.d.ts', ], }, + ...eslintPluginSvelte.configs['flat/recommended'], ...fixupConfigRules( compat.extends( 'eslint:recommended', diff --git a/knowledge.md b/knowledge.md new file mode 100644 index 00000000..84ed75bf --- /dev/null +++ b/knowledge.md @@ -0,0 +1,74 @@ +# TaskRatchet Web App + +## Verifying changes + +Use `pnpm verify` to check that code changes are valid. + +## Task Status Behavior + +- Pending tasks should have enabled controls (edit button, checkbox) +- Task status must persist across page refreshes +- Some controls should be disabled for non-pending tasks +- Task status determines available actions: + - Pending: Can edit, complete, uncle, copy + - Complete: Can mark incomplete, copy + - Expired: Can copy +- Completing tasks requires careful handling: + - Marking past-due tasks incomplete is a critical action + - Reversing incompleted past-due tasks requires support intervention + - UI should prevent accidental completion + +## Future Improvements + +### UI/UX + +- Add loading states for actions (uncle, edit, etc.) +- Improve mobile responsiveness +- Test dark mode thoroughly +- Add keyboard navigation +- Add ARIA labels for better accessibility + +### Error Handling + +- Add error handling for failed API calls +- Add error boundaries for component failures + +### Testing + +- Add unit tests for new Svelte components +- Test performance with large task lists +- Test across different browsers +- Add end-to-end tests for critical paths + +### Features + +- Consider adding task sorting options +- Consider adding task filtering by status +- Consider adding task categories or tags + +## Style Guide + +- Keep components minimal and focused +- Use CSS variables for theming +- Follow BEM naming convention for CSS +- Use TypeScript for type safety + +## Architecture + +- Components should be self-contained +- State management should be simple and local when possible +- Modal state should be managed by the modal component +- API calls should be centralized + +### Migration Strategy + +- Migrating from React to Svelte +- New features should be built in Svelte +- Keep React and Svelte code separate: + - React code lives in src/react/ + - React tests excluded from main test suite + - New tests should be written for Svelte components +- Test configuration: + - Exclude React tests while preserving Vitest defaults + - React tests remain in src/react/ but aren't run by default + - Maintain separate test suites for React and Svelte diff --git a/package.json b/package.json index 5e7a10ac..88417923 100644 --- a/package.json +++ b/package.json @@ -9,25 +9,29 @@ "preview": "astro preview", "astro": "astro", "preinstall": "npx only-allow pnpm", - "checkTs": "tsc --noEmit", "format": "prettier --write .", "lint": "eslint --max-warnings 0 .", "lint:fix": "eslint --max-warnings 0 --fix .", "test": "vitest", - "unused": "pnpm dlx depcheck" + "unused": "pnpm dlx depcheck", + "check": "svelte-check", + "checkTs": "npm run check", + "verify": "npm run build && npm run check && npm run lint && npm run test -- --run" }, "dependencies": { "@astrojs/check": "^0.9.3", "@astrojs/react": "^3.6.2", + "@astrojs/svelte": "^6.0.1", "@emotion/react": "^11.13.0", "@emotion/styled": "^11.13.0", "@mui/icons-material": "^5.16.6", "@mui/lab": "5.0.0-alpha.173", "@mui/material": "^5.16.6", "@mui/x-date-pickers": "^7.12.0", - "@taskratchet/sdk": "^0.0.7", + "@taskratchet/sdk": "^0.1.1", "astro": "^4.15.1", "dayjs": "^1.11.12", + "eslint-plugin-svelte": "^2.46.0", "humanize-duration": "^3.32.1", "query-string": "^9.1.0", "react": "^18.3.1", @@ -37,12 +41,14 @@ "react-query": "3.39.3", "react-router-dom": "^6.26.0", "react-toastify": "^10.0.5", - "typescript": "^5.5.4" + "svelte": "^5.2.2", + "typescript": "^5.6.3" }, "devDependencies": { "@eslint/compat": "^1.1.1", "@eslint/eslintrc": "^3.1.0", "@eslint/js": "^9.8.0", + "@iconify/json": "^2.2.273", "@testing-library/jest-dom": "^6.4.8", "@testing-library/react": "^16.0.0", "@testing-library/user-event": "^14.5.2", @@ -67,8 +73,10 @@ "jsdom": "^24.1.1", "prettier": "3.3.3", "rollup-plugin-visualizer": "^5.12.0", + "svelte-check": "^4.0.9", "type-fest": "^4.23.0", "typescript": "^5.5.4", + "unplugin-icons": "^0.20.1", "vitest": "^2.0.5", "vitest-fetch-mock": "^0.3.0" } diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 6b055ff6..3287cf36 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -10,10 +10,13 @@ importers: dependencies: '@astrojs/check': specifier: ^0.9.3 - version: 0.9.3(prettier@3.3.3)(typescript@5.5.4) + version: 0.9.3(prettier@3.3.3)(typescript@5.6.3) '@astrojs/react': specifier: ^3.6.2 version: 3.6.2(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)) + '@astrojs/svelte': + specifier: ^6.0.1 + version: 6.0.1(astro@4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.6.3))(svelte@5.2.2)(typescript@5.6.3)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)) '@emotion/react': specifier: ^11.13.0 version: 11.13.3(@types/react@18.3.5)(react@18.3.1) @@ -33,14 +36,17 @@ importers: specifier: ^7.12.0 version: 7.15.0(@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1))(@emotion/styled@11.13.0(@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1))(@types/react@18.3.5)(react@18.3.1))(@mui/material@5.16.7(@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1))(@emotion/styled@11.13.0(@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1))(@types/react@18.3.5)(react@18.3.1))(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@mui/system@5.16.7(@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1))(@emotion/styled@11.13.0(@emotion/react@11.13.3(@types/react@18.3.5)(react@18.3.1))(@types/react@18.3.5)(react@18.3.1))(@types/react@18.3.5)(react@18.3.1))(@types/react@18.3.5)(dayjs@1.11.13)(react-dom@18.3.1(react@18.3.1))(react@18.3.1) '@taskratchet/sdk': - specifier: ^0.0.7 - version: 0.0.7 + specifier: ^0.1.1 + version: 0.1.1 astro: specifier: ^4.15.1 - version: 4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.5.4) + version: 4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.6.3) dayjs: specifier: ^1.11.12 version: 1.11.13 + eslint-plugin-svelte: + specifier: ^2.46.0 + version: 2.46.0(eslint@9.9.1)(svelte@5.2.2) humanize-duration: specifier: ^3.32.1 version: 3.32.1 @@ -68,9 +74,12 @@ importers: react-toastify: specifier: ^10.0.5 version: 10.0.5(react-dom@18.3.1(react@18.3.1))(react@18.3.1) + svelte: + specifier: ^5.2.2 + version: 5.2.2 typescript: - specifier: ^5.5.4 - version: 5.5.4 + specifier: ^5.6.3 + version: 5.6.3 devDependencies: '@eslint/compat': specifier: ^1.1.1 @@ -81,6 +90,9 @@ importers: '@eslint/js': specifier: ^9.8.0 version: 9.9.1 + '@iconify/json': + specifier: ^2.2.273 + version: 2.2.273 '@testing-library/jest-dom': specifier: ^6.4.8 version: 6.5.0 @@ -113,10 +125,10 @@ importers: version: 5.3.3 '@typescript-eslint/eslint-plugin': specifier: ^8.0.1 - version: 8.3.0(@typescript-eslint/parser@8.3.0(eslint@9.9.1)(typescript@5.5.4))(eslint@9.9.1)(typescript@5.5.4) + version: 8.3.0(@typescript-eslint/parser@8.3.0(eslint@9.9.1)(typescript@5.6.3))(eslint@9.9.1)(typescript@5.6.3) '@typescript-eslint/parser': specifier: ^8.0.1 - version: 8.3.0(eslint@9.9.1)(typescript@5.5.4) + version: 8.3.0(eslint@9.9.1)(typescript@5.6.3) '@vitest/ui': specifier: ^2.0.5 version: 2.0.5(vitest@2.0.5) @@ -140,7 +152,7 @@ importers: version: 1.10.0(eslint@9.9.1) eslint-plugin-testing-library: specifier: ^6.2.2 - version: 6.3.0(eslint@9.9.1)(typescript@5.5.4) + version: 6.3.0(eslint@9.9.1)(typescript@5.6.3) globals: specifier: ^15.9.0 version: 15.9.0 @@ -153,9 +165,15 @@ importers: rollup-plugin-visualizer: specifier: ^5.12.0 version: 5.12.0(rollup@4.21.2) + svelte-check: + specifier: ^4.0.9 + version: 4.0.9(svelte@5.2.2)(typescript@5.6.3) type-fest: specifier: ^4.23.0 version: 4.26.0 + unplugin-icons: + specifier: ^0.20.1 + version: 0.20.1(svelte@5.2.2) vitest: specifier: ^2.0.5 version: 2.0.5(@types/node@22.5.2)(@vitest/ui@2.0.5)(jsdom@24.1.3)(sass@1.77.8) @@ -172,6 +190,12 @@ packages: resolution: {integrity: sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==} engines: {node: '>=6.0.0'} + '@antfu/install-pkg@0.4.1': + resolution: {integrity: sha512-T7yB5QNG29afhWVkVq7XeIMBa5U/vs9mX69YqayXypPRmYzUmzwnYltplHmPtZ4HPCn+sQKeXW8I47wCbuBOjw==} + + '@antfu/utils@0.7.10': + resolution: {integrity: sha512-+562v9k4aI80m1+VuMHehNJWLOFjBnXn3tdOitzD0il5b7smkSBal4+a3oKiQTbrwMmN/TBUMDvbdoWDehgOww==} + '@astrojs/check@0.9.3': resolution: {integrity: sha512-I6Dz45bMI5YRbp4yK2LKWsHH3/kkHRGdPGruGkLap6pqxhdcNh7oCgN04Ac+haDfc9ow5BYPGPmEhkwef15GQQ==} hasBin: true @@ -212,6 +236,14 @@ packages: react: ^17.0.2 || ^18.0.0 || ^19.0.0-beta react-dom: ^17.0.2 || ^18.0.0 || ^19.0.0-beta + '@astrojs/svelte@6.0.1': + resolution: {integrity: sha512-R5z7p4ZMIrIPa2hWqqSlWWuz/qKhBTTaTLtDFzWOTtriymowh5JHwG1tv3Zqddn75tAsBsbHyDBQ99y0EaK1/Q==} + engines: {node: ^18.17.1 || ^20.3.0 || >=21.0.0} + peerDependencies: + astro: ^4.0.0 + svelte: ^5.1.16 + typescript: ^5.3.3 + '@astrojs/telemetry@3.1.0': resolution: {integrity: sha512-/ca/+D8MIKEC8/A9cSaPUqQNZm+Es/ZinRv0ZAzvu2ios7POQSsVD+VOj7/hypWNsNM3T7RpfgNq7H2TU1KEHA==} engines: {node: ^18.17.1 || ^20.3.0 || >=21.0.0} @@ -793,6 +825,15 @@ packages: resolution: {integrity: sha512-d2CGZR2o7fS6sWB7DG/3a95bGKQyHMACZ5aW8qGkkqQpUoZV6C0X7Pc7l4ZNMZkfNBf4VWNe9E1jRsf0G146Ew==} engines: {node: '>=18.18'} + '@iconify/json@2.2.273': + resolution: {integrity: sha512-HTnvNpfeFxrMu39VaSTU+UE+ai75Yxa5BbuHmQz9EAtgyPqTqxLgBXPX1sPeMQP8Fxc2X9UrmDwbH5dOEaWthw==} + + '@iconify/types@2.0.0': + resolution: {integrity: sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg==} + + '@iconify/utils@2.1.33': + resolution: {integrity: sha512-jP9h6v/g0BIZx0p7XGJJVtkVnydtbgTgt9mVNcGDYwaa7UhdHdI9dvoq+gKj9sijMSJKxUPEG2JyjsgXjxL7Kw==} + '@img/sharp-darwin-arm64@0.33.5': resolution: {integrity: sha512-UT4p+iz/2H4twwAoLCqfA9UH5pI6DggwKEGuaPy7nCVQ8ZsiY5PIcrRvD1DzuY3qYL07NtIQcWnBSY/heikIFQ==} engines: {node: ^18.17.0 || ^20.3.0 || >=21.0.0} @@ -1217,8 +1258,23 @@ packages: '@shikijs/core@1.15.2': resolution: {integrity: sha512-hi6XZuwHYn6bU4wtXZxST8ynM55aiU2+rVU9aPIrSxqKmEKl4d65puwGsggwcZWTET+7zGXKe7AUj46iQ8Aq8w==} - '@taskratchet/sdk@0.0.7': - resolution: {integrity: sha512-FUbwTuBwl/+/c5dKN154TG+tNbRJ7SRUsrpLEJJU/EHjEGi3N8vDez45/Y0AYhNNsEtic0gF9ffnPZVxsIIojA==} + '@sveltejs/vite-plugin-svelte-inspector@3.0.1': + resolution: {integrity: sha512-2CKypmj1sM4GE7HjllT7UKmo4Q6L5xFRd7VMGEWhYnZ+wc6AUVU01IBd7yUi6WnFndEwWoMNOd6e8UjoN0nbvQ==} + engines: {node: ^18.0.0 || ^20.0.0 || >=22} + peerDependencies: + '@sveltejs/vite-plugin-svelte': ^4.0.0-next.0||^4.0.0 + svelte: ^5.0.0-next.96 || ^5.0.0 + vite: ^5.0.0 + + '@sveltejs/vite-plugin-svelte@4.0.1': + resolution: {integrity: sha512-prXoAE/GleD2C4pKgHa9vkdjpzdYwCSw/kmjw6adIyu0vk5YKCfqIztkLg10m+kOYnzZu3bb0NaPTxlWre2a9Q==} + engines: {node: ^18.0.0 || ^20.0.0 || >=22} + peerDependencies: + svelte: ^5.0.0-next.96 || ^5.0.0 + vite: ^5.0.0 + + '@taskratchet/sdk@0.1.1': + resolution: {integrity: sha512-tU5GDgPYhfIdfKL7f0JXoeqJoP4HqSZhd3i7/l45I4tBG/x/hCXEo+2JFLoBexRfWEvF+mCGWWfbcecZMwRzNw==} '@testing-library/dom@10.4.0': resolution: {integrity: sha512-pemlzrSESWbdAloYml3bAJMEfNh1Z7EduzqPKprCH5S341frlpYnUEW0H72dLxa6IsYr+mPno20GiSm+h9dEdQ==} @@ -1273,6 +1329,9 @@ packages: '@types/estree@1.0.5': resolution: {integrity: sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==} + '@types/estree@1.0.6': + resolution: {integrity: sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==} + '@types/hast@3.0.4': resolution: {integrity: sha512-WPs+bbQw5aCj+x6laNGWLH3wviHtoCv/P3+otBhbOhJgG8qtpdAMlTCxLtsTWA7LH1Oh/bFCHsBn0TPS5m30EQ==} @@ -1477,11 +1536,21 @@ packages: peerDependencies: acorn: ^6.0.0 || ^7.0.0 || ^8.0.0 + acorn-typescript@1.4.13: + resolution: {integrity: sha512-xsc9Xv0xlVfwp2o7sQ+GCQ1PgbkdcpWdTzrwXxO3xDMTAywVS3oXVOcOHuRjAPkS4P9b+yc/qNF15460v+jp4Q==} + peerDependencies: + acorn: '>=8.9.0' + acorn@8.12.1: resolution: {integrity: sha512-tcpGyI9zbizT9JbV6oYE477V6mTlXvvi0T0G3SNIYE2apm/G5huBa1+K89VGeovbg+jycCrfhl3ADxErOuO6Jg==} engines: {node: '>=0.4.0'} hasBin: true + acorn@8.14.0: + resolution: {integrity: sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==} + engines: {node: '>=0.4.0'} + hasBin: true + agent-base@7.1.1: resolution: {integrity: sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==} engines: {node: '>= 14'} @@ -1532,6 +1601,10 @@ packages: aria-query@5.3.0: resolution: {integrity: sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==} + aria-query@5.3.2: + resolution: {integrity: sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw==} + engines: {node: '>= 0.4'} + array-buffer-byte-length@1.0.1: resolution: {integrity: sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==} engines: {node: '>= 0.4'} @@ -1689,6 +1762,10 @@ packages: resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} engines: {node: '>= 8.10.0'} + chokidar@4.0.1: + resolution: {integrity: sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==} + engines: {node: '>= 14.16.0'} + ci-info@4.0.0: resolution: {integrity: sha512-TdHqgGf9odd8SXNuxtUBVx8Nv+qZOejE6qyqiy5NtbYYQOeFa6zmHkxlPzmaLxWWHsU6nJmB7AETdVPi+2NBUg==} engines: {node: '>=8'} @@ -1746,6 +1823,9 @@ packages: concat-map@0.0.1: resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + confbox@0.1.8: + resolution: {integrity: sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w==} + convert-source-map@1.9.0: resolution: {integrity: sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==} @@ -1810,6 +1890,15 @@ packages: supports-color: optional: true + debug@4.3.7: + resolution: {integrity: sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==} + engines: {node: '>=6.0'} + peerDependencies: + supports-color: '*' + peerDependenciesMeta: + supports-color: + optional: true + decimal.js@10.4.3: resolution: {integrity: sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==} @@ -1820,6 +1909,9 @@ packages: resolution: {integrity: sha512-+8VxcR21HhTy8nOt6jf20w0c9CADrw1O8d+VZ/YzzCt4bJ3uBjw+D1q2osAB8RnpwwaeYBxy0HyKQxD5JBMuuQ==} engines: {node: '>=14.16'} + dedent-js@1.0.1: + resolution: {integrity: sha512-OUepMozQULMLUmhxS95Vudo0jb0UchLimi3+pQ2plj61Fcy8axbP9hbiD4Sz6DPqn6XG3kfmziVfQ1rSys5AJQ==} + deep-eql@5.0.2: resolution: {integrity: sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q==} engines: {node: '>=6'} @@ -1827,6 +1919,10 @@ packages: deep-is@0.1.4: resolution: {integrity: sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==} + deepmerge@4.3.1: + resolution: {integrity: sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==} + engines: {node: '>=0.10.0'} + define-data-property@1.1.4: resolution: {integrity: sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==} engines: {node: '>= 0.4'} @@ -1972,6 +2068,12 @@ packages: resolution: {integrity: sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw==} engines: {node: '>=12'} + eslint-compat-utils@0.5.1: + resolution: {integrity: sha512-3z3vFexKIEnjHE3zCMRo6fn/e44U7T1khUjg+Hp0ZQMCigh28rALD0nPFBcGZuiLC5rLZa2ubQHDRln09JfU2Q==} + engines: {node: '>=12'} + peerDependencies: + eslint: '>=6.0.0' + eslint-config-prettier@9.1.0: resolution: {integrity: sha512-NSWl5BFQWEPi1j4TjVNItzYV7dZXZ+wP6I6ZhrBGpChQhZRUaElihE9uRRkcbRnNb76UMKDF3r+WTmNcGPKsqw==} hasBin: true @@ -2002,6 +2104,16 @@ packages: peerDependencies: eslint: '>=4.0.0' + eslint-plugin-svelte@2.46.0: + resolution: {integrity: sha512-1A7iEMkzmCZ9/Iz+EAfOGYL8IoIG6zeKEq1SmpxGeM5SXmoQq+ZNnCpXFVJpsxPWYx8jIVGMerQMzX20cqUl0g==} + engines: {node: ^14.17.0 || >=16.0.0} + peerDependencies: + eslint: ^7.0.0 || ^8.0.0-0 || ^9.0.0-0 + svelte: ^3.37.0 || ^4.0.0 || ^5.0.0 + peerDependenciesMeta: + svelte: + optional: true + eslint-plugin-testing-library@6.3.0: resolution: {integrity: sha512-GYcEErTt6EGwE0bPDY+4aehfEBpB2gDBFKohir8jlATSUvzStEyzCx8QWB/14xeKc/AwyXkzScSzMHnFojkWrA==} engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0, npm: '>=6'} @@ -2012,6 +2124,10 @@ packages: resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==} engines: {node: '>=8.0.0'} + eslint-scope@7.2.2: + resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + eslint-scope@8.0.2: resolution: {integrity: sha512-6E4xmrTw5wtxnLA5wYL3WDfhZ/1bUBGOXV0zQvVRDOtrR8D0p6W7fs3JweNYhwRYeGvd/1CKX2se0/2s7Q/nJA==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} @@ -2034,10 +2150,17 @@ packages: jiti: optional: true + esm-env@1.1.4: + resolution: {integrity: sha512-oO82nKPHKkzIj/hbtuDYy/JHqBHFlMIW36SDiPCVsj87ntDLcWN+sJ1erdVryd4NxODacFTsdrIE3b7IamqbOg==} + espree@10.1.0: resolution: {integrity: sha512-M1M6CpiE6ffoigIOWYO9UDP8TMUw9kqb21tf+08IgDYjCsOvCuDt4jQcZmoYxx+w7zlKw9/N0KXfto+I8/FrXA==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + espree@9.6.1: + resolution: {integrity: sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + esprima@4.0.1: resolution: {integrity: sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==} engines: {node: '>=4'} @@ -2047,6 +2170,9 @@ packages: resolution: {integrity: sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==} engines: {node: '>=0.10'} + esrap@1.2.2: + resolution: {integrity: sha512-F2pSJklxx1BlQIQgooczXCPHmcWpn6EsP5oo73LQfonG9fIlIENQ8vMmfGXeojP9MrkzUNAfyU5vdFlR9shHAw==} + esrecurse@4.3.0: resolution: {integrity: sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==} engines: {node: '>=4.0'} @@ -2106,6 +2232,14 @@ packages: resolution: {integrity: sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==} engines: {node: '>=0.8.0'} + fdir@6.4.2: + resolution: {integrity: sha512-KnhMXsKSPZlAhp7+IjUkRZKPb4fUyccpDrdFXbi4QL1qkmFh9kVY09Yox+n4MaOb3lHZ1Tv829C3oaaXoMYPDQ==} + peerDependencies: + picomatch: ^3 || ^4 + peerDependenciesMeta: + picomatch: + optional: true + fflate@0.8.2: resolution: {integrity: sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A==} @@ -2497,6 +2631,9 @@ packages: is-potential-custom-element-name@1.0.1: resolution: {integrity: sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==} + is-reference@3.0.3: + resolution: {integrity: sha512-ixkJoqQvAP88E6wLydLGGqCJsrFUnqoH6HnaczB8XmDH1oaWU+xxdptvikTgaEhtZ53Ky6YXiBuUI2WXLMCwjw==} + is-regex@1.1.4: resolution: {integrity: sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==} engines: {node: '>= 0.4'} @@ -2634,10 +2771,20 @@ packages: resolution: {integrity: sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==} engines: {node: '>=6'} + known-css-properties@0.35.0: + resolution: {integrity: sha512-a/RAk2BfKk+WFGhhOCAYqSiFLc34k8Mt/6NWRI4joER0EYUzXIcFivjjnoD3+XU1DggLn/tZc3DOAgke7l8a4A==} + + kolorist@1.8.0: + resolution: {integrity: sha512-Y+60/zizpJ3HRH8DCss+q95yr6145JXZo46OTpFvDZWLfRCE4qChOyk1b26nMaNpfHHgxagk9dXT5OP0Tfe+dQ==} + levn@0.4.1: resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} engines: {node: '>= 0.8.0'} + lilconfig@2.1.0: + resolution: {integrity: sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==} + engines: {node: '>=10'} + lines-and-columns@1.2.4: resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} @@ -2645,6 +2792,13 @@ packages: resolution: {integrity: sha512-OfCBkGEw4nN6JLtgRidPX6QxjBQGQf72q3si2uvqyFEMbycSFFHwAZeXx6cJgFM9wmLrf9zBwCP3Ivqa+LLZPw==} engines: {node: '>=6'} + local-pkg@0.5.0: + resolution: {integrity: sha512-ok6z3qlYyCDS4ZEU27HaU6x/xZa9Whf8jD4ptH5UZTQYZVYeb9bnZ3ojVhiJNLiXK1Hfc0GNbLXcmZ5plLDDBg==} + engines: {node: '>=14'} + + locate-character@3.0.0: + resolution: {integrity: sha512-SW13ws7BjaeJ6p7Q6CO2nchbYEc3X3J6WrmTTDto7yMPqVSZTUyY5Tjbid+Ab8gLnATtygYtiDIJGQRRn2ZOiA==} + locate-path@5.0.0: resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==} engines: {node: '>=8'} @@ -2679,6 +2833,9 @@ packages: loupe@3.1.1: resolution: {integrity: sha512-edNu/8D5MKVfGVFRhFf8aAxiTM6Wumfz5XsaatSxlD3w4R1d/WEKUTydCdPGbl9K7QG/Ca3GnDV2sIKIpXRQcw==} + lower-case@2.0.2: + resolution: {integrity: sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==} + lru-cache@5.1.1: resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==} @@ -2689,6 +2846,9 @@ packages: magic-string@0.30.11: resolution: {integrity: sha512-+Wri9p0QHMy+545hKww7YAu5NyzF8iomPL/RQazugQ9+Ez4Ic3mERMd8ZTX5rfK944j+560ZJi8iAwgak1Ac7A==} + magic-string@0.30.12: + resolution: {integrity: sha512-Ea8I3sQMVXr8JhN4z+H/d8zwo+tYDgHE9+5G4Wnrwhs0gaK9fXTKx0Tw5Xwsd/bCPTTZNRAdpyzvoeORe9LYpw==} + magicast@0.3.5: resolution: {integrity: sha512-L0WhttDl+2BOsybvEOLK7fW3UA0OQ0IQ2d6Zl2x/a6vVRs3bAY0ECOSHHeL5jD+SbOpOCUEi0y1DgHEn9Qn1AQ==} @@ -2862,6 +3022,13 @@ packages: resolution: {integrity: sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==} engines: {node: '>=16 || 14 >=14.17'} + mlly@1.7.3: + resolution: {integrity: sha512-xUsx5n/mN0uQf4V548PKQ+YShA4/IW0KI1dZhrNrPCLG+xizETbHTkOa1f8/xut9JRPp8kQuMnz0oqwkTiLo/A==} + + mri@1.2.0: + resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==} + engines: {node: '>=4'} + mrmime@2.0.0: resolution: {integrity: sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==} engines: {node: '>=10'} @@ -2869,6 +3036,9 @@ packages: ms@2.1.2: resolution: {integrity: sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==} + ms@2.1.3: + resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} + muggle-string@0.4.1: resolution: {integrity: sha512-VNTrAak/KhO2i8dqqnqnAHOa3cYBwXEZe9h+D5h/1ZqFSTEFHdM65lR7RoIqq3tBBYavsOXV84NoHXZ0AkPyqQ==} @@ -2890,6 +3060,9 @@ packages: nlcst-to-string@4.0.0: resolution: {integrity: sha512-YKLBCcUYKAg0FNlOBT6aI91qFmSiFKiluk655WzPF+DDMA02qIyy8uiRqI8QXtcFpEvll12LpL5MXqEmAZ+dcA==} + no-case@3.0.4: + resolution: {integrity: sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==} + node-fetch@2.7.0: resolution: {integrity: sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==} engines: {node: 4.x || >=6.0.0} @@ -2999,6 +3172,9 @@ packages: resolution: {integrity: sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==} engines: {node: '>=6'} + package-manager-detector@0.2.4: + resolution: {integrity: sha512-H/OUu9/zUfP89z1APcBf2X8Us0tt8dUK4lUmKqz12QNXif3DxAs1/YqjGtcutZi1zQqeNQRWr9C+EbQnnvSSFA==} + parent-module@1.0.1: resolution: {integrity: sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==} engines: {node: '>=6'} @@ -3013,6 +3189,9 @@ packages: parse5@7.1.2: resolution: {integrity: sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==} + pascal-case@3.1.2: + resolution: {integrity: sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==} + path-browserify@1.0.1: resolution: {integrity: sha512-b7uo2UCUOYZcnF/3ID0lulOJi/bafxa1xPe7ZPsammBSpjSWQkjNxlt635YGS2MiR9GjvuXCtz2emr3jbsz98g==} @@ -3064,10 +3243,41 @@ packages: resolution: {integrity: sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==} engines: {node: '>=8'} + pkg-types@1.2.1: + resolution: {integrity: sha512-sQoqa8alT3nHjGuTjuKgOnvjo4cljkufdtLMnO2LBP/wRwuDlo1tkaEdMxCRhyGRPacv/ztlZgDPm2b7FAmEvw==} + possible-typed-array-names@1.0.0: resolution: {integrity: sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==} engines: {node: '>= 0.4'} + postcss-load-config@3.1.4: + resolution: {integrity: sha512-6DiM4E7v4coTE4uzA8U//WhtPwyhiim3eyjEMFCnUpzbrkK9wJHgKDT2mR+HbtSrd/NubVaYTOpSpjUl8NQeRg==} + engines: {node: '>= 10'} + peerDependencies: + postcss: '>=8.0.9' + ts-node: '>=9.0.0' + peerDependenciesMeta: + postcss: + optional: true + ts-node: + optional: true + + postcss-safe-parser@6.0.0: + resolution: {integrity: sha512-FARHN8pwH+WiS2OPCxJI8FuRJpTVnn6ZNFiqAM2aeW2LwTHWWmWgIyKC6cUo0L8aeKiF/14MNvnpls6R2PBeMQ==} + engines: {node: '>=12.0'} + peerDependencies: + postcss: ^8.3.3 + + postcss-scss@4.0.9: + resolution: {integrity: sha512-AjKOeiwAitL/MXxQW2DliT28EKukvvbEWx3LBmJIRN8KfBGZbRTxNYW0kSqi1COiTZ57nZ9NW06S6ux//N1c9A==} + engines: {node: '>=12.0'} + peerDependencies: + postcss: ^8.4.29 + + postcss-selector-parser@6.1.2: + resolution: {integrity: sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==} + engines: {node: '>=4'} + postcss@8.4.42: resolution: {integrity: sha512-hywKUQB9Ra4dR1mGhldy5Aj1X3MWDSIA1cEi+Uy0CjheLvP6Ual5RlwMCh8i/X121yEDLDIKBsrCQ8ba3FDMfQ==} engines: {node: ^10 || ^12 || >=14} @@ -3202,6 +3412,10 @@ packages: resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} engines: {node: '>=8.10.0'} + readdirp@4.0.2: + resolution: {integrity: sha512-yDMz9g+VaZkqBYS/ozoBJwaBhTbZo3UNYQHNRw1D3UFQB8oHB4uS/tAODO+ZLjGWmUbKnIlOWO+aaIiAxrUWHA==} + engines: {node: '>= 14.16.0'} + redent@3.0.0: resolution: {integrity: sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==} engines: {node: '>=8'} @@ -3326,6 +3540,10 @@ packages: run-parallel@1.2.0: resolution: {integrity: sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==} + sade@1.8.1: + resolution: {integrity: sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A==} + engines: {node: '>=6'} + safe-array-concat@1.1.2: resolution: {integrity: sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==} engines: {node: '>=0.4'} @@ -3521,6 +3739,33 @@ packages: resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} engines: {node: '>= 0.4'} + svelte-check@4.0.9: + resolution: {integrity: sha512-SVNCz2L+9ZELGli7G0n3B3QE5kdf0u27RtKr2ZivWQhcWIXatZxwM4VrQ6AiA2k9zKp2mk5AxkEhdjbpjv7rEw==} + engines: {node: '>= 18.0.0'} + hasBin: true + peerDependencies: + svelte: ^4.0.0 || ^5.0.0-next.0 + typescript: '>=5.0.0' + + svelte-eslint-parser@0.43.0: + resolution: {integrity: sha512-GpU52uPKKcVnh8tKN5P4UZpJ/fUDndmq7wfsvoVXsyP+aY0anol7Yqo01fyrlaWGMFfm4av5DyrjlaXdLRJvGA==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + peerDependencies: + svelte: ^3.37.0 || ^4.0.0 || ^5.0.0 + peerDependenciesMeta: + svelte: + optional: true + + svelte2tsx@0.7.26: + resolution: {integrity: sha512-cBGGZ3Hejuh9gYICQcGKg1zyYhrzZR04iTUdP5pD+CIbjoZiyzS4cpCi4IHh0PA2Vf6cFuwVRmDs9wnOfJotJA==} + peerDependencies: + svelte: ^3.55 || ^4.0.0-next.0 || ^4.0 || ^5.0.0-next.0 + typescript: ^4.9.4 || ^5.0.0 + + svelte@5.2.2: + resolution: {integrity: sha512-eHIJRcvA6iuXdRGMESTmBtWTQCcCiol4gyH9DA60ybS35W1x27cvtbndNvWDqX72blyf+AYeQ4gzZ0XGg3L8sw==} + engines: {node: '>=18'} + symbol-tree@3.2.4: resolution: {integrity: sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==} @@ -3636,11 +3881,14 @@ packages: typescript-auto-import-cache@0.3.3: resolution: {integrity: sha512-ojEC7+Ci1ij9eE6hp8Jl9VUNnsEKzztktP5gtYNRMrTmfXVwA1PITYYAkpxCvvupdSYa/Re51B6KMcv1CTZEUA==} - typescript@5.5.4: - resolution: {integrity: sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==} + typescript@5.6.3: + resolution: {integrity: sha512-hjcS1mhfuyi4WW8IWtjP7brDrG2cuDZukyrYrSauoXGNgx0S7zceP07adYkJycEr56BOUTNPzbInooiN3fn1qw==} engines: {node: '>=14.17'} hasBin: true + ufo@1.5.4: + resolution: {integrity: sha512-UsUk3byDzKd04EyoZ7U4DOlxQaD14JUKQl6/P7wiX4FNvUfm3XL246n9W5AmqwW5RSFJ27NAuM0iLscAOYUiGQ==} + ultrahtml@1.5.3: resolution: {integrity: sha512-GykOvZwgDWZlTQMtp5jrD4BVL+gNn2NVlVafjcFUJ7taY20tqYdwdoWBFy6GBJsNTZe1GkGPkSl5knQAjtgceg==} @@ -3691,6 +3939,33 @@ packages: unload@2.2.0: resolution: {integrity: sha512-B60uB5TNBLtN6/LsgAf3udH9saB5p7gqJwcFfbOEZ8BcBHnGwCf6G/TGiEqkRAxX7zAFIUtzdrXQSdL3Q/wqNA==} + unplugin-icons@0.20.1: + resolution: {integrity: sha512-0z5sYGx07Q69ZrJB4kjmx7a5LYLNSWwyq95Ox9OuSG2y/sbhJaHUapRPOJcKmKhOAyToDVRdy9P7gxJ05lYipw==} + peerDependencies: + '@svgr/core': '>=7.0.0' + '@svgx/core': ^1.0.1 + '@vue/compiler-sfc': ^3.0.2 || ^2.7.0 + svelte: ^3.0.0 || ^4.0.0 || ^5.0.0 + vue-template-compiler: ^2.6.12 + vue-template-es2015-compiler: ^1.9.0 + peerDependenciesMeta: + '@svgr/core': + optional: true + '@svgx/core': + optional: true + '@vue/compiler-sfc': + optional: true + svelte: + optional: true + vue-template-compiler: + optional: true + vue-template-es2015-compiler: + optional: true + + unplugin@1.16.0: + resolution: {integrity: sha512-5liCNPuJW8dqh3+DM6uNM2EI3MLLpCKp/KY+9pB5M2S2SR2qvvDHhKgBOaTWEbZTAws3CXfB0rKTIolWKL05VQ==} + engines: {node: '>=14.0.0'} + update-browserslist-db@1.1.0: resolution: {integrity: sha512-EdRAaAyk2cUE1wOf2DkEhzxqOQvFOoRJFNS6NeyJ01Gp2beMRpBAINjM2iDXE3KCuKhwnvHIQCJm6ThL2Z+HzQ==} hasBin: true @@ -3703,6 +3978,9 @@ packages: url-parse@1.5.10: resolution: {integrity: sha512-WypcfiRhfeUP9vvF0j6rw0J3hrWrw6iZv3+22h6iRMJ/8z1Tj6XfLP4DsUix5MhMPnXpiHDoKyoZ/bdCkwBCiQ==} + util-deprecate@1.0.2: + resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} + vfile-location@5.0.3: resolution: {integrity: sha512-5yXvWDEgqeiYiBe1lbxYF7UMAIm/IcopxMHrMQDq3nvKcjPKIhZklUKL+AE7J7uApI4kwe2snsK+eI6UTj9EHg==} @@ -3756,6 +4034,14 @@ packages: vite: optional: true + vitefu@1.0.3: + resolution: {integrity: sha512-iKKfOMBHob2WxEJbqbJjHAkmYgvFDPhuqrO82om83S8RLk+17FtyMBfcyeH8GqD0ihShtkMW/zzJgiA51hCNCQ==} + peerDependencies: + vite: ^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0-beta.0 + peerDependenciesMeta: + vite: + optional: true + vitest-fetch-mock@0.3.0: resolution: {integrity: sha512-g6upWcL8/32fXL43/5f4VHcocuwQIi9Fj5othcK9gPO8XqSEGtnIZdenr2IaipDr61ReRFt+vaOEgo8jiUUX5w==} engines: {node: '>=14.14.0'} @@ -3910,6 +4196,9 @@ packages: resolution: {integrity: sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==} engines: {node: '>=12'} + webpack-virtual-modules@0.6.2: + resolution: {integrity: sha512-66/V2i5hQanC51vBQKPH4aI8NMAcBW59FVBs+rC7eGHupMyfn34q7rZIE+ETlJ+XTevqfUhVVBgSUNSW2flEUQ==} + websocket-driver@0.7.4: resolution: {integrity: sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==} engines: {node: '>=0.8.0'} @@ -4047,6 +4336,9 @@ packages: resolution: {integrity: sha512-b4JR1PFR10y1mKjhHY9LaGo6tmrgjit7hxVIeAmyMw3jegXR4dhYqLaQF5zMXZxY7tLpMyJeLjr1C4rLmkVe8g==} engines: {node: '>=12.20'} + zimmerframe@1.1.2: + resolution: {integrity: sha512-rAbqEGa8ovJy4pyBxZM70hg4pE6gDgaQ0Sl9M3enG3I0d6H4XSAM3GeNGLKnsBpuijUow064sf7ww1nutC5/3w==} + zod-to-json-schema@3.23.2: resolution: {integrity: sha512-uSt90Gzc/tUfyNqxnjlfBs8W6WSGpNBv0rVsNxP/BVSMHMKGdthPYff4xtCHYloJGM0CFxFsb3NbC0eqPhfImw==} peerDependencies: @@ -4073,13 +4365,20 @@ snapshots: '@jridgewell/gen-mapping': 0.3.5 '@jridgewell/trace-mapping': 0.3.25 - '@astrojs/check@0.9.3(prettier@3.3.3)(typescript@5.5.4)': + '@antfu/install-pkg@0.4.1': dependencies: - '@astrojs/language-server': 2.14.1(prettier@3.3.3)(typescript@5.5.4) + package-manager-detector: 0.2.4 + tinyexec: 0.3.0 + + '@antfu/utils@0.7.10': {} + + '@astrojs/check@0.9.3(prettier@3.3.3)(typescript@5.6.3)': + dependencies: + '@astrojs/language-server': 2.14.1(prettier@3.3.3)(typescript@5.6.3) chokidar: 3.6.0 fast-glob: 3.3.2 kleur: 4.1.5 - typescript: 5.5.4 + typescript: 5.6.3 yargs: 17.7.2 transitivePeerDependencies: - prettier @@ -4089,12 +4388,12 @@ snapshots: '@astrojs/internal-helpers@0.4.1': {} - '@astrojs/language-server@2.14.1(prettier@3.3.3)(typescript@5.5.4)': + '@astrojs/language-server@2.14.1(prettier@3.3.3)(typescript@5.6.3)': dependencies: '@astrojs/compiler': 2.10.3 '@astrojs/yaml2ts': 0.2.1 '@jridgewell/sourcemap-codec': 1.5.0 - '@volar/kit': 2.4.1(typescript@5.5.4) + '@volar/kit': 2.4.1(typescript@5.6.3) '@volar/language-core': 2.4.1 '@volar/language-server': 2.4.1 '@volar/language-service': 2.4.1 @@ -4154,6 +4453,17 @@ snapshots: - supports-color - vite + '@astrojs/svelte@6.0.1(astro@4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.6.3))(svelte@5.2.2)(typescript@5.6.3)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8))': + dependencies: + '@sveltejs/vite-plugin-svelte': 4.0.1(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)) + astro: 4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.6.3) + svelte: 5.2.2 + svelte2tsx: 0.7.26(svelte@5.2.2)(typescript@5.6.3) + typescript: 5.6.3 + transitivePeerDependencies: + - supports-color + - vite + '@astrojs/telemetry@3.1.0': dependencies: ci-info: 4.0.0 @@ -4879,6 +5189,25 @@ snapshots: '@humanwhocodes/retry@0.3.0': {} + '@iconify/json@2.2.273': + dependencies: + '@iconify/types': 2.0.0 + pathe: 1.1.2 + + '@iconify/types@2.0.0': {} + + '@iconify/utils@2.1.33': + dependencies: + '@antfu/install-pkg': 0.4.1 + '@antfu/utils': 0.7.10 + '@iconify/types': 2.0.0 + debug: 4.3.7 + kolorist: 1.8.0 + local-pkg: 0.5.0 + mlly: 1.7.3 + transitivePeerDependencies: + - supports-color + '@img/sharp-darwin-arm64@0.33.5': optionalDependencies: '@img/sharp-libvips-darwin-arm64': 1.0.4 @@ -5207,7 +5536,29 @@ snapshots: dependencies: '@types/hast': 3.0.4 - '@taskratchet/sdk@0.0.7': + '@sveltejs/vite-plugin-svelte-inspector@3.0.1(@sveltejs/vite-plugin-svelte@4.0.1(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)))(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8))': + dependencies: + '@sveltejs/vite-plugin-svelte': 4.0.1(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)) + debug: 4.3.7 + svelte: 5.2.2 + vite: 5.4.2(@types/node@22.5.2)(sass@1.77.8) + transitivePeerDependencies: + - supports-color + + '@sveltejs/vite-plugin-svelte@4.0.1(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8))': + dependencies: + '@sveltejs/vite-plugin-svelte-inspector': 3.0.1(@sveltejs/vite-plugin-svelte@4.0.1(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)))(svelte@5.2.2)(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)) + debug: 4.3.7 + deepmerge: 4.3.1 + kleur: 4.1.5 + magic-string: 0.30.12 + svelte: 5.2.2 + vite: 5.4.2(@types/node@22.5.2)(sass@1.77.8) + vitefu: 1.0.3(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)) + transitivePeerDependencies: + - supports-color + + '@taskratchet/sdk@0.1.1': dependencies: firebase: 10.13.1 transitivePeerDependencies: @@ -5279,6 +5630,8 @@ snapshots: '@types/estree@1.0.5': {} + '@types/estree@1.0.6': {} + '@types/hast@3.0.4': dependencies: '@types/unist': 3.0.3 @@ -5343,34 +5696,34 @@ snapshots: '@types/unist@3.0.3': {} - '@typescript-eslint/eslint-plugin@8.3.0(@typescript-eslint/parser@8.3.0(eslint@9.9.1)(typescript@5.5.4))(eslint@9.9.1)(typescript@5.5.4)': + '@typescript-eslint/eslint-plugin@8.3.0(@typescript-eslint/parser@8.3.0(eslint@9.9.1)(typescript@5.6.3))(eslint@9.9.1)(typescript@5.6.3)': dependencies: '@eslint-community/regexpp': 4.11.0 - '@typescript-eslint/parser': 8.3.0(eslint@9.9.1)(typescript@5.5.4) + '@typescript-eslint/parser': 8.3.0(eslint@9.9.1)(typescript@5.6.3) '@typescript-eslint/scope-manager': 8.3.0 - '@typescript-eslint/type-utils': 8.3.0(eslint@9.9.1)(typescript@5.5.4) - '@typescript-eslint/utils': 8.3.0(eslint@9.9.1)(typescript@5.5.4) + '@typescript-eslint/type-utils': 8.3.0(eslint@9.9.1)(typescript@5.6.3) + '@typescript-eslint/utils': 8.3.0(eslint@9.9.1)(typescript@5.6.3) '@typescript-eslint/visitor-keys': 8.3.0 eslint: 9.9.1 graphemer: 1.4.0 ignore: 5.3.2 natural-compare: 1.4.0 - ts-api-utils: 1.3.0(typescript@5.5.4) + ts-api-utils: 1.3.0(typescript@5.6.3) optionalDependencies: - typescript: 5.5.4 + typescript: 5.6.3 transitivePeerDependencies: - supports-color - '@typescript-eslint/parser@8.3.0(eslint@9.9.1)(typescript@5.5.4)': + '@typescript-eslint/parser@8.3.0(eslint@9.9.1)(typescript@5.6.3)': dependencies: '@typescript-eslint/scope-manager': 8.3.0 '@typescript-eslint/types': 8.3.0 - '@typescript-eslint/typescript-estree': 8.3.0(typescript@5.5.4) + '@typescript-eslint/typescript-estree': 8.3.0(typescript@5.6.3) '@typescript-eslint/visitor-keys': 8.3.0 debug: 4.3.6 eslint: 9.9.1 optionalDependencies: - typescript: 5.5.4 + typescript: 5.6.3 transitivePeerDependencies: - supports-color @@ -5384,14 +5737,14 @@ snapshots: '@typescript-eslint/types': 8.3.0 '@typescript-eslint/visitor-keys': 8.3.0 - '@typescript-eslint/type-utils@8.3.0(eslint@9.9.1)(typescript@5.5.4)': + '@typescript-eslint/type-utils@8.3.0(eslint@9.9.1)(typescript@5.6.3)': dependencies: - '@typescript-eslint/typescript-estree': 8.3.0(typescript@5.5.4) - '@typescript-eslint/utils': 8.3.0(eslint@9.9.1)(typescript@5.5.4) + '@typescript-eslint/typescript-estree': 8.3.0(typescript@5.6.3) + '@typescript-eslint/utils': 8.3.0(eslint@9.9.1)(typescript@5.6.3) debug: 4.3.6 - ts-api-utils: 1.3.0(typescript@5.5.4) + ts-api-utils: 1.3.0(typescript@5.6.3) optionalDependencies: - typescript: 5.5.4 + typescript: 5.6.3 transitivePeerDependencies: - eslint - supports-color @@ -5400,7 +5753,7 @@ snapshots: '@typescript-eslint/types@8.3.0': {} - '@typescript-eslint/typescript-estree@5.62.0(typescript@5.5.4)': + '@typescript-eslint/typescript-estree@5.62.0(typescript@5.6.3)': dependencies: '@typescript-eslint/types': 5.62.0 '@typescript-eslint/visitor-keys': 5.62.0 @@ -5408,13 +5761,13 @@ snapshots: globby: 11.1.0 is-glob: 4.0.3 semver: 7.6.3 - tsutils: 3.21.0(typescript@5.5.4) + tsutils: 3.21.0(typescript@5.6.3) optionalDependencies: - typescript: 5.5.4 + typescript: 5.6.3 transitivePeerDependencies: - supports-color - '@typescript-eslint/typescript-estree@8.3.0(typescript@5.5.4)': + '@typescript-eslint/typescript-estree@8.3.0(typescript@5.6.3)': dependencies: '@typescript-eslint/types': 8.3.0 '@typescript-eslint/visitor-keys': 8.3.0 @@ -5423,20 +5776,20 @@ snapshots: is-glob: 4.0.3 minimatch: 9.0.5 semver: 7.6.3 - ts-api-utils: 1.3.0(typescript@5.5.4) + ts-api-utils: 1.3.0(typescript@5.6.3) optionalDependencies: - typescript: 5.5.4 + typescript: 5.6.3 transitivePeerDependencies: - supports-color - '@typescript-eslint/utils@5.62.0(eslint@9.9.1)(typescript@5.5.4)': + '@typescript-eslint/utils@5.62.0(eslint@9.9.1)(typescript@5.6.3)': dependencies: '@eslint-community/eslint-utils': 4.4.0(eslint@9.9.1) '@types/json-schema': 7.0.15 '@types/semver': 7.5.8 '@typescript-eslint/scope-manager': 5.62.0 '@typescript-eslint/types': 5.62.0 - '@typescript-eslint/typescript-estree': 5.62.0(typescript@5.5.4) + '@typescript-eslint/typescript-estree': 5.62.0(typescript@5.6.3) eslint: 9.9.1 eslint-scope: 5.1.1 semver: 7.6.3 @@ -5444,12 +5797,12 @@ snapshots: - supports-color - typescript - '@typescript-eslint/utils@8.3.0(eslint@9.9.1)(typescript@5.5.4)': + '@typescript-eslint/utils@8.3.0(eslint@9.9.1)(typescript@5.6.3)': dependencies: '@eslint-community/eslint-utils': 4.4.0(eslint@9.9.1) '@typescript-eslint/scope-manager': 8.3.0 '@typescript-eslint/types': 8.3.0 - '@typescript-eslint/typescript-estree': 8.3.0(typescript@5.5.4) + '@typescript-eslint/typescript-estree': 8.3.0(typescript@5.6.3) eslint: 9.9.1 transitivePeerDependencies: - supports-color @@ -5522,12 +5875,12 @@ snapshots: loupe: 3.1.1 tinyrainbow: 1.2.0 - '@volar/kit@2.4.1(typescript@5.5.4)': + '@volar/kit@2.4.1(typescript@5.6.3)': dependencies: '@volar/language-service': 2.4.1 '@volar/typescript': 2.4.1 typesafe-path: 0.2.2 - typescript: 5.5.4 + typescript: 5.6.3 vscode-languageserver-textdocument: 1.0.12 vscode-uri: 3.0.8 @@ -5576,8 +5929,18 @@ snapshots: dependencies: acorn: 8.12.1 + acorn-jsx@5.3.2(acorn@8.14.0): + dependencies: + acorn: 8.14.0 + + acorn-typescript@1.4.13(acorn@8.12.1): + dependencies: + acorn: 8.12.1 + acorn@8.12.1: {} + acorn@8.14.0: {} + agent-base@7.1.1: dependencies: debug: 4.3.6 @@ -5633,6 +5996,8 @@ snapshots: dependencies: dequal: 2.0.3 + aria-query@5.3.2: {} + array-buffer-byte-length@1.0.1: dependencies: call-bind: 1.0.7 @@ -5695,7 +6060,7 @@ snapshots: assertion-error@2.0.1: {} - astro@4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.5.4): + astro@4.15.1(@types/node@22.5.2)(rollup@4.21.2)(sass@1.77.8)(typescript@5.6.3): dependencies: '@astrojs/compiler': 2.10.3 '@astrojs/internal-helpers': 0.4.1 @@ -5752,7 +6117,7 @@ snapshots: string-width: 7.2.0 strip-ansi: 7.1.0 tinyexec: 0.3.0 - tsconfck: 3.1.3(typescript@5.5.4) + tsconfck: 3.1.3(typescript@5.6.3) unist-util-visit: 5.0.0 vfile: 6.0.3 vite: 5.4.2(@types/node@22.5.2)(sass@1.77.8) @@ -5762,7 +6127,7 @@ snapshots: yargs-parser: 21.1.1 zod: 3.23.8 zod-to-json-schema: 3.23.2(zod@3.23.8) - zod-to-ts: 1.2.0(typescript@5.5.4)(zod@3.23.8) + zod-to-ts: 1.2.0(typescript@5.6.3)(zod@3.23.8) optionalDependencies: sharp: 0.33.5 transitivePeerDependencies: @@ -5908,6 +6273,10 @@ snapshots: optionalDependencies: fsevents: 2.3.3 + chokidar@4.0.1: + dependencies: + readdirp: 4.0.2 + ci-info@4.0.0: {} cli-boxes@3.0.0: {} @@ -5960,6 +6329,8 @@ snapshots: concat-map@0.0.1: {} + confbox@0.1.8: {} + convert-source-map@1.9.0: {} convert-source-map@2.0.0: {} @@ -6025,6 +6396,10 @@ snapshots: dependencies: ms: 2.1.2 + debug@4.3.7: + dependencies: + ms: 2.1.3 + decimal.js@10.4.3: {} decode-named-character-reference@1.0.2: @@ -6033,10 +6408,14 @@ snapshots: decode-uri-component@0.4.1: {} + dedent-js@1.0.1: {} + deep-eql@5.0.2: {} deep-is@0.1.4: {} + deepmerge@4.3.1: {} + define-data-property@1.1.4: dependencies: es-define-property: 1.0.0 @@ -6242,6 +6621,11 @@ snapshots: escape-string-regexp@5.0.0: {} + eslint-compat-utils@0.5.1(eslint@9.9.1): + dependencies: + eslint: 9.9.1 + semver: 7.6.3 + eslint-config-prettier@9.1.0(eslint@9.9.1): dependencies: eslint: 9.9.1 @@ -6281,9 +6665,28 @@ snapshots: dependencies: eslint: 9.9.1 - eslint-plugin-testing-library@6.3.0(eslint@9.9.1)(typescript@5.5.4): + eslint-plugin-svelte@2.46.0(eslint@9.9.1)(svelte@5.2.2): + dependencies: + '@eslint-community/eslint-utils': 4.4.0(eslint@9.9.1) + '@jridgewell/sourcemap-codec': 1.5.0 + eslint: 9.9.1 + eslint-compat-utils: 0.5.1(eslint@9.9.1) + esutils: 2.0.3 + known-css-properties: 0.35.0 + postcss: 8.4.42 + postcss-load-config: 3.1.4(postcss@8.4.42) + postcss-safe-parser: 6.0.0(postcss@8.4.42) + postcss-selector-parser: 6.1.2 + semver: 7.6.3 + svelte-eslint-parser: 0.43.0(svelte@5.2.2) + optionalDependencies: + svelte: 5.2.2 + transitivePeerDependencies: + - ts-node + + eslint-plugin-testing-library@6.3.0(eslint@9.9.1)(typescript@5.6.3): dependencies: - '@typescript-eslint/utils': 5.62.0(eslint@9.9.1)(typescript@5.5.4) + '@typescript-eslint/utils': 5.62.0(eslint@9.9.1)(typescript@5.6.3) eslint: 9.9.1 transitivePeerDependencies: - supports-color @@ -6294,6 +6697,11 @@ snapshots: esrecurse: 4.3.0 estraverse: 4.3.0 + eslint-scope@7.2.2: + dependencies: + esrecurse: 4.3.0 + estraverse: 5.3.0 + eslint-scope@8.0.2: dependencies: esrecurse: 4.3.0 @@ -6342,18 +6750,31 @@ snapshots: transitivePeerDependencies: - supports-color + esm-env@1.1.4: {} + espree@10.1.0: dependencies: acorn: 8.12.1 acorn-jsx: 5.3.2(acorn@8.12.1) eslint-visitor-keys: 4.0.0 + espree@9.6.1: + dependencies: + acorn: 8.14.0 + acorn-jsx: 5.3.2(acorn@8.14.0) + eslint-visitor-keys: 3.4.3 + esprima@4.0.1: {} esquery@1.6.0: dependencies: estraverse: 5.3.0 + esrap@1.2.2: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.0 + '@types/estree': 1.0.5 + esrecurse@4.3.0: dependencies: estraverse: 5.3.0 @@ -6414,6 +6835,8 @@ snapshots: dependencies: websocket-driver: 0.7.4 + fdir@6.4.2: {} + fflate@0.8.2: {} file-entry-cache@8.0.0: @@ -6853,6 +7276,10 @@ snapshots: is-potential-custom-element-name@1.0.1: {} + is-reference@3.0.3: + dependencies: + '@types/estree': 1.0.6 + is-regex@1.1.4: dependencies: call-bind: 1.0.7 @@ -6989,11 +7416,17 @@ snapshots: kleur@4.1.5: {} + known-css-properties@0.35.0: {} + + kolorist@1.8.0: {} + levn@0.4.1: dependencies: prelude-ls: 1.2.1 type-check: 0.4.0 + lilconfig@2.1.0: {} + lines-and-columns@1.2.4: {} load-yaml-file@0.2.0: @@ -7003,6 +7436,13 @@ snapshots: pify: 4.0.1 strip-bom: 3.0.0 + local-pkg@0.5.0: + dependencies: + mlly: 1.7.3 + pkg-types: 1.2.1 + + locate-character@3.0.0: {} + locate-path@5.0.0: dependencies: p-locate: 4.1.0 @@ -7034,6 +7474,10 @@ snapshots: dependencies: get-func-name: 2.0.2 + lower-case@2.0.2: + dependencies: + tslib: 2.7.0 + lru-cache@5.1.1: dependencies: yallist: 3.1.1 @@ -7044,6 +7488,10 @@ snapshots: dependencies: '@jridgewell/sourcemap-codec': 1.5.0 + magic-string@0.30.12: + dependencies: + '@jridgewell/sourcemap-codec': 1.5.0 + magicast@0.3.5: dependencies: '@babel/parser': 7.25.6 @@ -7352,7 +7800,7 @@ snapshots: micromark@4.0.0: dependencies: '@types/debug': 4.1.12 - debug: 4.3.6 + debug: 4.3.7 decode-named-character-reference: 1.0.2 devlop: 1.1.0 micromark-core-commonmark: 2.0.1 @@ -7398,10 +7846,21 @@ snapshots: dependencies: brace-expansion: 2.0.1 + mlly@1.7.3: + dependencies: + acorn: 8.14.0 + pathe: 1.1.2 + pkg-types: 1.2.1 + ufo: 1.5.4 + + mri@1.2.0: {} + mrmime@2.0.0: {} ms@2.1.2: {} + ms@2.1.3: {} + muggle-string@0.4.1: {} nano-time@1.0.0: @@ -7418,6 +7877,11 @@ snapshots: dependencies: '@types/nlcst': 2.0.3 + no-case@3.0.4: + dependencies: + lower-case: 2.0.2 + tslib: 2.7.0 + node-fetch@2.7.0: dependencies: whatwg-url: 5.0.0 @@ -7534,6 +7998,8 @@ snapshots: p-try@2.2.0: {} + package-manager-detector@0.2.4: {} + parent-module@1.0.1: dependencies: callsites: 3.1.0 @@ -7558,6 +8024,11 @@ snapshots: dependencies: entities: 4.5.0 + pascal-case@3.1.2: + dependencies: + no-case: 3.0.4 + tslib: 2.7.0 + path-browserify@1.0.1: {} path-exists@4.0.0: {} @@ -7588,8 +8059,34 @@ snapshots: dependencies: find-up: 4.1.0 + pkg-types@1.2.1: + dependencies: + confbox: 0.1.8 + mlly: 1.7.3 + pathe: 1.1.2 + possible-typed-array-names@1.0.0: {} + postcss-load-config@3.1.4(postcss@8.4.42): + dependencies: + lilconfig: 2.1.0 + yaml: 1.10.2 + optionalDependencies: + postcss: 8.4.42 + + postcss-safe-parser@6.0.0(postcss@8.4.42): + dependencies: + postcss: 8.4.42 + + postcss-scss@4.0.9(postcss@8.4.42): + dependencies: + postcss: 8.4.42 + + postcss-selector-parser@6.1.2: + dependencies: + cssesc: 3.0.0 + util-deprecate: 1.0.2 + postcss@8.4.42: dependencies: nanoid: 3.3.7 @@ -7726,6 +8223,8 @@ snapshots: dependencies: picomatch: 2.3.1 + readdirp@4.0.2: {} + redent@3.0.0: dependencies: indent-string: 4.0.0 @@ -7917,6 +8416,10 @@ snapshots: dependencies: queue-microtask: 1.2.3 + sade@1.8.1: + dependencies: + mri: 1.2.0 + safe-array-concat@1.1.2: dependencies: call-bind: 1.0.7 @@ -8150,6 +8653,51 @@ snapshots: supports-preserve-symlinks-flag@1.0.0: {} + svelte-check@4.0.9(svelte@5.2.2)(typescript@5.6.3): + dependencies: + '@jridgewell/trace-mapping': 0.3.25 + chokidar: 4.0.1 + fdir: 6.4.2 + picocolors: 1.0.1 + sade: 1.8.1 + svelte: 5.2.2 + typescript: 5.6.3 + transitivePeerDependencies: + - picomatch + + svelte-eslint-parser@0.43.0(svelte@5.2.2): + dependencies: + eslint-scope: 7.2.2 + eslint-visitor-keys: 3.4.3 + espree: 9.6.1 + postcss: 8.4.42 + postcss-scss: 4.0.9(postcss@8.4.42) + optionalDependencies: + svelte: 5.2.2 + + svelte2tsx@0.7.26(svelte@5.2.2)(typescript@5.6.3): + dependencies: + dedent-js: 1.0.1 + pascal-case: 3.1.2 + svelte: 5.2.2 + typescript: 5.6.3 + + svelte@5.2.2: + dependencies: + '@ampproject/remapping': 2.3.0 + '@jridgewell/sourcemap-codec': 1.5.0 + '@types/estree': 1.0.5 + acorn: 8.12.1 + acorn-typescript: 1.4.13(acorn@8.12.1) + aria-query: 5.3.2 + axobject-query: 4.1.0 + esm-env: 1.1.4 + esrap: 1.2.2 + is-reference: 3.0.3 + locate-character: 3.0.0 + magic-string: 0.30.11 + zimmerframe: 1.1.2 + symbol-tree@3.2.4: {} text-table@0.2.0: {} @@ -8189,22 +8737,22 @@ snapshots: trough@2.2.0: {} - ts-api-utils@1.3.0(typescript@5.5.4): + ts-api-utils@1.3.0(typescript@5.6.3): dependencies: - typescript: 5.5.4 + typescript: 5.6.3 - tsconfck@3.1.3(typescript@5.5.4): + tsconfck@3.1.3(typescript@5.6.3): optionalDependencies: - typescript: 5.5.4 + typescript: 5.6.3 tslib@1.14.1: {} tslib@2.7.0: {} - tsutils@3.21.0(typescript@5.5.4): + tsutils@3.21.0(typescript@5.6.3): dependencies: tslib: 1.14.1 - typescript: 5.5.4 + typescript: 5.6.3 type-check@0.4.0: dependencies: @@ -8252,7 +8800,9 @@ snapshots: dependencies: semver: 7.6.3 - typescript@5.5.4: {} + typescript@5.6.3: {} + + ufo@1.5.4: {} ultrahtml@1.5.3: {} @@ -8326,6 +8876,25 @@ snapshots: '@babel/runtime': 7.25.6 detect-node: 2.1.0 + unplugin-icons@0.20.1(svelte@5.2.2): + dependencies: + '@antfu/install-pkg': 0.4.1 + '@antfu/utils': 0.7.10 + '@iconify/utils': 2.1.33 + debug: 4.3.7 + kolorist: 1.8.0 + local-pkg: 0.5.0 + unplugin: 1.16.0 + optionalDependencies: + svelte: 5.2.2 + transitivePeerDependencies: + - supports-color + + unplugin@1.16.0: + dependencies: + acorn: 8.14.0 + webpack-virtual-modules: 0.6.2 + update-browserslist-db@1.1.0(browserslist@4.23.3): dependencies: browserslist: 4.23.3 @@ -8341,6 +8910,8 @@ snapshots: querystringify: 2.2.0 requires-port: 1.0.0 + util-deprecate@1.0.2: {} + vfile-location@5.0.3: dependencies: '@types/unist': 3.0.3 @@ -8388,6 +8959,10 @@ snapshots: optionalDependencies: vite: 5.4.2(@types/node@22.5.2)(sass@1.77.8) + vitefu@1.0.3(vite@5.4.2(@types/node@22.5.2)(sass@1.77.8)): + optionalDependencies: + vite: 5.4.2(@types/node@22.5.2)(sass@1.77.8) + vitest-fetch-mock@0.3.0(vitest@2.0.5(@types/node@22.5.2)(@vitest/ui@2.0.5)(jsdom@24.1.3)(sass@1.77.8)): dependencies: cross-fetch: 4.0.0 @@ -8552,6 +9127,8 @@ snapshots: webidl-conversions@7.0.0: {} + webpack-virtual-modules@0.6.2: {} + websocket-driver@0.7.4: dependencies: http-parser-js: 0.5.8 @@ -8698,13 +9275,15 @@ snapshots: yocto-queue@1.1.1: {} + zimmerframe@1.1.2: {} + zod-to-json-schema@3.23.2(zod@3.23.8): dependencies: zod: 3.23.8 - zod-to-ts@1.2.0(typescript@5.5.4)(zod@3.23.8): + zod-to-ts@1.2.0(typescript@5.6.3)(zod@3.23.8): dependencies: - typescript: 5.5.4 + typescript: 5.6.3 zod: 3.23.8 zod@3.23.8: {} diff --git a/qa-checklist.md b/qa-checklist.md new file mode 100644 index 00000000..ba1cc890 --- /dev/null +++ b/qa-checklist.md @@ -0,0 +1,73 @@ +# QA Checklist + +## Authentication +- [ ] Login works +- [ ] Logout works +- [ ] Redirects to login when not authenticated +- [ ] Preserves intended destination after login +- [ ] Password reset flow works + +## Task Management +- [ ] Can create new task +- [ ] Can create multiple tasks at once (newline separated) +- [ ] Can edit task (increase stakes only) +- [ ] Can't edit expired tasks +- [ ] Can't postpone due dates +- [ ] Can mark task complete +- [ ] Can uncle task (with confirmation) +- [ ] Task list refreshes after actions +- [ ] Search filter works +- [ ] Archive shows past tasks +- [ ] Archive hides edit/uncle buttons + +## Account Settings +- [ ] Can update name +- [ ] Can update email +- [ ] Can update timezone (dropdown works) +- [ ] Can update password +- [ ] Can manage payment details +- [ ] Can set up Beeminder integration +- [ ] Can generate API token +- [ ] Settings tabs work + +## UI/UX +- [ ] Dark mode toggle works +- [ ] Dark mode persists across reloads +- [ ] Loading indicator shows during API calls +- [ ] Modals close properly +- [ ] Forms show validation errors +- [ ] Forms show success messages +- [ ] Long task descriptions wrap properly +- [ ] Buttons disable appropriately +- [ ] Tooltips show on icons + +## Cross-browser +- [ ] Works in Chrome +- [ ] Works in Firefox +- [ ] Works in Safari +- [ ] Works in Edge +- [ ] Mobile layout is usable + +## Error Handling +- [ ] Shows error messages for failed API calls +- [ ] Handles network errors gracefully +- [ ] Validates form inputs +- [ ] Confirms destructive actions + +## Performance +- [ ] Page loads quickly +- [ ] Actions are responsive +- [ ] No visible lag with many tasks +- [ ] No memory leaks from subscriptions + +## Security +- [ ] No sensitive data in logs +- [ ] API tokens handled securely +- [ ] Session management works +- [ ] Password requirements enforced + +## Environment +- [ ] All required env vars documented +- [ ] Development setup works +- [ ] Production build succeeds +- [ ] Staging deployment works diff --git a/src/components/ApiSettings.svelte b/src/components/ApiSettings.svelte new file mode 100644 index 00000000..d6a869cb --- /dev/null +++ b/src/components/ApiSettings.svelte @@ -0,0 +1,138 @@ + + +
+ {#if error} +
{error}
+ {/if} + {#if success} +
{success}
+ {/if} + +
+

Warning

+ +

+ Requesting a new token will replace your existing token if you have one, + meaning you'll need to replace it wherever you're using it. +

+ +

We don't store your token, so save it somewhere safe.

+ + + Documentation + +
+ + + + + + + + + + + + +
User ID{me?.id || '—'}
API Token{token || '—'}
+ + +
+ + diff --git a/src/components/BeeminderSettings.svelte b/src/components/BeeminderSettings.svelte new file mode 100644 index 00000000..0aaffbae --- /dev/null +++ b/src/components/BeeminderSettings.svelte @@ -0,0 +1,96 @@ + + +{#if error} +
{error}
+{/if} + +{#if bmUser} +
+

Beeminder user: {bmUser}

+ + + + +
+{:else} + Enable Beeminder integration +{/if} + + diff --git a/src/components/ConfirmModal.svelte b/src/components/ConfirmModal.svelte new file mode 100644 index 00000000..29dbdb9b --- /dev/null +++ b/src/components/ConfirmModal.svelte @@ -0,0 +1,66 @@ + + +{#if isOpen} + +{/if} + + diff --git a/src/components/GeneralSettings.svelte b/src/components/GeneralSettings.svelte new file mode 100644 index 00000000..b43763f2 --- /dev/null +++ b/src/components/GeneralSettings.svelte @@ -0,0 +1,138 @@ + + +{#if loading} +
Loading settings...
+{:else} +
+
+ {#if error} +
{error}
+ {/if} + {#if success} +
{success}
+ {/if} + + + + + + + + +
+
+{/if} + + diff --git a/src/components/Nav.svelte b/src/components/Nav.svelte new file mode 100644 index 00000000..dba5e983 --- /dev/null +++ b/src/components/Nav.svelte @@ -0,0 +1,158 @@ + + + + + diff --git a/src/components/PasswordSettings.svelte b/src/components/PasswordSettings.svelte new file mode 100644 index 00000000..58fbab21 --- /dev/null +++ b/src/components/PasswordSettings.svelte @@ -0,0 +1,141 @@ + + +
+
+ {#if error} +
{error}
+ {/if} + {#if success} +
{success}
+ {/if} + + + + + + + + +
+
+ + diff --git a/src/components/PaymentSettings.svelte b/src/components/PaymentSettings.svelte new file mode 100644 index 00000000..9f2f9063 --- /dev/null +++ b/src/components/PaymentSettings.svelte @@ -0,0 +1,80 @@ + + +{#if error} +
{error}
+{/if} + +{#if hasStripeCustomer} + + Manage with Stripe + +{:else} + +{/if} + + diff --git a/src/components/TaskAdd.svelte b/src/components/TaskAdd.svelte new file mode 100644 index 00000000..10c02932 --- /dev/null +++ b/src/components/TaskAdd.svelte @@ -0,0 +1,31 @@ + + + diff --git a/src/components/TaskCheckbox.spec.ts b/src/components/TaskCheckbox.spec.ts new file mode 100644 index 00000000..149336ed --- /dev/null +++ b/src/components/TaskCheckbox.spec.ts @@ -0,0 +1,97 @@ +import { describe, it, expect, vi } from 'vitest'; +import { updateTask } from '@taskratchet/sdk'; +import { toggleComplete } from './TaskCheckbox'; + +vi.mock('@taskratchet/sdk', () => ({ + updateTask: vi.fn(), +})); + +describe('TaskCheckbox', () => { + it('should update task when toggled', async () => { + const task: TaskType = { + id: '123', + task: 'test task', + due: '2024-01-01T00:00:00', + cents: 100, + complete: false, + status: 'pending', + timezone: 'UTC' + }; + + vi.setSystemTime(new Date('2023-12-01')); + const event = new Event('change'); + + await toggleComplete({ event, task, isConfirming: false }); + + expect(updateTask).toHaveBeenCalledWith('123', { complete: true }); + }); + + it('should not update task without id', async () => { + const task: TaskType = { + task: 'test task', + due: '2024-01-01T00:00:00', + cents: 100, + complete: false, + status: 'pending', + timezone: 'UTC' + }; + + const event = new Event('change'); + + await toggleComplete({ event, task, isConfirming: false }); + + expect(updateTask).not.toHaveBeenCalled(); + }); + + it('should prompt for confirmation when uncompleting past due task', async () => { + const task: TaskType = { + id: '123', + task: 'test task', + due: '2023-01-01T00:00:00', + cents: 100, + complete: true, + status: 'complete', + timezone: 'UTC' + }; + + vi.setSystemTime(new Date('2024-01-01')); + const event = new Event('change'); + const checkbox = document.createElement('input'); + checkbox.type = 'checkbox'; + vi.spyOn(event, 'target', 'get').mockReturnValue(checkbox); + vi.spyOn(window, 'confirm').mockReturnValue(false); + + await toggleComplete({ event, task, isConfirming: false }); + + expect(window.confirm).toHaveBeenCalledWith( + 'This task is past due. Marking it incomplete will require contacting support to undo. Continue?' + ); + expect(updateTask).not.toHaveBeenCalled(); + expect(checkbox.checked).toBe(true); + }); + + it('should update past due task when confirmed', async () => { + const task: TaskType = { + id: '123', + task: 'test task', + due: '2023-01-01T00:00:00', + cents: 100, + complete: true, + status: 'complete', + timezone: 'UTC' + }; + + vi.setSystemTime(new Date('2024-01-01')); + const event = new Event('change'); + const checkbox = document.createElement('input'); + checkbox.type = 'checkbox'; + vi.spyOn(event, 'target', 'get').mockReturnValue(checkbox); + vi.spyOn(window, 'confirm').mockReturnValue(true); + + await toggleComplete({ event, task, isConfirming: false }); + + expect(updateTask).toHaveBeenCalledWith('123', { complete: false }); + expect(task.complete).toBe(false); + expect(task.status).toBe('pending'); + }); +}); diff --git a/src/components/TaskCheckbox.svelte b/src/components/TaskCheckbox.svelte new file mode 100644 index 00000000..235b19d2 --- /dev/null +++ b/src/components/TaskCheckbox.svelte @@ -0,0 +1,19 @@ + + + diff --git a/src/components/TaskCheckbox.ts b/src/components/TaskCheckbox.ts new file mode 100644 index 00000000..322a755f --- /dev/null +++ b/src/components/TaskCheckbox.ts @@ -0,0 +1,30 @@ +import { updateTask } from '@taskratchet/sdk'; + +type ToggleCompleteOptions = { + event: Event; + task: TaskType; + isConfirming: boolean; +}; + +export async function toggleComplete({ event, task, isConfirming }: ToggleCompleteOptions) { + if (!task.id) return; + + const checkbox = event.target as HTMLInputElement; + const taskDue = new Date(task.due); + const now = new Date(); + const isPastDue = taskDue < now; + + if (isPastDue && task.complete && !isConfirming) { + checkbox.checked = true; + const confirmed = confirm( + 'This task is past due. Marking it incomplete will require contacting support to undo. Continue?' + ); + if (!confirmed) return; + isConfirming = true; + } + + await updateTask(task.id, { complete: !task.complete }); + task.complete = !task.complete; + task.status = task.complete ? 'complete' : 'pending'; + isConfirming = false; +} diff --git a/src/components/TaskEdit.svelte b/src/components/TaskEdit.svelte new file mode 100644 index 00000000..bfefa951 --- /dev/null +++ b/src/components/TaskEdit.svelte @@ -0,0 +1,29 @@ + + + diff --git a/src/components/TaskModal.svelte b/src/components/TaskModal.svelte new file mode 100644 index 00000000..abad77e4 --- /dev/null +++ b/src/components/TaskModal.svelte @@ -0,0 +1,223 @@ + + +{#if isOpen} + +{/if} + + diff --git a/src/components/list.svelte b/src/components/list.svelte new file mode 100644 index 00000000..4766d9d7 --- /dev/null +++ b/src/components/list.svelte @@ -0,0 +1,200 @@ + + +
+
+

{page === 'next' ? 'Next' : 'Archived'} Tasks

+ +
+ +
    + {#each tasks.filter((task) => task.task + .toLowerCase() + .includes(searchQuery.toLowerCase())) as task} + { + taskToCopy = sourceTask; + isAddOpen = true; + }} + onEdit={(sourceTask) => { + taskToCopy = sourceTask; + isEditing = true; + isAddOpen = true; + }} + onUncle={async (task) => { + if (!task.id) return; + const response = await updateTask(task.id, { uncle: true }); + if (response.ok) { + // Refresh task list + const allTasks = (await getTasks()) as TaskType[]; + const now = new Date(); + const cutoff = new Date(now.getTime() - 24 * 60 * 60 * 1000); + tasks = allTasks + .filter((task) => + page === 'next' + ? new Date(task.due) > cutoff + : new Date(task.due) <= cutoff, + ) + .sort((a, b) => + page === 'next' + ? new Date(a.due).getTime() - new Date(b.due).getTime() + : new Date(b.due).getTime() - new Date(a.due).getTime(), + ); + } + }} + /> + {/each} +
+
+ + + + undefined} + on:tasksAdded={async () => { + const allTasks = (await getTasks()) as TaskType[]; + const now = new Date(); + const cutoff = new Date(now.getTime() - 24 * 60 * 60 * 1000); + tasks = allTasks + .filter((task) => + page === 'next' + ? new Date(task.due) > cutoff + : new Date(task.due) <= cutoff, + ) + .sort((a, b) => new Date(a.due).getTime() - new Date(b.due).getTime()); + }} +/> + + diff --git a/src/components/task.svelte b/src/components/task.svelte new file mode 100644 index 00000000..6c6f232a --- /dev/null +++ b/src/components/task.svelte @@ -0,0 +1,140 @@ + + +
  • +
    + + +
    +
    {task.task}
    +
    + ${task.cents / 100} • due by {formatDue(new Date(task.due))} +
    +
    +
    +
  • + + { + onUncle(task); + showUncleConfirm = false; + }} + onCancel={() => (showUncleConfirm = false)} +/> + + { + showEditModal = false; + onEdit(task); + }} +/> + + diff --git a/src/env.d.ts b/src/env.d.ts index bb66b407..9c8dc85f 100644 --- a/src/env.d.ts +++ b/src/env.d.ts @@ -3,6 +3,8 @@ interface ImportMetaEnv { readonly PUBLIC_WEB3FORMS_ACCESS_KEY: string; readonly PUBLIC_STRIPE_KEY: string; + readonly PUBLIC_BEEMINDER_CLIENT_ID: string; + readonly PUBLIC_BEEMINDER_REDIRECT_URI: string; } interface ImportMeta { diff --git a/src/layouts/Layout.astro b/src/layouts/Layout.astro index 727424d2..4a9b4dbf 100644 --- a/src/layouts/Layout.astro +++ b/src/layouts/Layout.astro @@ -1,5 +1,6 @@ --- import { COLORS } from '../constants.js'; +import Nav from '../components/Nav.svelte'; interface Props { title: string; @@ -55,8 +56,35 @@ const primaryDark = COLORS.warm.lighter, rel="stylesheet" href="https://fonts.googleapis.com/icon?family=Material+Icons" /> + + + +