From 07f4361573c93634f42842bcf2756a55ec1ad9b1 Mon Sep 17 00:00:00 2001 From: zemion Date: Sun, 17 May 2026 02:05:27 +0200 Subject: [PATCH] refactoring, linting, formatting --- CHANGELOG.md | 33 +- README | 54 +- eslint.config.mjs | 55 + package-lock.json | 4183 +++++++++++++---- package.json | 28 +- pdf-tools.zip | Bin 0 -> 103974 bytes prettierignore.txt | 4 + prettierrc.json | 6 + src/App.tsx | 844 +--- src/components/ActionDialog.tsx | 120 +- src/components/ActionsPanel.tsx | 91 +- src/components/FileLoader.tsx | 8 +- src/components/HelpDialog.tsx | 67 +- src/components/Layout.tsx | 6 +- src/components/PagePreviewModal.tsx | 164 +- .../PageWorkspace/CopyPagesDialog.tsx | 224 + .../PageWorkspace/DropIndicator.tsx | 27 + src/components/PageWorkspace/PageCard.tsx | 213 + src/components/PageWorkspace/PageGrid.tsx | 130 + .../PageWorkspace/PageSelectionToolbar.tsx | 112 + src/components/ReorderPanel.tsx | 681 +-- src/components/WorkspacePanel.tsx | 614 +-- src/hooks/usePdfGeneratedOutputs.ts | 128 + src/main.tsx | 12 +- src/pdf/pdfService.ts | 68 +- src/pdf/pdfThumbnailService.ts | 47 +- src/pdf/pdfTypes.ts | 2 +- src/pdf/usePdfThumbnails.ts | 212 + src/styles.css | 6 +- src/version.ts | 2 +- src/vite-env.d.ts | 1 + src/workspace/useWorkspaceState.test.tsx | 208 + src/workspace/useWorkspaceState.ts | 246 + src/workspace/workspaceCommands.test.ts | 105 + src/workspace/workspaceCommands.ts | 10 +- src/workspace/workspaceDb.ts | 44 +- src/workspace/workspaceTypes.ts | 6 +- vite.config.ts | 7 +- 38 files changed, 6121 insertions(+), 2647 deletions(-) create mode 100644 eslint.config.mjs create mode 100644 pdf-tools.zip create mode 100644 prettierignore.txt create mode 100644 prettierrc.json create mode 100644 src/components/PageWorkspace/CopyPagesDialog.tsx create mode 100644 src/components/PageWorkspace/DropIndicator.tsx create mode 100644 src/components/PageWorkspace/PageCard.tsx create mode 100644 src/components/PageWorkspace/PageGrid.tsx create mode 100644 src/components/PageWorkspace/PageSelectionToolbar.tsx create mode 100644 src/hooks/usePdfGeneratedOutputs.ts create mode 100644 src/pdf/usePdfThumbnails.ts create mode 100644 src/vite-env.d.ts create mode 100644 src/workspace/useWorkspaceState.test.tsx create mode 100644 src/workspace/useWorkspaceState.ts create mode 100644 src/workspace/workspaceCommands.test.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index 5c7d6fe..a2a58c6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,37 @@ All notable changes to `pdf-tools` are documented here. The project follows a pragmatic versioning scheme while the app is still below `1.0.0`: minor versions mark coherent user-facing milestones; patch versions mark fixes and small improvements. +## Unreleased + +### Added + +- Added TypeScript type-check, ESLint, Prettier, and aggregate `check` scripts. +- Added ESLint flat config with TypeScript, React Hooks, React Refresh, browser, and Node config support. +- Added Prettier configuration and ignore file. +- Added Vite client type declarations for worker URL imports. +- Added Vitest-based test scripts for one-off and watch-mode test runs. +- Added pure tests for workspace command cloning, snapshot command stability, and serializable command record round-tripping. +- Added hook-level tests for workspace load/replace behavior, command execution, undo, redo, history, redo clearing, dirty-state updates, and content-change callbacks. + +### Changed + +- Marked the package as an ES module package to remove the Vite CJS Node API deprecation warning during local tooling runs. +- Ran Prettier across the project after adding the formatting configuration. +- Split the former monolithic `ReorderPanel` into focused page-workspace components: `PageGrid`, `PageCard`, `PageSelectionToolbar`, `DropIndicator`, and `CopyPagesDialog`. +- Kept drag/drop move, Ctrl/⌘ copy-drag, selection, rotation, deletion, preview opening, and copy-by-position behavior wired through the existing `ReorderPanel` API. +- Extracted generated PDF download URL creation and cleanup for split, extract, and export results into `src/hooks/usePdfGeneratedOutputs.ts`. +- Updated `ActionsPanel` to render prepared download objects instead of creating object URLs during render. +- Extracted workspace page, selection, dirty-state, message, undo/redo history, command creation, command execution, and reset/load helpers from `App.tsx` into `src/workspace/useWorkspaceState.ts`. +- Extracted thumbnail state, caching, invalidation, progressive rendering, rotation-aware rendering, copied-page thumbnail reuse, and thumbnail error reporting into `src/pdf/usePdfThumbnails.ts`. +- Kept PDF loading, IndexedDB persistence, dialogs, preview, merge, export, and split orchestration in `App.tsx` for now. + +### Fixed + +- Fixed existing `tsc --noEmit` failures for Vite worker URL imports and `Uint8Array`/`BlobPart` PDF byte handling. +- Removed a duplicate copy-dialog validation error assignment in `ReorderPanel`. +- Rotated thumbnails from loaded/saved workspaces are now regenerated from the actual current page rotation instead of relying only on rotation changes after load. +- Copied/duplicated pages now receive thumbnails through the shared thumbnail hook/cache path instead of ad-hoc copy handling in `App.tsx`. + ## 0.2.0 — Browser-only PDF workspace baseline ### Added @@ -18,4 +49,4 @@ The project follows a pragmatic versioning scheme while the app is still below ` ### Notes -This release is the baseline for the next refactoring phase. The goal of the upcoming internal changes is to preserve behavior while extracting workspace state, thumbnail handling, generated object URLs, UI subcomponents, tests, and linting into clearer modules. \ No newline at end of file +This release is the baseline for the next refactoring phase. The goal of the upcoming internal changes is to preserve behavior while extracting workspace state, thumbnail handling, generated object URLs, UI subcomponents, tests, and linting into clearer modules. diff --git a/README b/README index 850eb5f..0631671 100644 --- a/README +++ b/README @@ -75,17 +75,17 @@ This makes the project especially useful for self-hosted environments, public-se ### Keyboard shortcuts -| Shortcut | Action | -| --- | --- | -| `F1` / `?` | Open in-app help and tutorial | -| `Ctrl`/`⌘` + `A` | Select all pages | -| `Delete` / `Backspace` | Delete selected pages after confirmation | -| `Esc` | Clear the current selection or close an open dialog | -| `Ctrl`/`⌘` + `Z` | Undo | -| `Ctrl`/`⌘` + `Shift` + `Z` | Redo | -| `Ctrl`/`⌘` + `Y` | Redo | -| `←` / `→` in preview | Move to previous / next page | -| `Esc` in preview | Close preview | +| Shortcut | Action | +| -------------------------- | --------------------------------------------------- | +| `F1` / `?` | Open in-app help and tutorial | +| `Ctrl`/`⌘` + `A` | Select all pages | +| `Delete` / `Backspace` | Delete selected pages after confirmation | +| `Esc` | Clear the current selection or close an open dialog | +| `Ctrl`/`⌘` + `Z` | Undo | +| `Ctrl`/`⌘` + `Shift` + `Z` | Redo | +| `Ctrl`/`⌘` + `Y` | Redo | +| `←` / `→` in preview | Move to previous / next page | +| `Esc` in preview | Close preview | Keyboard shortcuts are ignored while typing in form fields. @@ -115,25 +115,35 @@ browser <-- HTTPS --> static web server / reverse proxy --> built app assets There is no application server to operate. Administrators only need to host the built files from `dist/`. -### Build from source +### Build and test from source ```bash npm ci +npm run check +``` + +`npm run check` runs the main project quality gate: + +```bash +npm run typecheck +npm run lint +npm run test npm run build ``` The production build is written to `dist/`. -To preview the production build locally: +Useful individual development commands: ```bash -npm run preview -``` - -For development: - -```bash -npm run dev +npm run dev # start the Vite development server +npm run preview # preview the production build locally +npm run test # run tests once +npm run test:watch # run tests in watch mode +npm run typecheck # run TypeScript without emitting files +npm run lint # run ESLint +npm run format # format the project with Prettier +npm run format:check # verify Prettier formatting ``` ### Static hosting @@ -219,11 +229,13 @@ The current development baseline is: v0.2.0 — Browser-only PDF workspace baseline ``` +This baseline is preserved through a staged refactoring path. Workspace state, thumbnail handling, generated download URLs, page-grid components, tests, type-checking, linting, and formatting are now separated enough to support the next feature phase without turning `App.tsx` back into a monolith. + ## Project structure ```text src/ - App.tsx Main application state and orchestration + App.tsx Main application orchestration and UI wiring components/ ActionDialog.tsx Reusable confirmation/action dialog ActionsPanel.tsx Export, extract, and split actions diff --git a/eslint.config.mjs b/eslint.config.mjs new file mode 100644 index 0000000..5388e5a --- /dev/null +++ b/eslint.config.mjs @@ -0,0 +1,55 @@ +import js from "@eslint/js"; +import eslintConfigPrettier from "eslint-config-prettier"; +import reactHooks from "eslint-plugin-react-hooks"; +import reactRefresh from "eslint-plugin-react-refresh"; +import globals from "globals"; +import tseslint from "typescript-eslint"; + +export default tseslint.config( + { + ignores: ["dist", "coverage", "node_modules"], + }, + js.configs.recommended, + ...tseslint.configs.recommended, + { + files: ["**/*.{ts,tsx}"], + languageOptions: { + ecmaVersion: 2022, + sourceType: "module", + globals: { + ...globals.browser, + ...globals.es2022, + }, + }, + plugins: { + "react-hooks": reactHooks, + "react-refresh": reactRefresh, + }, + rules: { + ...reactHooks.configs.recommended.rules, + "react-refresh/only-export-components": [ + "warn", + { allowConstantExport: true }, + ], + "react-hooks/set-state-in-effect": "off", + "@typescript-eslint/no-unused-vars": [ + "warn", + { + argsIgnorePattern: "^_", + varsIgnorePattern: "^_", + caughtErrorsIgnorePattern: "^_", + }, + ], + }, + }, + { + files: ["*.config.{js,ts}", "eslint.config.js"], + languageOptions: { + globals: { + ...globals.node, + ...globals.es2022, + }, + }, + }, + eslintConfigPrettier, +); diff --git a/package-lock.json b/package-lock.json index 422f29f..185edff 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "pdf-tools", - "version": "0.2.0", + "version": "0.2.1", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "pdf-tools", - "version": "0.2.0", + "version": "0.2.1", "dependencies": { "pdf-lib": "^1.17.1", "pdfjs-dist": "^4.6.82", @@ -14,402 +14,811 @@ "react-dom": "^18.3.1" }, "devDependencies": { + "@eslint/js": "^10.0.1", + "@testing-library/react": "^16.3.2", + "@testing-library/user-event": "^14.6.1", + "@types/node": "^25.8.0", "@types/react": "^18.3.11", "@types/react-dom": "^18.3.2", - "@vitejs/plugin-react-swc": "^3.7.0", + "@vitejs/plugin-react": "^6.0.2", + "eslint": "^10.4.0", + "eslint-config-prettier": "^10.1.8", + "eslint-plugin-react-hooks": "^7.1.1", + "eslint-plugin-react-refresh": "^0.5.2", + "globals": "^17.6.0", + "jsdom": "^29.1.1", + "prettier": "^3.8.3", "typescript": "^5.6.3", - "vite": "^5.4.10" + "typescript-eslint": "^8.59.3", + "vite": "^8.0.13", + "vitest": "^4.1.6" } }, - "node_modules/@esbuild/aix-ppc64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", - "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==", - "cpu": [ - "ppc64" + "node_modules/@asamuzakjp/css-color": { + "version": "5.1.11", + "resolved": "https://registry.npmjs.org/@asamuzakjp/css-color/-/css-color-5.1.11.tgz", + "integrity": "sha512-KVw6qIiCTUQhByfTd78h2yD1/00waTmm9uy/R7Ck/ctUyAPj+AEDLkQIdJW0T8+qGgj3j5bpNKK7Q3G+LedJWg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@asamuzakjp/generational-cache": "^1.0.1", + "@csstools/css-calc": "^3.2.0", + "@csstools/css-color-parser": "^4.1.0", + "@csstools/css-parser-algorithms": "^4.0.0", + "@csstools/css-tokenizer": "^4.0.0" + }, + "engines": { + "node": "^20.19.0 || ^22.12.0 || >=24.0.0" + } + }, + "node_modules/@asamuzakjp/dom-selector": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/@asamuzakjp/dom-selector/-/dom-selector-7.1.1.tgz", + "integrity": "sha512-67RZDnYRc8H/8MLDgQCDE//zoqVFwajkepHZgmXrbwybzXOEwOWGPYGmALYl9J2DOLfFPPs6kKCqmbzV895hTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@asamuzakjp/generational-cache": "^1.0.1", + "@asamuzakjp/nwsapi": "^2.3.9", + "bidi-js": "^1.0.3", + "css-tree": "^3.2.1", + "is-potential-custom-element-name": "^1.0.1" + }, + "engines": { + "node": "^20.19.0 || ^22.12.0 || >=24.0.0" + } + }, + "node_modules/@asamuzakjp/generational-cache": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@asamuzakjp/generational-cache/-/generational-cache-1.0.1.tgz", + "integrity": "sha512-wajfB8KqzMCN2KGNFdLkReeHncd0AslUSrvHVvvYWuU8ghncRJoA50kT3zP9MVL0+9g4/67H+cdvBskj9THPzg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^20.19.0 || ^22.12.0 || >=24.0.0" + } + }, + "node_modules/@asamuzakjp/nwsapi": { + "version": "2.3.9", + "resolved": "https://registry.npmjs.org/@asamuzakjp/nwsapi/-/nwsapi-2.3.9.tgz", + "integrity": "sha512-n8GuYSrI9bF7FFZ/SjhwevlHc8xaVlb/7HmHelnc/PZXBD2ZR49NnN9sMMuDdEGPeeRQ5d0hqlSlEpgCX3Wl0Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/@babel/code-frame": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz", + "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.28.5", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.29.3", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.3.tgz", + "integrity": "sha512-LIVqM46zQWZhj17qA8wb4nW/ixr2y1Nw+r1etiAWgRM6U1IqP+LNhL1yg440jYZR72jCWcWbLWzIosH+uP1fqg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz", + "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helpers": "^7.28.6", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/traverse": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.29.1", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz", + "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.29.0", + "@babel/types": "^7.29.0", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz", + "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.28.6", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets/node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz", + "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz", + "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-validator-identifier": "^7.28.5", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.2.tgz", + "integrity": "sha512-HoGuUs4sCZNezVEKdVcwqmZN8GoHirLUcLaYVNBK2J0DadGtdcqgr3BCbvH8+XUo4NGjNl3VOtSjEKNzqfFgKw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.29.3", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.3.tgz", + "integrity": "sha512-b3ctpQwp+PROvU/cttc4OYl4MzfJUWy6FZg+PMXfzmt/+39iHVF0sDfqay8TQM3JA2EUOyKcFZt75jWriQijsA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.29.0" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/runtime": { + "version": "7.29.2", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.2.tgz", + "integrity": "sha512-JiDShH45zKHWyGe4ZNVRrCjBz8Nh9TMmZG1kh4QTK8hCBTWBi8Da+i7s1fJw7/lYpM4ccepSNfqzZ/QvABBi5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/template": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz", + "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz", + "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.29.0", + "@babel/generator": "^7.29.0", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.29.0", + "@babel/template": "^7.28.6", + "@babel/types": "^7.29.0", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz", + "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bramus/specificity": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/@bramus/specificity/-/specificity-2.4.2.tgz", + "integrity": "sha512-ctxtJ/eA+t+6q2++vj5j7FYX3nRu311q1wfYH3xjlLOsczhlhxAg2FWNUXhpGvAw3BWo1xBcvOV6/YLc2r5FJw==", + "dev": true, + "license": "MIT", + "dependencies": { + "css-tree": "^3.0.0" + }, + "bin": { + "specificity": "bin/cli.js" + } + }, + "node_modules/@csstools/color-helpers": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/@csstools/color-helpers/-/color-helpers-6.0.2.tgz", + "integrity": "sha512-LMGQLS9EuADloEFkcTBR3BwV/CGHV7zyDxVRtVDTwdI2Ca4it0CCVTT9wCkxSgokjE5Ho41hEPgb8OEUwoXr6Q==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } ], + "license": "MIT-0", + "engines": { + "node": ">=20.19.0" + } + }, + "node_modules/@csstools/css-calc": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/@csstools/css-calc/-/css-calc-3.2.1.tgz", + "integrity": "sha512-DtdHlgXh5ZkA43cwBcAm+huzgJiwx3ZTWVjBs94kwz2xKqSimDA3lBgCjphYgwgVUMWatSM0pDd8TILB1yrVVg==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "engines": { + "node": ">=20.19.0" + }, + "peerDependencies": { + "@csstools/css-parser-algorithms": "^4.0.0", + "@csstools/css-tokenizer": "^4.0.0" + } + }, + "node_modules/@csstools/css-color-parser": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/@csstools/css-color-parser/-/css-color-parser-4.1.1.tgz", + "integrity": "sha512-eZ5XOtyhK+mggRafYUWzA0tvaYOFgdY8AkgQiCJF9qNAePnUo/zmsqqYubBBb3sQ8uNUaSKTY9s9klfRaAXL0g==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "dependencies": { + "@csstools/color-helpers": "^6.0.2", + "@csstools/css-calc": "^3.2.1" + }, + "engines": { + "node": ">=20.19.0" + }, + "peerDependencies": { + "@csstools/css-parser-algorithms": "^4.0.0", + "@csstools/css-tokenizer": "^4.0.0" + } + }, + "node_modules/@csstools/css-parser-algorithms": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@csstools/css-parser-algorithms/-/css-parser-algorithms-4.0.0.tgz", + "integrity": "sha512-+B87qS7fIG3L5h3qwJ/IFbjoVoOe/bpOdh9hAjXbvx0o8ImEmUsGXN0inFOnk2ChCFgqkkGFQ+TpM5rbhkKe4w==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "peer": true, + "engines": { + "node": ">=20.19.0" + }, + "peerDependencies": { + "@csstools/css-tokenizer": "^4.0.0" + } + }, + "node_modules/@csstools/css-syntax-patches-for-csstree": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@csstools/css-syntax-patches-for-csstree/-/css-syntax-patches-for-csstree-1.1.4.tgz", + "integrity": "sha512-wgsqt92b7C7tQhIdPNxj0n9zuUbQlvAuI1exyzeNrOKOi62SD7ren8zqszmpVREjAOqg8cD2FqYhQfAuKjk4sw==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT-0", + "peerDependencies": { + "css-tree": "^3.2.1" + }, + "peerDependenciesMeta": { + "css-tree": { + "optional": true + } + } + }, + "node_modules/@csstools/css-tokenizer": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@csstools/css-tokenizer/-/css-tokenizer-4.0.0.tgz", + "integrity": "sha512-QxULHAm7cNu72w97JUNCBFODFaXpbDg+dP8b/oWFAZ2MTRppA3U00Y2L1HqaS4J6yBqxwa/Y3nMBaxVKbB/NsA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/csstools" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/csstools" + } + ], + "license": "MIT", + "peer": true, + "engines": { + "node": ">=20.19.0" + } + }, + "node_modules/@emnapi/core": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz", + "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==", "dev": true, "license": "MIT", "optional": true, - "os": [ - "aix" - ], - "engines": { - "node": ">=12" + "dependencies": { + "@emnapi/wasi-threads": "1.2.1", + "tslib": "^2.4.0" } }, - "node_modules/@esbuild/android-arm": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz", - "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==", - "cpu": [ - "arm" - ], + "node_modules/@emnapi/core/node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true + }, + "node_modules/@emnapi/runtime": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz", + "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==", "dev": true, "license": "MIT", "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=12" + "dependencies": { + "tslib": "^2.4.0" } }, - "node_modules/@esbuild/android-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz", - "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==", - "cpu": [ - "arm64" - ], + "node_modules/@emnapi/runtime/node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true + }, + "node_modules/@emnapi/wasi-threads": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz", + "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==", "dev": true, "license": "MIT", "optional": true, - "os": [ - "android" - ], - "engines": { - "node": ">=12" + "dependencies": { + "tslib": "^2.4.0" } }, - "node_modules/@esbuild/android-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz", - "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==", - "cpu": [ - "x64" - ], + "node_modules/@emnapi/wasi-threads/node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "android" - ], + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, "engines": { - "node": ">=12" + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" } }, - "node_modules/@esbuild/darwin-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz", - "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==", - "cpu": [ - "arm64" - ], + "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], + "license": "Apache-2.0", "engines": { - "node": ">=12" + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "node_modules/@esbuild/darwin-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz", - "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==", - "cpu": [ - "x64" - ], + "node_modules/@eslint-community/regexpp": { + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "darwin" - ], "engines": { - "node": ">=12" + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, - "node_modules/@esbuild/freebsd-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz", - "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==", - "cpu": [ - "arm64" - ], + "node_modules/@eslint/config-array": { + "version": "0.23.5", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.23.5.tgz", + "integrity": "sha512-Y3kKLvC1dvTOT+oGlqNQ1XLqK6D1HU2YXPc52NmAlJZbMMWDzGYXMiPRJ8TYD39muD/OTjlZmNJ4ib7dvSrMBA==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^3.0.5", + "debug": "^4.3.1", + "minimatch": "^10.2.4" + }, "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "node_modules/@esbuild/freebsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz", - "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==", - "cpu": [ - "x64" - ], + "node_modules/@eslint/config-helpers": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.6.0.tgz", + "integrity": "sha512-ii6Bw9jJ2zi2cWA2Z+9/QZ/+3DX6kwaV5Q986D/CdP3Lap3w/pgQZ373FV7byY/i7L4IRH/G43I5dz1ClsCbpA==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ], + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^1.2.1" + }, "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "node_modules/@esbuild/linux-arm": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz", - "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==", - "cpu": [ - "arm" - ], + "node_modules/@eslint/core": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-1.2.1.tgz", + "integrity": "sha512-MwcE1P+AZ4C6DWlpin/OmOA54mmIZ/+xZuJiQd4SyB29oAJjN30UW9wkKNptW2ctp4cEsvhlLY/CsQ1uoHDloQ==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "node_modules/@esbuild/linux-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz", - "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==", - "cpu": [ - "arm64" - ], + "node_modules/@eslint/js": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-10.0.1.tgz", + "integrity": "sha512-zeR9k5pd4gxjZ0abRoIaxdc7I3nDktoXZk2qOv9gCNWx3mVwEn32VRhyLaRsDiJjTs0xq/T8mfPtyuXu7GWBcA==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "linux" - ], "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "eslint": "^10.0.0" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + } } }, - "node_modules/@esbuild/linux-ia32": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz", - "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==", - "cpu": [ - "ia32" - ], + "node_modules/@eslint/object-schema": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-3.0.5.tgz", + "integrity": "sha512-vqTaUEgxzm+YDSdElad6PiRoX4t8VGDjCtt05zn4nU810UIx/uNEV7/lZJ6KwFThKZOzOxzXy48da+No7HZaMw==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "node_modules/@esbuild/linux-loong64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz", - "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==", - "cpu": [ - "loong64" - ], + "node_modules/@eslint/plugin-kit": { + "version": "0.7.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.7.1.tgz", + "integrity": "sha512-rZAP3aVgB9ds9KOeUSL+zZ21hPmo8dh6fnIFwRQj5EAZl9gzR7wxYbYXYysAM8CTqGmUGyp2S4kUdV17MnGuWQ==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^1.2.1", + "levn": "^0.4.1" + }, "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "node_modules/@esbuild/linux-mips64el": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz", - "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==", - "cpu": [ - "mips64el" - ], + "node_modules/@exodus/bytes": { + "version": "1.15.0", + "resolved": "https://registry.npmjs.org/@exodus/bytes/-/bytes-1.15.0.tgz", + "integrity": "sha512-UY0nlA+feH81UGSHv92sLEPLCeZFjXOuHhrIo0HQydScuQc8s0A7kL/UdgwgDq8g8ilksmuoF35YVTNphV2aBQ==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "linux" - ], "engines": { - "node": ">=12" + "node": "^20.19.0 || ^22.12.0 || >=24.0.0" + }, + "peerDependencies": { + "@noble/hashes": "^1.8.0 || ^2.0.0" + }, + "peerDependenciesMeta": { + "@noble/hashes": { + "optional": true + } } }, - "node_modules/@esbuild/linux-ppc64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz", - "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==", - "cpu": [ - "ppc64" - ], + "node_modules/@humanfs/core": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.2.tgz", + "integrity": "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", + "dependencies": { + "@humanfs/types": "^0.15.0" + }, "engines": { - "node": ">=12" + "node": ">=18.18.0" } }, - "node_modules/@esbuild/linux-riscv64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz", - "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==", - "cpu": [ - "riscv64" - ], + "node_modules/@humanfs/node": { + "version": "0.16.8", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.8.tgz", + "integrity": "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.2", + "@humanfs/types": "^0.15.0", + "@humanwhocodes/retry": "^0.4.0" + }, "engines": { - "node": ">=12" + "node": ">=18.18.0" } }, - "node_modules/@esbuild/linux-s390x": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz", - "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==", - "cpu": [ - "s390x" - ], + "node_modules/@humanfs/types": { + "version": "0.15.0", + "resolved": "https://registry.npmjs.org/@humanfs/types/-/types-0.15.0.tgz", + "integrity": "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", "engines": { - "node": ">=12" + "node": ">=18.18.0" } }, - "node_modules/@esbuild/linux-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz", - "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==", - "cpu": [ - "x64" - ], + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ], + "license": "Apache-2.0", "engines": { - "node": ">=12" + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" } }, - "node_modules/@esbuild/netbsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz", - "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==", - "cpu": [ - "x64" - ], + "node_modules/@humanwhocodes/retry": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", + "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "netbsd" - ], + "license": "Apache-2.0", "engines": { - "node": ">=12" + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" } }, - "node_modules/@esbuild/openbsd-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz", - "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==", - "cpu": [ - "x64" - ], + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ], - "engines": { - "node": ">=12" + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" } }, - "node_modules/@esbuild/sunos-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz", - "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==", - "cpu": [ - "x64" - ], + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "sunos" - ], - "engines": { - "node": ">=12" + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" } }, - "node_modules/@esbuild/win32-arm64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz", - "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==", - "cpu": [ - "arm64" - ], + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "win32" - ], "engines": { - "node": ">=12" + "node": ">=6.0.0" } }, - "node_modules/@esbuild/win32-ia32": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz", - "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==", - "cpu": [ - "ia32" - ], + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=12" - } + "license": "MIT" }, - "node_modules/@esbuild/win32-x64": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz", - "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==", - "cpu": [ - "x64" - ], + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=12" + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" } }, "node_modules/@napi-rs/canvas": { @@ -597,6 +1006,35 @@ "node": ">= 10" } }, + "node_modules/@napi-rs/wasm-runtime": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-1.1.4.tgz", + "integrity": "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow==", + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@tybys/wasm-util": "^0.10.1" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/Brooooooklyn" + }, + "peerDependencies": { + "@emnapi/core": "^1.7.1", + "@emnapi/runtime": "^1.7.1" + } + }, + "node_modules/@oxc-project/types": { + "version": "0.130.0", + "resolved": "https://registry.npmjs.org/@oxc-project/types/-/types-0.130.0.tgz", + "integrity": "sha512-ibD2usx9JRu7f5pu2tMKMI4cpA4NgXJQoYRP4pQ7Pxmn1l6k/53qWtQWZayhYy3X4QZkt90Ot+mJEaeXouio6Q==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/Boshen" + } + }, "node_modules/@pdf-lib/standard-fonts": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/@pdf-lib/standard-fonts/-/standard-fonts-1.0.0.tgz", @@ -615,31 +1053,10 @@ "pako": "^1.0.10" } }, - "node_modules/@rolldown/pluginutils": { - "version": "1.0.0-beta.27", - "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.27.tgz", - "integrity": "sha512-+d0F4MKMCbeVUJwG96uQ4SgAznZNSq93I3V+9NHA4OpvqG8mRCpGdKmK8l/dl02h2CCDHwW2FqilnTyDcAnqjA==", - "dev": true, - "license": "MIT" - }, - "node_modules/@rollup/rollup-android-arm-eabi": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.60.4.tgz", - "integrity": "sha512-F5QXMSiFebS9hKZj02XhWLLnRpJ3B3AROP0tWbFBSj+6kCbg5m9j5JoHKd4mmSVy5mS/IMQloYgYxCuJC0fxEQ==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "android" - ] - }, - "node_modules/@rollup/rollup-android-arm64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.60.4.tgz", - "integrity": "sha512-GxxTKApUpzRhof7poWvCJHRF51C67u1R7D6DiluBE8wKU1u5GWE8t+v81JvJYtbawoBFX1hLv5Ei4eVjkWokaw==", + "node_modules/@rolldown/binding-android-arm64": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.1.tgz", + "integrity": "sha512-fJI3I0r3C3Oj/zdBCpaCmBRZYf07xpaq4yCfDDoSFm+beWNzbIl26puW8RraUdugoJw/95zerNOn6jasAhzSmg==", "cpu": [ "arm64" ], @@ -648,12 +1065,15 @@ "optional": true, "os": [ "android" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-darwin-arm64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.60.4.tgz", - "integrity": "sha512-tua0TaJxMOB1R0V0RS1jFZ/RpURFDJIOR2A6jWwQeawuFyS4gBW+rntLRaQd0EQ4bd6Vp44Z2rXW+YYDBsj6IA==", + "node_modules/@rolldown/binding-darwin-arm64": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-arm64/-/binding-darwin-arm64-1.0.1.tgz", + "integrity": "sha512-cKnAhWEsV7TPcA/5EAteDp6KcJZBQ2G+BqE7zayMMi7kMvwRsbv7WT9aOnn0WNl4SKEIf43vjS31iUPu80nzXg==", "cpu": [ "arm64" ], @@ -662,12 +1082,15 @@ "optional": true, "os": [ "darwin" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-darwin-x64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.60.4.tgz", - "integrity": "sha512-CSKq7MsP+5PFIcydhAiR1K0UhEI1A2jWXVKHPCBZ151yOutENwvnPocgVHkivu2kviURtCEB6zUQw0vs8RrhMg==", + "node_modules/@rolldown/binding-darwin-x64": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-x64/-/binding-darwin-x64-1.0.1.tgz", + "integrity": "sha512-YKrVwQjIRBPo+5G/u03wGjbdy4q7pyzCe93DK9VJ7zkVmeg8LJ7GbgsiHWdR4xSoe4CAXRD7Bcjgbtr64bkXNg==", "cpu": [ "x64" ], @@ -676,26 +1099,15 @@ "optional": true, "os": [ "darwin" - ] - }, - "node_modules/@rollup/rollup-freebsd-arm64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.60.4.tgz", - "integrity": "sha512-+O8OkVdyvXMtJEciu2wS/pzm1IxntEEQx3z5TAVy4l32G0etZn+RsA48ARRrFm6Ri8fvqPQfgrvNxSjKAbnd3g==", - "cpu": [ - "arm64" ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "freebsd" - ] + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-freebsd-x64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.60.4.tgz", - "integrity": "sha512-Iw3oMskH3AfNuhU0MSN7vNbdi4me/NiYo2azqPz/Le16zHSa+3RRmliCMWWQmh4lcndccU40xcJuTYJZxNo/lw==", + "node_modules/@rolldown/binding-freebsd-x64": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-freebsd-x64/-/binding-freebsd-x64-1.0.1.tgz", + "integrity": "sha512-z/oBsREo46SsFqBwYtFe0kpJeBijAT48O/WXLI4suiCLBkr03RTtTJMCzSdDd2znlh8VJizL09XVkQgk8IZonw==", "cpu": [ "x64" ], @@ -704,12 +1116,15 @@ "optional": true, "os": [ "freebsd" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-arm-gnueabihf": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.60.4.tgz", - "integrity": "sha512-EIPRXTVQpHyF8WOo219AD2yEltPehLTcTMz2fn6JsatLYSzQf00hj3rulF+yauOlF9/FtM2WpkT/hJh/KJFGhA==", + "node_modules/@rolldown/binding-linux-arm-gnueabihf": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm-gnueabihf/-/binding-linux-arm-gnueabihf-1.0.1.tgz", + "integrity": "sha512-ik8q7GM11zxvYxFc2PeDcT6TBvhCQMaUxfph/M5l9sKuTs/Sjg3L+Byw0F7w0ZVLBZmx30P+gG0ECzzN+MFcmQ==", "cpu": [ "arm" ], @@ -718,26 +1133,15 @@ "optional": true, "os": [ "linux" - ] - }, - "node_modules/@rollup/rollup-linux-arm-musleabihf": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.60.4.tgz", - "integrity": "sha512-J3Yh9PzzF1Ovah2At+lHiGQdsYgArxBbXv/zHfSyaiFQEqvNv7DcW98pCrmdjCZBrqBiKrKKe2V+aaSGWuBe/w==", - "cpu": [ - "arm" ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-arm64-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.60.4.tgz", - "integrity": "sha512-BFDEZMYfUvLn37ONE1yMBojPxnMlTFsdyNoqncT0qFq1mAfllL+ATMMJd8TeuVMiX84s1KbcxcZbXInmcO2mRg==", + "node_modules/@rolldown/binding-linux-arm64-gnu": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-gnu/-/binding-linux-arm64-gnu-1.0.1.tgz", + "integrity": "sha512-QoSx2EkyrrdZ6kcyE8stqZ62t0Yra8Fs5ia9lOxJrh6TMQJK7gQKmscdTHf7pOXKREKrVwOtJcQG3qVSfc866A==", "cpu": [ "arm64" ], @@ -746,12 +1150,15 @@ "optional": true, "os": [ "linux" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-arm64-musl": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.60.4.tgz", - "integrity": "sha512-pc9EYOSlOgdQ2uPl1o9PF6/kLSgaUosia7gOuS8mB69IxJvlclko1MECXysjs5ryez1/5zjYqx3+xYU0TU6R1A==", + "node_modules/@rolldown/binding-linux-arm64-musl": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-musl/-/binding-linux-arm64-musl-1.0.1.tgz", + "integrity": "sha512-uwNwFpwKeNiZawfAWBgg0VIztPTV3ihhh1vV334h9ivnNLorxnQMU6Fz8wG1Zb4Qh9LC1/MkcyT3YlDXG3Rsgg==", "cpu": [ "arm64" ], @@ -760,40 +1167,15 @@ "optional": true, "os": [ "linux" - ] - }, - "node_modules/@rollup/rollup-linux-loong64-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.60.4.tgz", - "integrity": "sha512-NxnomyxYerDh5n4iLrNa+sH+Z+U4BMEE46V2PgQ/hoB909i8gV1M5wPojWg9fk1jWpO3IQnOs20K4wyZuFLEFQ==", - "cpu": [ - "loong64" ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-loong64-musl": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.60.4.tgz", - "integrity": "sha512-nbJnQ8a3z1mtmrwImCYhc6BGpThAyYVRQxw9uKSKG4wR6aAYno9sVjJ0zaZcW9BPJX1GbrDPf+SvdWjgTuDmnw==", - "cpu": [ - "loong64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-ppc64-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.60.4.tgz", - "integrity": "sha512-2EU6acNrQLd8tYvo/LXW535wupT3m6fo7HKo6lr7ktQoItxTyOL1ZCR/GfGCuXl2vR+zmfI6eRXkSemafv+iVg==", + "node_modules/@rolldown/binding-linux-ppc64-gnu": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-ppc64-gnu/-/binding-linux-ppc64-gnu-1.0.1.tgz", + "integrity": "sha512-zY1bul7OWr7DFBiJ++wofXvnr8B45ce3QsQUhKrIhXsygAh7bTkwyeM1bi1a2g5C/yC/N8TZyGDEoMfm/l9mpg==", "cpu": [ "ppc64" ], @@ -802,54 +1184,15 @@ "optional": true, "os": [ "linux" - ] - }, - "node_modules/@rollup/rollup-linux-ppc64-musl": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.60.4.tgz", - "integrity": "sha512-WeBtoMuaMxiiIrO2IYP3xs6GMWkJP2C0EoT8beTLkUPmzV1i/UcOSVw1d5r9KBODtHKilG5yFxsGRnBbK3wJ4A==", - "cpu": [ - "ppc64" ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-riscv64-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.60.4.tgz", - "integrity": "sha512-FJHFfqpKUI3A10WrWKiFbBZ7yVbGT4q4B5o1qKFFojqpaYoh9LrQgqWCmmcxQzVSXYtyB5bzkXrYzlHTs21MYA==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-riscv64-musl": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.60.4.tgz", - "integrity": "sha512-mcEl6CUT5IAUmQf1m9FYSmVqCJlpQ8r8eyftFUHG8i9OhY7BkBXSUdnLH5DOf0wCOjcP9v/QO93zpmF1SptCCw==", - "cpu": [ - "riscv64" - ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "linux" - ] - }, - "node_modules/@rollup/rollup-linux-s390x-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.60.4.tgz", - "integrity": "sha512-ynt3JxVd2w2buzoKDWIyiV1pJW93xlQic1THVLXilz429oijRpSHivZAgp65KBu+cMcgf1eVVjdnTLvPxgCuoQ==", + "node_modules/@rolldown/binding-linux-s390x-gnu": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-s390x-gnu/-/binding-linux-s390x-gnu-1.0.1.tgz", + "integrity": "sha512-0frlsT/f4Ft6I7SMESTKnF3cZsdicQn1dCMkF/jT9wDLE+gGoiQfv1nmT9e+s7s/fekvvy6tZM2jHvI2tkbJDQ==", "cpu": [ "s390x" ], @@ -858,12 +1201,15 @@ "optional": true, "os": [ "linux" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-x64-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.60.4.tgz", - "integrity": "sha512-Boiz5+MsaROEWDf+GGEwF8VMHGhlUoQMtIPjOgA5fv4osupqTVnJteQNKJwUcnUog2G55jYXH7KZFFiJe0TEzQ==", + "node_modules/@rolldown/binding-linux-x64-gnu": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.1.tgz", + "integrity": "sha512-XABVmGp9Tg0WspTVvwduTc4fpqy6JnAUrSQe6OuyqD/03nI7r0O9OWUkMIwFrjKAIqolvqoA4ZrJppgwE0Gxmw==", "cpu": [ "x64" ], @@ -872,12 +1218,15 @@ "optional": true, "os": [ "linux" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-linux-x64-musl": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.60.4.tgz", - "integrity": "sha512-+qfSY27qIrFfI/Hom04KYFw3GKZSGU4lXus51wsb5EuySfFlWRwjkKWoE9emgRw/ukoT4Udsj4W/+xxG8VbPKg==", + "node_modules/@rolldown/binding-linux-x64-musl": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.1.tgz", + "integrity": "sha512-bV4fzswuzVcKD90o/VM6QqKxnxlDq0g2BISDLNVmxrnhpv1DDbyPhCIjYfvzYLV+MvkKKnQt2Q6AO86SEBULUQ==", "cpu": [ "x64" ], @@ -886,26 +1235,15 @@ "optional": true, "os": [ "linux" - ] - }, - "node_modules/@rollup/rollup-openbsd-x64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.60.4.tgz", - "integrity": "sha512-VpTfOPHgVXEBeeR8hZ2O0F3aSso+JDWqTWmTmzcQKted54IAdUVbxE+j/MVxUsKa8L20HJhv3vUezVPoquqWjA==", - "cpu": [ - "x64" ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "openbsd" - ] + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-openharmony-arm64": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.60.4.tgz", - "integrity": "sha512-IPOsh5aRYuLv/nkU51X10Bf75Bsf6+gZdx1X+QP5QM6lIJFHHqbHLG0uJn/hWthzo13UAc2umiUorqZy3axoZg==", + "node_modules/@rolldown/binding-openharmony-arm64": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-openharmony-arm64/-/binding-openharmony-arm64-1.0.1.tgz", + "integrity": "sha512-/Mh0Zhq3OP7fVs0kcQHZP6lZEthMGTaSf8UBQYSFEZDWGXXlEC+nJ6EqenaK2t4LBXMe3A+K/G2BVXXdtOr4PQ==", "cpu": [ "arm64" ], @@ -914,12 +1252,34 @@ "optional": true, "os": [ "openharmony" - ] + ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-win32-arm64-msvc": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.60.4.tgz", - "integrity": "sha512-4QzE9E81OohJ/HKzHhsqU+zcYYojVOXlFMs1DdyMT6qXl/niOH7AVElmmEdUNHHS/oRkc++d5k6Vy85zFs0DEw==", + "node_modules/@rolldown/binding-wasm32-wasi": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-wasm32-wasi/-/binding-wasm32-wasi-1.0.1.tgz", + "integrity": "sha512-+1xc9X45l8ufsBAm6Gjvx2qDRIY9lTVt0cgWNcJ+1gdhXvkbxePA60yRTwSTuXL09CMhyJmjpV7E3NoyxbqFQQ==", + "cpu": [ + "wasm32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "dependencies": { + "@emnapi/core": "1.10.0", + "@emnapi/runtime": "1.10.0", + "@napi-rs/wasm-runtime": "^1.1.4" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@rolldown/binding-win32-arm64-msvc": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-arm64-msvc/-/binding-win32-arm64-msvc-1.0.1.tgz", + "integrity": "sha512-1D+UqZdfnuR+Jy1GgMJwi85bD40H21uNmOPRWQhw4oRSuolZ/B5rixZ45DK2KXOTCvmVCecauWgEhbw8bI7tOw==", "cpu": [ "arm64" ], @@ -928,26 +1288,15 @@ "optional": true, "os": [ "win32" - ] - }, - "node_modules/@rollup/rollup-win32-ia32-msvc": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.60.4.tgz", - "integrity": "sha512-zTPgT1YuHHcd+Tmx7h8aml0FWFVelV5N54oHow9SLj+GfoDy/huQ+UV396N/C7KpMDMiPspRktzM1/0r1usYEA==", - "cpu": [ - "ia32" ], - "dev": true, - "license": "MIT", - "optional": true, - "os": [ - "win32" - ] + "engines": { + "node": "^20.19.0 || >=22.12.0" + } }, - "node_modules/@rollup/rollup-win32-x64-gnu": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.60.4.tgz", - "integrity": "sha512-DRS4G7mi9lJxqEDezIkKCaUIKCrLUUDCUaCsTPCi/rtqaC6D/jjwslMQyiDU50Ka0JKpeXeRBFBAXwArY52vBw==", + "node_modules/@rolldown/binding-win32-x64-msvc": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-x64-msvc/-/binding-win32-x64-msvc-1.0.1.tgz", + "integrity": "sha512-INAycaWuhlOK3wk4mRHGsdgwYWmd9cChdPdE9bwWmy6rn9VqVNYNFGhOdXrofXUxwHIncSiPNb8tNm8knDVIeQ==", "cpu": [ "x64" ], @@ -956,248 +1305,132 @@ "optional": true, "os": [ "win32" - ] - }, - "node_modules/@rollup/rollup-win32-x64-msvc": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.60.4.tgz", - "integrity": "sha512-QVTUovf40zgTqlFVrKA1uXMVvU2QWEFWfAH8Wdc48IxLvrJMQVMBRjuQyUpzZCDkakImib9eVazbWlC6ksWtJw==", - "cpu": [ - "x64" ], + "engines": { + "node": "^20.19.0 || >=22.12.0" + } + }, + "node_modules/@standard-schema/spec": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@standard-schema/spec/-/spec-1.1.0.tgz", + "integrity": "sha512-l2aFy5jALhniG5HgqrD6jXLi/rUWrKvqN/qJx6yoJsgKhblVd+iqqU4RCXavm/jPityDo5TCvKMnpjKnOriy0w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@testing-library/dom": { + "version": "10.4.1", + "resolved": "https://registry.npmjs.org/@testing-library/dom/-/dom-10.4.1.tgz", + "integrity": "sha512-o4PXJQidqJl82ckFaXUeoAW+XysPLauYI43Abki5hABd853iMhitooc6znOnczgbTYmEP6U6/y1ZyKAIsvMKGg==", "dev": true, "license": "MIT", - "optional": true, - "os": [ - "win32" - ] - }, - "node_modules/@swc/core": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core/-/core-1.15.3.tgz", - "integrity": "sha512-Qd8eBPkUFL4eAONgGjycZXj1jFCBW8Fd+xF0PzdTlBCWQIV1xnUT7B93wUANtW3KGjl3TRcOyxwSx/u/jyKw/Q==", - "dev": true, - "hasInstallScript": true, - "license": "Apache-2.0", + "peer": true, "dependencies": { - "@swc/counter": "^0.1.3", - "@swc/types": "^0.1.25" + "@babel/code-frame": "^7.10.4", + "@babel/runtime": "^7.12.5", + "@types/aria-query": "^5.0.1", + "aria-query": "5.3.0", + "dom-accessibility-api": "^0.5.9", + "lz-string": "^1.5.0", + "picocolors": "1.1.1", + "pretty-format": "^27.0.2" }, "engines": { - "node": ">=10" + "node": ">=18" + } + }, + "node_modules/@testing-library/react": { + "version": "16.3.2", + "resolved": "https://registry.npmjs.org/@testing-library/react/-/react-16.3.2.tgz", + "integrity": "sha512-XU5/SytQM+ykqMnAnvB2umaJNIOsLF3PVv//1Ew4CTcpz0/BRyy/af40qqrt7SjKpDdT1saBMc42CUok5gaw+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/runtime": "^7.12.5" }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/swc" - }, - "optionalDependencies": { - "@swc/core-darwin-arm64": "1.15.3", - "@swc/core-darwin-x64": "1.15.3", - "@swc/core-linux-arm-gnueabihf": "1.15.3", - "@swc/core-linux-arm64-gnu": "1.15.3", - "@swc/core-linux-arm64-musl": "1.15.3", - "@swc/core-linux-x64-gnu": "1.15.3", - "@swc/core-linux-x64-musl": "1.15.3", - "@swc/core-win32-arm64-msvc": "1.15.3", - "@swc/core-win32-ia32-msvc": "1.15.3", - "@swc/core-win32-x64-msvc": "1.15.3" + "engines": { + "node": ">=18" }, "peerDependencies": { - "@swc/helpers": ">=0.5.17" + "@testing-library/dom": "^10.0.0", + "@types/react": "^18.0.0 || ^19.0.0", + "@types/react-dom": "^18.0.0 || ^19.0.0", + "react": "^18.0.0 || ^19.0.0", + "react-dom": "^18.0.0 || ^19.0.0" }, "peerDependenciesMeta": { - "@swc/helpers": { + "@types/react": { + "optional": true + }, + "@types/react-dom": { "optional": true } } }, - "node_modules/@swc/core-darwin-arm64": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-darwin-arm64/-/core-darwin-arm64-1.15.3.tgz", - "integrity": "sha512-AXfeQn0CvcQ4cndlIshETx6jrAM45oeUrK8YeEY6oUZU/qzz0Id0CyvlEywxkWVC81Ajpd8TQQ1fW5yx6zQWkQ==", - "cpu": [ - "arm64" - ], + "node_modules/@testing-library/user-event": { + "version": "14.6.1", + "resolved": "https://registry.npmjs.org/@testing-library/user-event/-/user-event-14.6.1.tgz", + "integrity": "sha512-vq7fv0rnt+QTXgPxr5Hjc210p6YKq2kmdziLgnsZGgLJ9e6VAShx1pACLuRjd/AS/sr7phAR58OIIpf0LlmQNw==", "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "darwin" - ], + "license": "MIT", "engines": { - "node": ">=10" + "node": ">=12", + "npm": ">=6" + }, + "peerDependencies": { + "@testing-library/dom": ">=7.21.4" } }, - "node_modules/@swc/core-darwin-x64": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-darwin-x64/-/core-darwin-x64-1.15.3.tgz", - "integrity": "sha512-p68OeCz1ui+MZYG4wmfJGvcsAcFYb6Sl25H9TxWl+GkBgmNimIiRdnypK9nBGlqMZAcxngNPtnG3kEMNnvoJ2A==", - "cpu": [ - "x64" - ], + "node_modules/@tybys/wasm-util": { + "version": "0.10.2", + "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz", + "integrity": "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg==", "dev": true, - "license": "Apache-2.0 AND MIT", + "license": "MIT", "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm-gnueabihf": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.15.3.tgz", - "integrity": "sha512-Nuj5iF4JteFgwrai97mUX+xUOl+rQRHqTvnvHMATL/l9xE6/TJfPBpd3hk/PVpClMXG3Uvk1MxUFOEzM1JrMYg==", - "cpu": [ - "arm" - ], - "dev": true, - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm64-gnu": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.15.3.tgz", - "integrity": "sha512-2Nc/s8jE6mW2EjXWxO/lyQuLKShcmTrym2LRf5Ayp3ICEMX6HwFqB1EzDhwoMa2DcUgmnZIalesq2lG3krrUNw==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-arm64-musl": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.15.3.tgz", - "integrity": "sha512-j4SJniZ/qaZ5g8op+p1G9K1z22s/EYGg1UXIb3+Cg4nsxEpF5uSIGEE4mHUfA70L0BR9wKT2QF/zv3vkhfpX4g==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-x64-gnu": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.15.3.tgz", - "integrity": "sha512-aKttAZnz8YB1VJwPQZtyU8Uk0BfMP63iDMkvjhJzRZVgySmqt/apWSdnoIcZlUoGheBrcqbMC17GGUmur7OT5A==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-linux-x64-musl": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.15.3.tgz", - "integrity": "sha512-oe8FctPu1gnUsdtGJRO2rvOUIkkIIaHqsO9xxN0bTR7dFTlPTGi2Fhk1tnvXeyAvCPxLIcwD8phzKg6wLv9yug==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-arm64-msvc": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.15.3.tgz", - "integrity": "sha512-L9AjzP2ZQ/Xh58e0lTRMLvEDrcJpR7GwZqAtIeNLcTK7JVE+QineSyHp0kLkO1rttCHyCy0U74kDTj0dRz6raA==", - "cpu": [ - "arm64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-ia32-msvc": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.15.3.tgz", - "integrity": "sha512-B8UtogMzErUPDWUoKONSVBdsgKYd58rRyv2sHJWKOIMCHfZ22FVXICR4O/VwIYtlnZ7ahERcjayBHDlBZpR0aw==", - "cpu": [ - "ia32" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/core-win32-x64-msvc": { - "version": "1.15.3", - "resolved": "https://registry.npmjs.org/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.15.3.tgz", - "integrity": "sha512-SpZKMR9QBTecHeqpzJdYEfgw30Oo8b/Xl6rjSzBt1g0ZsXyy60KLXrp6IagQyfTYqNYE/caDvwtF2FPn7pomog==", - "cpu": [ - "x64" - ], - "dev": true, - "license": "Apache-2.0 AND MIT", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": ">=10" - } - }, - "node_modules/@swc/counter": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz", - "integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==", - "dev": true, - "license": "Apache-2.0" - }, - "node_modules/@swc/types": { - "version": "0.1.25", - "resolved": "https://registry.npmjs.org/@swc/types/-/types-0.1.25.tgz", - "integrity": "sha512-iAoY/qRhNH8a/hBvm3zKj9qQ4oc2+3w1unPJa2XvTK3XjeLXtzcCingVPw/9e5mn1+0yPqxcBGp9Jf0pkfMb1g==", - "dev": true, - "license": "Apache-2.0", "dependencies": { - "@swc/counter": "^0.1.3" + "tslib": "^2.4.0" } }, + "node_modules/@tybys/wasm-util/node_modules/tslib": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz", + "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==", + "dev": true, + "license": "0BSD", + "optional": true + }, + "node_modules/@types/aria-query": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/@types/aria-query/-/aria-query-5.0.4.tgz", + "integrity": "sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/chai": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-5.2.3.tgz", + "integrity": "sha512-Mw558oeA9fFbv65/y4mHtXDs9bPnFMZAL/jxdPFUpOHHIXX91mcgEHbS5Lahr+pwZFR8A7GQleRWeI6cGFC2UA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/deep-eql": "*", + "assertion-error": "^2.0.1" + } + }, + "node_modules/@types/deep-eql": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/@types/deep-eql/-/deep-eql-4.0.2.tgz", + "integrity": "sha512-c9h9dVVMigMPc4bwTvC5dxqtqJZwQPePsWjPlpSOnojbor6pGqdk541lfA7AqFQr5pB1BRdq0juY9db81BwyFw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/esrecurse": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/@types/esrecurse/-/esrecurse-4.3.1.tgz", + "integrity": "sha512-xJBAbDifo5hpffDBuHl0Y8ywswbiAp/Wi7Y/GtAgSlZyIABppyurxVueOPE8LUQOxdlgi6Zqce7uoEpqNTeiUw==", + "dev": true, + "license": "MIT" + }, "node_modules/@types/estree": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", @@ -1205,6 +1438,24 @@ "dev": true, "license": "MIT" }, + "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, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "25.8.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-25.8.0.tgz", + "integrity": "sha512-TCFSk8IZh+iLX1xtksoBVtdmgL+1IX0fC9BeU4QqFSuNdN/K+HUlhqOzEmSYYpZUVsLYcPqc9KX+60iDuninSQ==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "undici-types": ">=7.24.0 <7.24.7" + } + }, "node_modules/@types/prop-types": { "version": "15.7.15", "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz", @@ -1230,22 +1481,604 @@ "integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==", "dev": true, "license": "MIT", + "peer": true, "peerDependencies": { "@types/react": "^18.0.0" } }, - "node_modules/@vitejs/plugin-react-swc": { - "version": "3.11.0", - "resolved": "https://registry.npmjs.org/@vitejs/plugin-react-swc/-/plugin-react-swc-3.11.0.tgz", - "integrity": "sha512-YTJCGFdNMHCMfjODYtxRNVAYmTWQ1Lb8PulP/2/f/oEEtglw8oKxKIZmmRkyXrVrHfsKOaVkAc3NT9/dMutO5w==", + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.59.3.tgz", + "integrity": "sha512-PwFvSKsXGShKGW6n5bZOhGHEcCZXM8HofLK9fNsEwZXzFRjoY+XT1Vsf1zgyXdwTr0ZYz1/2tkZ0DBTT9jZjhw==", "dev": true, "license": "MIT", "dependencies": { - "@rolldown/pluginutils": "1.0.0-beta.27", - "@swc/core": "^1.12.11" + "@eslint-community/regexpp": "^4.12.2", + "@typescript-eslint/scope-manager": "8.59.3", + "@typescript-eslint/type-utils": "8.59.3", + "@typescript-eslint/utils": "8.59.3", + "@typescript-eslint/visitor-keys": "8.59.3", + "ignore": "^7.0.5", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.5.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" }, "peerDependencies": { - "vite": "^4 || ^5 || ^6 || ^7" + "@typescript-eslint/parser": "^8.59.3", + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/ignore": { + "version": "7.0.5", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-7.0.5.tgz", + "integrity": "sha512-Hs59xBNfUIunMFgWAbGX5cq6893IbWg4KnrjbYwX3tx0ztorVgTDA6B2sxf8ejHJ4wz8BqGUMYlnzNBer5NvGg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.59.3.tgz", + "integrity": "sha512-HPwA+hVkfcriajbNvTmZv4VRauibay+cWArYUYq7u7W7PmGShMxbPxLvrwDme55a6d5alG3nrYfhyJ/G28XlLg==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@typescript-eslint/scope-manager": "8.59.3", + "@typescript-eslint/types": "8.59.3", + "@typescript-eslint/typescript-estree": "8.59.3", + "@typescript-eslint/visitor-keys": "8.59.3", + "debug": "^4.4.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/project-service": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.59.3.tgz", + "integrity": "sha512-ECiUWa/KYRGDFUqTNehaRgzDshnJfkTABJxVemHk4ko22gcr0ukloKjWvyQ64g8YCV/UI47kN1dbmjf/GaQYng==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/tsconfig-utils": "^8.59.3", + "@typescript-eslint/types": "^8.59.3", + "debug": "^4.4.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.59.3.tgz", + "integrity": "sha512-t2LvZnoEfzKtnPjgeEu41xw5gxq9mQVfYy4OoZ4Vlt0sk3JwxmhCca/AR7DwOiHrjWgjAj6as4AhRLKSDfvZIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.59.3", + "@typescript-eslint/visitor-keys": "8.59.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/tsconfig-utils": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.59.3.tgz", + "integrity": "sha512-PcIJHjmaREXLgIAIzLnSY9VucEzz8FKXsRgFa1DmdGCK/5tJpW03TKJF01Q6VZd1lLdz2sIKPWaDUZN9dp//dw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.59.3.tgz", + "integrity": "sha512-g71d8QD8UaiHGvrJwyIS1hCX5r63w6Jll+4VEYhEAHXTDIqX1JgxhTAbEHtKntL9kuc4jRo7/GWw5xfCepSccQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.59.3", + "@typescript-eslint/typescript-estree": "8.59.3", + "@typescript-eslint/utils": "8.59.3", + "debug": "^4.4.3", + "ts-api-utils": "^2.5.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/types": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.59.3.tgz", + "integrity": "sha512-ePFoH0g4ludssdRFqqDxQePCxU4WQyRa9+XVwjm7yLn0FKhMeoetC+qBEEI1Eyb1pGSDveTIT09Bvw2WhlGayg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.59.3.tgz", + "integrity": "sha512-CbRjVRAf7Lr9Kr8RopKcbY45p2VfmmHrm0ygOCYFi7oU8q19m0Fs/6iHS7kNOmwpp+ob07ZVcAqlxUod9lYdmg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/project-service": "8.59.3", + "@typescript-eslint/tsconfig-utils": "8.59.3", + "@typescript-eslint/types": "8.59.3", + "@typescript-eslint/visitor-keys": "8.59.3", + "debug": "^4.4.3", + "minimatch": "^10.2.2", + "semver": "^7.7.3", + "tinyglobby": "^0.2.15", + "ts-api-utils": "^2.5.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": { + "version": "7.8.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.8.0.tgz", + "integrity": "sha512-AcM7dV/5ul4EekoQ29Agm5vri8JNqRyj39o0qpX6vDF2GZrtutZl5RwgD1XnZjiTAfncsJhMI48QQH3sN87YNA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.59.3.tgz", + "integrity": "sha512-JAvT14goBzRzzzZyqq3P9BLArIxTtQURUtFgQ/V7FO+eU+Gg6ES+5ymOPP1wRxXcxAYeivCk4uS3jCKWI1K8Zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.9.1", + "@typescript-eslint/scope-manager": "8.59.3", + "@typescript-eslint/types": "8.59.3", + "@typescript-eslint/typescript-estree": "8.59.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.59.3.tgz", + "integrity": "sha512-f1UQF7ggd42YiwI5wGrRaPsa+P0CINBlrkLPmGfpq/u/I/oVtecoEIfFR9ag/oa1sLOsRNZ6xehf6qMZhQGBDg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.59.3", + "eslint-visitor-keys": "^5.0.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@vitejs/plugin-react": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.2.tgz", + "integrity": "sha512-DlSMqo4WhThw4vB8Mpn0Woe9J+Jfq1geJ61AKW0QEgLzGMNwtIMdxbDUzLxcun8W7NbJO0e2Jg/Nxm3cCSVzzg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@rolldown/pluginutils": "^1.0.0" + }, + "engines": { + "node": "^20.19.0 || >=22.12.0" + }, + "peerDependencies": { + "@rolldown/plugin-babel": "^0.1.7 || ^0.2.0", + "babel-plugin-react-compiler": "^1.0.0", + "vite": "^8.0.0" + }, + "peerDependenciesMeta": { + "@rolldown/plugin-babel": { + "optional": true + }, + "babel-plugin-react-compiler": { + "optional": true + } + } + }, + "node_modules/@vitejs/plugin-react/node_modules/@rolldown/pluginutils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.1.tgz", + "integrity": "sha512-2j9bGt5Jh8hj+vPtgzPtl72j0yRxHAyumoo6TNfAjsLB04UtpSvPbPcDcBMxz7n+9CYB0c1GxQFxYRg2jimqGw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@vitest/expect": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/expect/-/expect-4.1.6.tgz", + "integrity": "sha512-7EHDquPthALSV0jhhjgEW8FXaviMx7rSqu8W6oqCoAuOhKov814P99QDV1pxMA3QPv21YudvJngIhjrNI4opLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@standard-schema/spec": "^1.1.0", + "@types/chai": "^5.2.2", + "@vitest/spy": "4.1.6", + "@vitest/utils": "4.1.6", + "chai": "^6.2.2", + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/pretty-format": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-4.1.6.tgz", + "integrity": "sha512-h5SxD/IzNhZYnrSZRsUZQIC+vD0GY8cUvq0iwsmkFKixRCKLLWqCXa/FIQ4S1R+sI+PGoojkHsdNrbZiM9Qpgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/runner": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/runner/-/runner-4.1.6.tgz", + "integrity": "sha512-nOPCmn2+yD0ZNmKdsXGv/UxMMWbMuKeD6GyYncNwdkYDxpQvrPSKYj2rWuDjC2Y4b6w6hjip5dBKFzEUuZe3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/utils": "4.1.6", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/snapshot": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/snapshot/-/snapshot-4.1.6.tgz", + "integrity": "sha512-YhsdE6xAVfTDmzjxL2ZDUvjj+ZsgyOKe+TdQzqkD72wIOmHka8NuGQ6NpTNZv9D2Z63fbwWKJPeVpEw4EQgYxw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "4.1.6", + "@vitest/utils": "4.1.6", + "magic-string": "^0.30.21", + "pathe": "^2.0.3" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/spy": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/spy/-/spy-4.1.6.tgz", + "integrity": "sha512-JFKxMx6udhwKh/Ldo270e17QX710vgunMkuPAvXjHSvC6oqLWAHhVhjg/I71q0u0CBSErIODV1Kjv0FQNSWjdg==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/@vitest/utils": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/utils/-/utils-4.1.6.tgz", + "integrity": "sha512-FxIY+U81R3LGKCxaHHFRQ5+g6/iRgGLmeHWdp2Amj4ljQRrEIWHmZyDfDYBRZlpyqA7qKxtS9DD1dhk8RnRIVQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/pretty-format": "4.1.6", + "convert-source-map": "^2.0.0", + "tinyrainbow": "^3.1.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + } + }, + "node_modules/acorn": { + "version": "8.16.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", + "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", + "dev": true, + "license": "MIT", + "peer": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/ajv": { + "version": "6.15.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.15.0.tgz", + "integrity": "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw==", + "dev": true, + "license": "MIT", + "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/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/aria-query": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "dequal": "^2.0.3" + } + }, + "node_modules/assertion-error": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/assertion-error/-/assertion-error-2.0.1.tgz", + "integrity": "sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + } + }, + "node_modules/balanced-match": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", + "dev": true, + "license": "MIT", + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/baseline-browser-mapping": { + "version": "2.10.30", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.30.tgz", + "integrity": "sha512-xjOFN16Ha1+Rz4nFYKqHU/LSB+gx/Vi3yQLX7r7sAW+Wa+8hhF2h4pvqTrTMc8+WcDBEunnUurr46Jvv0jk3Vg==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.cjs" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/bidi-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/bidi-js/-/bidi-js-1.0.3.tgz", + "integrity": "sha512-RKshQI1R3YQ+n9YJz2QQ147P66ELpa1FQEg20Dk8oW9t2KgLbpDLLp9aGZ7y8WHSshDknG0bknqGw5/tyCs5tw==", + "dev": true, + "license": "MIT", + "dependencies": { + "require-from-string": "^2.0.2" + } + }, + "node_modules/brace-expansion": { + "version": "5.0.6", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz", + "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^4.0.2" + }, + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/browserslist": { + "version": "4.28.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz", + "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==", + "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" + } + ], + "license": "MIT", + "peer": true, + "dependencies": { + "baseline-browser-mapping": "^2.10.12", + "caniuse-lite": "^1.0.30001782", + "electron-to-chromium": "^1.5.328", + "node-releases": "^2.0.36", + "update-browserslist-db": "^1.2.3" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001792", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001792.tgz", + "integrity": "sha512-hVLMUZFgR4JJ6ACt1uEESvQN1/dBVqPAKY0hgrV70eN3391K6juAfTjKZLKvOMsx8PxA7gsY1/tLMMTcfFLLpw==", + "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" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chai": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/chai/-/chai-6.2.2.tgz", + "integrity": "sha512-NUPRluOfOiTKBKvWPtSD4PhFvWCqOi0BGStNWs57X9js7XGTprSmFoz5F0tWhR4WPjNeR9jXqdC7/UpSJTnlRg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/css-tree": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-3.2.1.tgz", + "integrity": "sha512-X7sjQzceUhu1u7Y/ylrRZFU2FS6LRiFVp6rKLPg23y3x3c3DOKAwuXGDp+PAGjh6CSnCjYeAul8pcT8bAl+lSA==", + "dev": true, + "license": "MIT", + "dependencies": { + "mdn-data": "2.27.1", + "source-map-js": "^1.2.1" + }, + "engines": { + "node": "^10 || ^12.20.0 || ^14.13.0 || >=15.0.0" } }, "node_modules/csstype": { @@ -1255,45 +2088,438 @@ "dev": true, "license": "MIT" }, - "node_modules/esbuild": { - "version": "0.21.5", - "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", - "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==", + "node_modules/data-urls": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-7.0.0.tgz", + "integrity": "sha512-23XHcCF+coGYevirZceTVD7NdJOqVn+49IHyxgszm+JIiHLoB2TkmPtsYkNWT1pvRSGkc35L6NHs0yHkN2SumA==", "dev": true, - "hasInstallScript": true, "license": "MIT", - "bin": { - "esbuild": "bin/esbuild" + "dependencies": { + "whatwg-mimetype": "^5.0.0", + "whatwg-url": "^16.0.0" }, "engines": { - "node": ">=12" - }, - "optionalDependencies": { - "@esbuild/aix-ppc64": "0.21.5", - "@esbuild/android-arm": "0.21.5", - "@esbuild/android-arm64": "0.21.5", - "@esbuild/android-x64": "0.21.5", - "@esbuild/darwin-arm64": "0.21.5", - "@esbuild/darwin-x64": "0.21.5", - "@esbuild/freebsd-arm64": "0.21.5", - "@esbuild/freebsd-x64": "0.21.5", - "@esbuild/linux-arm": "0.21.5", - "@esbuild/linux-arm64": "0.21.5", - "@esbuild/linux-ia32": "0.21.5", - "@esbuild/linux-loong64": "0.21.5", - "@esbuild/linux-mips64el": "0.21.5", - "@esbuild/linux-ppc64": "0.21.5", - "@esbuild/linux-riscv64": "0.21.5", - "@esbuild/linux-s390x": "0.21.5", - "@esbuild/linux-x64": "0.21.5", - "@esbuild/netbsd-x64": "0.21.5", - "@esbuild/openbsd-x64": "0.21.5", - "@esbuild/sunos-x64": "0.21.5", - "@esbuild/win32-arm64": "0.21.5", - "@esbuild/win32-ia32": "0.21.5", - "@esbuild/win32-x64": "0.21.5" + "node": "^20.19.0 || ^22.12.0 || >=24.0.0" } }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decimal.js": { + "version": "10.6.0", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.6.0.tgz", + "integrity": "sha512-YpgQiITW3JXGntzdUmyUR1V812Hn8T1YVXhCu+wO3OpS4eU9l4YdD3qjyiKdV6mvV29zapkMeD390UVEf2lkUg==", + "dev": true, + "license": "MIT" + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/detect-libc": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz", + "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=8" + } + }, + "node_modules/dom-accessibility-api": { + "version": "0.5.16", + "resolved": "https://registry.npmjs.org/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz", + "integrity": "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg==", + "dev": true, + "license": "MIT" + }, + "node_modules/electron-to-chromium": { + "version": "1.5.357", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.357.tgz", + "integrity": "sha512-NHlTIQDK8fmVwHwuIzmXYEJ1Ewq3D9wDNc0cWXxDGysP6Pb21giwGNkxiTifyKy/4SoPuN5l6GLP1W9Sv7zB2g==", + "dev": true, + "license": "ISC" + }, + "node_modules/entities": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-8.0.0.tgz", + "integrity": "sha512-zwfzJecQ/Uej6tusMqwAqU/6KL2XaB2VZ2Jg54Je6ahNBGNH6Ek6g3jjNCF0fG9EWQKGZNddNjU5F1ZQn/sBnA==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=20.19.0" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/es-module-lexer": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/es-module-lexer/-/es-module-lexer-2.1.0.tgz", + "integrity": "sha512-n27zTYMjYu1aj4MjCWzSP7G9r75utsaoc8m61weK+W8JMBGGQybd43GstCXZ3WNmSFtGT9wi59qQTW6mhTR5LQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-10.4.0.tgz", + "integrity": "sha512-loXy6bWOoP3EP6JA7jo6p5jMpBJmHmsNZM5SFRHLdh1MGOPurMnNBj4ZlAbaqUAaQWbCr7jHV4P7gzAyryZWkQ==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.2", + "@eslint/config-array": "^0.23.5", + "@eslint/config-helpers": "^0.6.0", + "@eslint/core": "^1.2.1", + "@eslint/plugin-kit": "^0.7.1", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "ajv": "^6.14.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^9.1.2", + "eslint-visitor-keys": "^5.0.1", + "espree": "^11.2.0", + "esquery": "^1.7.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "minimatch": "^10.2.4", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true + } + } + }, + "node_modules/eslint-config-prettier": { + "version": "10.1.8", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-10.1.8.tgz", + "integrity": "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w==", + "dev": true, + "license": "MIT", + "bin": { + "eslint-config-prettier": "bin/cli.js" + }, + "funding": { + "url": "https://opencollective.com/eslint-config-prettier" + }, + "peerDependencies": { + "eslint": ">=7.0.0" + } + }, + "node_modules/eslint-plugin-react-hooks": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.1.1.tgz", + "integrity": "sha512-f2I7Gw6JbvCexzIInuSbZpfdQ44D7iqdWX01FKLvrPgqxoE7oMj8clOfto8U6vYiz4yd5oKu39rRSVOe1zRu0g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.24.4", + "@babel/parser": "^7.24.4", + "hermes-parser": "^0.25.1", + "zod": "^3.25.0 || ^4.0.0", + "zod-validation-error": "^3.5.0 || ^4.0.0" + }, + "engines": { + "node": ">=18" + }, + "peerDependencies": { + "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 || ^10.0.0" + } + }, + "node_modules/eslint-plugin-react-refresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.5.2.tgz", + "integrity": "sha512-hmgTH57GfzoTFjVN0yBwTggnsVUF2tcqi7RJZHqi9lIezSs4eFyAMktA68YD4r5kNw1mxyY4dmkyoFDb3FIqrA==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "eslint": "^9 || ^10" + } + }, + "node_modules/eslint-scope": { + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-9.1.2.tgz", + "integrity": "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "@types/esrecurse": "^4.3.1", + "@types/estree": "^1.0.8", + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", + "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree": { + "version": "11.2.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-11.2.0.tgz", + "integrity": "sha512-7p3DrVEIopW1B1avAGLuCSh1jubc01H2JHc8B4qqGblmg5gI9yumBgACjWo4JlIc04ufug4xJ3SQI8HkS/Rgzw==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.16.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^5.0.1" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "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, + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "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, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estree-walker": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", + "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/estree": "^1.0.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expect-type": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/expect-type/-/expect-type-1.3.0.tgz", + "integrity": "sha512-knvyeauYhqjOYvQ66MznSMs83wmHrCycNEN6Ao+2AeYEfxUIkuiVxdEa1qlGEPK+We3n0THiDciYSsCcgW/DoA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.0.0" + } + }, + "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, + "license": "MIT" + }, + "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, + "license": "MIT" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fdir": { + "version": "6.5.0", + "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz", + "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12.0.0" + }, + "peerDependencies": { + "picomatch": "^3 || ^4" + }, + "peerDependenciesMeta": { + "picomatch": { + "optional": true + } + } + }, + "node_modules/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", + "dev": true, + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/flatted": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz", + "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==", + "dev": true, + "license": "ISC" + }, "node_modules/fsevents": { "version": "2.3.3", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", @@ -1309,12 +2535,525 @@ "node": "^8.16.0 || ^10.6.0 || >=11.0.0" } }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "17.6.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-17.6.0.tgz", + "integrity": "sha512-sepffkT8stwnIYbsMBpoCHJuJM5l98FUF2AnE07hfvE0m/qp3R586hw4jF4uadbhvg1ooIdzuu7CsfD2jzCaNA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/hermes-estree": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz", + "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==", + "dev": true, + "license": "MIT" + }, + "node_modules/hermes-parser": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz", + "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "hermes-estree": "0.25.1" + } + }, + "node_modules/html-encoding-sniffer": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-6.0.0.tgz", + "integrity": "sha512-CV9TW3Y3f8/wT0BRFc1/KAVQ3TUHiXmaAb6VW9vtiMFf7SLoMd1PdAc4W3KFOFETBJUb90KatHqlsZMWV+R9Gg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@exodus/bytes": "^1.6.0" + }, + "engines": { + "node": "^20.19.0 || ^22.12.0 || >=24.0.0" + } + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-potential-custom-element-name": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", + "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, "node_modules/js-tokens": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", "license": "MIT" }, + "node_modules/jsdom": { + "version": "29.1.1", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-29.1.1.tgz", + "integrity": "sha512-ECi4Fi2f7BdJtUKTflYRTiaMxIB0O6zfR1fX0GXpUrf6flp8QIYn1UT20YQqdSOfk2dfkCwS8LAFoJDEppNK5Q==", + "dev": true, + "license": "MIT", + "peer": true, + "dependencies": { + "@asamuzakjp/css-color": "^5.1.11", + "@asamuzakjp/dom-selector": "^7.1.1", + "@bramus/specificity": "^2.4.2", + "@csstools/css-syntax-patches-for-csstree": "^1.1.3", + "@exodus/bytes": "^1.15.0", + "css-tree": "^3.2.1", + "data-urls": "^7.0.0", + "decimal.js": "^10.6.0", + "html-encoding-sniffer": "^6.0.0", + "is-potential-custom-element-name": "^1.0.1", + "lru-cache": "^11.3.5", + "parse5": "^8.0.1", + "saxes": "^6.0.0", + "symbol-tree": "^3.2.4", + "tough-cookie": "^6.0.1", + "undici": "^7.25.0", + "w3c-xmlserializer": "^5.0.0", + "webidl-conversions": "^8.0.1", + "whatwg-mimetype": "^5.0.0", + "whatwg-url": "^16.0.1", + "xml-name-validator": "^5.0.0" + }, + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24.0.0" + }, + "peerDependencies": { + "canvas": "^3.0.0" + }, + "peerDependenciesMeta": { + "canvas": { + "optional": true + } + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT" + }, + "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, + "license": "MIT" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/lightningcss": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz", + "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==", + "dev": true, + "license": "MPL-2.0", + "dependencies": { + "detect-libc": "^2.0.3" + }, + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + }, + "optionalDependencies": { + "lightningcss-android-arm64": "1.32.0", + "lightningcss-darwin-arm64": "1.32.0", + "lightningcss-darwin-x64": "1.32.0", + "lightningcss-freebsd-x64": "1.32.0", + "lightningcss-linux-arm-gnueabihf": "1.32.0", + "lightningcss-linux-arm64-gnu": "1.32.0", + "lightningcss-linux-arm64-musl": "1.32.0", + "lightningcss-linux-x64-gnu": "1.32.0", + "lightningcss-linux-x64-musl": "1.32.0", + "lightningcss-win32-arm64-msvc": "1.32.0", + "lightningcss-win32-x64-msvc": "1.32.0" + } + }, + "node_modules/lightningcss-android-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz", + "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-arm64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz", + "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-darwin-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz", + "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-freebsd-x64": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz", + "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm-gnueabihf": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz", + "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz", + "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-arm64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz", + "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-gnu": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz", + "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-linux-x64-musl": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz", + "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-arm64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz", + "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/lightningcss-win32-x64-msvc": { + "version": "1.32.0", + "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz", + "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MPL-2.0", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">= 12.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/parcel" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/loose-envify": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", @@ -1327,6 +3066,66 @@ "loose-envify": "cli.js" } }, + "node_modules/lru-cache": { + "version": "11.3.6", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-11.3.6.tgz", + "integrity": "sha512-Gf/KoL3C/MlI7Bt0PGI9I+TeTC/I6r/csU58N4BSNc4lppLBeKsOdFYkK+dX0ABDUMJNfCHTyPpzwwO21Awd3A==", + "dev": true, + "license": "BlueOak-1.0.0", + "engines": { + "node": "20 || >=22" + } + }, + "node_modules/lz-string": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/lz-string/-/lz-string-1.5.0.tgz", + "integrity": "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==", + "dev": true, + "license": "MIT", + "bin": { + "lz-string": "bin/bin.js" + } + }, + "node_modules/magic-string": { + "version": "0.30.21", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz", + "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.5" + } + }, + "node_modules/mdn-data": { + "version": "2.27.1", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.27.1.tgz", + "integrity": "sha512-9Yubnt3e8A0OKwxYSXyhLymGW4sCufcLG6VdiDdUGVkPhpqLxlvP5vl1983gQjJl3tqbrM731mjaZaP68AgosQ==", + "dev": true, + "license": "CC0-1.0" + }, + "node_modules/minimatch": { + "version": "10.2.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz", + "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==", + "dev": true, + "license": "BlueOak-1.0.0", + "dependencies": { + "brace-expansion": "^5.0.5" + }, + "engines": { + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, "node_modules/nanoid": { "version": "3.3.11", "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz", @@ -1346,12 +3145,127 @@ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" } }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.44", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.44.tgz", + "integrity": "sha512-5WUyunoPMsvvEhS8AxHtRzP+oA8UCkJ7YRxatWKjngndhDGLiqEVAQKWjFAiAiuL8zMRGzGSJxFnLetoa43qGQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/obug": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/obug/-/obug-2.1.1.tgz", + "integrity": "sha512-uTqF9MuPraAQ+IsnPf366RG4cP9RtUi7MLO1N3KEc+wb0a6yKpeL0lmk2IB1jY5KHPAlTc6T/JRdC/YqxHNwkQ==", + "dev": true, + "funding": [ + "https://github.com/sponsors/sxzz", + "https://opencollective.com/debug" + ], + "license": "MIT" + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/pako": { "version": "1.0.11", "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", "license": "(MIT AND Zlib)" }, + "node_modules/parse5": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-8.0.1.tgz", + "integrity": "sha512-z1e/HMG90obSGeidlli3hj7cbocou0/wa5HacvI3ASx34PecNjNQeaHNo5WIZpWofN9kgkqV1q5YvXe3F0FoPw==", + "dev": true, + "license": "MIT", + "dependencies": { + "entities": "^8.0.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, + "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, + "license": "MIT", + "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, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/pathe": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/pathe/-/pathe-2.0.3.tgz", + "integrity": "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w==", + "dev": true, + "license": "MIT" + }, "node_modules/pdf-lib": { "version": "1.17.1", "resolved": "https://registry.npmjs.org/pdf-lib/-/pdf-lib-1.17.1.tgz", @@ -1383,6 +3297,20 @@ "dev": true, "license": "ISC" }, + "node_modules/picomatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz", + "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==", + "dev": true, + "license": "MIT", + "peer": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, "node_modules/postcss": { "version": "8.5.14", "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.14.tgz", @@ -1412,6 +3340,57 @@ "node": "^10 || ^12 || >=14" } }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/prettier": { + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.8.3.tgz", + "integrity": "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw==", + "dev": true, + "license": "MIT", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, + "node_modules/pretty-format": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1", + "ansi-styles": "^5.0.0", + "react-is": "^17.0.1" + }, + "engines": { + "node": "^10.13.0 || ^12.13.0 || ^14.15.0 || >=15.0.0" + } + }, + "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, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, "node_modules/react": { "version": "18.3.1", "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz", @@ -1430,6 +3409,7 @@ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz", "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==", "license": "MIT", + "peer": true, "dependencies": { "loose-envify": "^1.1.0", "scheduler": "^0.23.2" @@ -1438,49 +3418,75 @@ "react": "^18.3.1" } }, - "node_modules/rollup": { - "version": "4.60.4", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.60.4.tgz", - "integrity": "sha512-WHeFSbZYsPu3+bLoNRUuAO+wavNlocOPf3wSHTP7hcFKVnJeWsYlCDbr3mTS14FCizf9ccIxXA8sGL8zKeQN3g==", + "node_modules/react-is": { + "version": "17.0.2", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", + "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", + "dev": true, + "license": "MIT" + }, + "node_modules/require-from-string": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/require-from-string/-/require-from-string-2.0.2.tgz", + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/rolldown": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.1.tgz", + "integrity": "sha512-X0KQHljNnEkWNqqiz9zJrGunh1B0HgOxLXvnFpCOcadzcy5qohZ3tqMEUg00vncoRovXuK3ZqCT9KnnKzoInFQ==", "dev": true, "license": "MIT", "dependencies": { - "@types/estree": "1.0.8" + "@oxc-project/types": "=0.130.0", + "@rolldown/pluginutils": "^1.0.0" }, "bin": { - "rollup": "dist/bin/rollup" + "rolldown": "bin/cli.mjs" }, "engines": { - "node": ">=18.0.0", - "npm": ">=8.0.0" + "node": "^20.19.0 || >=22.12.0" }, "optionalDependencies": { - "@rollup/rollup-android-arm-eabi": "4.60.4", - "@rollup/rollup-android-arm64": "4.60.4", - "@rollup/rollup-darwin-arm64": "4.60.4", - "@rollup/rollup-darwin-x64": "4.60.4", - "@rollup/rollup-freebsd-arm64": "4.60.4", - "@rollup/rollup-freebsd-x64": "4.60.4", - "@rollup/rollup-linux-arm-gnueabihf": "4.60.4", - "@rollup/rollup-linux-arm-musleabihf": "4.60.4", - "@rollup/rollup-linux-arm64-gnu": "4.60.4", - "@rollup/rollup-linux-arm64-musl": "4.60.4", - "@rollup/rollup-linux-loong64-gnu": "4.60.4", - "@rollup/rollup-linux-loong64-musl": "4.60.4", - "@rollup/rollup-linux-ppc64-gnu": "4.60.4", - "@rollup/rollup-linux-ppc64-musl": "4.60.4", - "@rollup/rollup-linux-riscv64-gnu": "4.60.4", - "@rollup/rollup-linux-riscv64-musl": "4.60.4", - "@rollup/rollup-linux-s390x-gnu": "4.60.4", - "@rollup/rollup-linux-x64-gnu": "4.60.4", - "@rollup/rollup-linux-x64-musl": "4.60.4", - "@rollup/rollup-openbsd-x64": "4.60.4", - "@rollup/rollup-openharmony-arm64": "4.60.4", - "@rollup/rollup-win32-arm64-msvc": "4.60.4", - "@rollup/rollup-win32-ia32-msvc": "4.60.4", - "@rollup/rollup-win32-x64-gnu": "4.60.4", - "@rollup/rollup-win32-x64-msvc": "4.60.4", - "fsevents": "~2.3.2" + "@rolldown/binding-android-arm64": "1.0.1", + "@rolldown/binding-darwin-arm64": "1.0.1", + "@rolldown/binding-darwin-x64": "1.0.1", + "@rolldown/binding-freebsd-x64": "1.0.1", + "@rolldown/binding-linux-arm-gnueabihf": "1.0.1", + "@rolldown/binding-linux-arm64-gnu": "1.0.1", + "@rolldown/binding-linux-arm64-musl": "1.0.1", + "@rolldown/binding-linux-ppc64-gnu": "1.0.1", + "@rolldown/binding-linux-s390x-gnu": "1.0.1", + "@rolldown/binding-linux-x64-gnu": "1.0.1", + "@rolldown/binding-linux-x64-musl": "1.0.1", + "@rolldown/binding-openharmony-arm64": "1.0.1", + "@rolldown/binding-wasm32-wasi": "1.0.1", + "@rolldown/binding-win32-arm64-msvc": "1.0.1", + "@rolldown/binding-win32-x64-msvc": "1.0.1" + } + }, + "node_modules/rolldown/node_modules/@rolldown/pluginutils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.1.tgz", + "integrity": "sha512-2j9bGt5Jh8hj+vPtgzPtl72j0yRxHAyumoo6TNfAjsLB04UtpSvPbPcDcBMxz7n+9CYB0c1GxQFxYRg2jimqGw==", + "dev": true, + "license": "MIT" + }, + "node_modules/saxes": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/saxes/-/saxes-6.0.0.tgz", + "integrity": "sha512-xAg7SOnEhrm5zI3puOOKyy1OMcMlIJZYNJY7xLBwSze0UjhPLnWfj2GF2EpT0jmzaJKIWKHLsaSSajf35bcYnA==", + "dev": true, + "license": "ISC", + "dependencies": { + "xmlchars": "^2.2.0" + }, + "engines": { + "node": ">=v12.22.7" } }, "node_modules/scheduler": { @@ -1492,6 +3498,46 @@ "loose-envify": "^1.1.0" } }, + "node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "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, + "license": "MIT", + "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, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/siginfo": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/siginfo/-/siginfo-2.0.0.tgz", + "integrity": "sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g==", + "dev": true, + "license": "ISC" + }, "node_modules/source-map-js": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", @@ -1502,18 +3548,156 @@ "node": ">=0.10.0" } }, + "node_modules/stackback": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/stackback/-/stackback-0.0.2.tgz", + "integrity": "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw==", + "dev": true, + "license": "MIT" + }, + "node_modules/std-env": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/std-env/-/std-env-4.1.0.tgz", + "integrity": "sha512-Rq7ybcX2RuC55r9oaPVEW7/xu3tj8u4GeBYHBWCychFtzMIr86A7e3PPEBPT37sHStKX3+TiX/Fr/ACmJLVlLQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/symbol-tree": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", + "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinybench": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/tinybench/-/tinybench-2.9.0.tgz", + "integrity": "sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg==", + "dev": true, + "license": "MIT" + }, + "node_modules/tinyexec": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-1.1.2.tgz", + "integrity": "sha512-dAqSqE/RabpBKI8+h26GfLq6Vb3JVXs30XYQjdMjaj/c2tS8IYYMbIzP599KtRj7c57/wYApb3QjgRgXmrCukA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + } + }, + "node_modules/tinyglobby": { + "version": "0.2.16", + "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz", + "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==", + "dev": true, + "license": "MIT", + "dependencies": { + "fdir": "^6.5.0", + "picomatch": "^4.0.4" + }, + "engines": { + "node": ">=12.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/SuperchupuDev" + } + }, + "node_modules/tinyrainbow": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/tinyrainbow/-/tinyrainbow-3.1.0.tgz", + "integrity": "sha512-Bf+ILmBgretUrdJxzXM0SgXLZ3XfiaUuOj/IKQHuTXip+05Xn+uyEYdVg0kYDipTBcLrCVyUzAPz7QmArb0mmw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=14.0.0" + } + }, + "node_modules/tldts": { + "version": "7.0.30", + "resolved": "https://registry.npmjs.org/tldts/-/tldts-7.0.30.tgz", + "integrity": "sha512-ELrFxuqsDdHUwoh0XxDbxuLD3Wnz49Z57IFvTtvWy1hJdcMZjXLIuonjilCiWHlT2GbE4Wlv1wKVTzDFnXH1aw==", + "dev": true, + "license": "MIT", + "dependencies": { + "tldts-core": "^7.0.30" + }, + "bin": { + "tldts": "bin/cli.js" + } + }, + "node_modules/tldts-core": { + "version": "7.0.30", + "resolved": "https://registry.npmjs.org/tldts-core/-/tldts-core-7.0.30.tgz", + "integrity": "sha512-uiHN8PIB1VmWyS98eZYja4xzlYqeFZVjb4OuYlJQnZAuJhMw4PbKQOKgHKhBdJR3FE/t5mUQ1Kd80++B+qhD1Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/tough-cookie": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-6.0.1.tgz", + "integrity": "sha512-LktZQb3IeoUWB9lqR5EWTHgW/VTITCXg4D21M+lvybRVdylLrRMnqaIONLVb5mav8vM19m44HIcGq4qASeu2Qw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tldts": "^7.0.5" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/tr46": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-6.0.0.tgz", + "integrity": "sha512-bLVMLPtstlZ4iMQHpFHTR7GAGj2jxi8Dg0s2h2MafAE4uSWF98FC/3MomU51iQAMf8/qDUbKWf5GxuvvVcXEhw==", + "dev": true, + "license": "MIT", + "dependencies": { + "punycode": "^2.3.1" + }, + "engines": { + "node": ">=20" + } + }, + "node_modules/ts-api-utils": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz", + "integrity": "sha512-OJ/ibxhPlqrMM0UiNHJ/0CKQkoKF243/AEmplt3qpRgkW8VG7IfOS41h7V8TjITqdByHzrjcS/2si+y4lIh8NA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.12" + }, + "peerDependencies": { + "typescript": ">=4.8.4" + } + }, "node_modules/tslib": { "version": "1.14.1", "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", "license": "0BSD" }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, "node_modules/typescript": { "version": "5.9.3", "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz", "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==", "dev": true, "license": "Apache-2.0", + "peer": true, "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" @@ -1522,23 +3706,107 @@ "node": ">=14.17" } }, + "node_modules/typescript-eslint": { + "version": "8.59.3", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-8.59.3.tgz", + "integrity": "sha512-KgusgyDgG4LI8Ih/sWaCtZ06tckLAS5CvT5A4D1Q7bYVoAAyzwiZvE4BmwDHkhRVkvhRBepKeASoFzQetha7Fg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/eslint-plugin": "8.59.3", + "@typescript-eslint/parser": "8.59.3", + "@typescript-eslint/typescript-estree": "8.59.3", + "@typescript-eslint/utils": "8.59.3" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", + "typescript": ">=4.8.4 <6.1.0" + } + }, + "node_modules/undici": { + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/undici/-/undici-7.25.0.tgz", + "integrity": "sha512-xXnp4kTyor2Zq+J1FfPI6Eq3ew5h6Vl0F/8d9XU5zZQf1tX9s2Su1/3PiMmUANFULpmksxkClamIZcaUqryHsQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=20.18.1" + } + }, + "node_modules/undici-types": { + "version": "7.24.6", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.24.6.tgz", + "integrity": "sha512-WRNW+sJgj5OBN4/0JpHFqtqzhpbnV0GuB+OozA9gCL7a993SmU+1JBZCzLNxYsbMfIeDL+lTsphD5jN5N+n0zg==", + "dev": true, + "license": "MIT" + }, + "node_modules/update-browserslist-db": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", + "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", + "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" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "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, + "license": "BSD-2-Clause", + "dependencies": { + "punycode": "^2.1.0" + } + }, "node_modules/vite": { - "version": "5.4.21", - "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.21.tgz", - "integrity": "sha512-o5a9xKjbtuhY6Bi5S3+HvbRERmouabWbyUcpXXUA1u+GNUKoROi9byOJ8M0nHbHYHkYICiMlqxkg1KkYmm25Sw==", + "version": "8.0.13", + "resolved": "https://registry.npmjs.org/vite/-/vite-8.0.13.tgz", + "integrity": "sha512-MFtjBYgzmSxmgA4RAfjIyXWpGe1oALnjgUTzzV7QLx/TKxCzjtMH6Fd9/eVK+5Fg1qNoz5VAwsmMs/NofrmJvw==", "dev": true, "license": "MIT", "peer": true, "dependencies": { - "esbuild": "^0.21.3", - "postcss": "^8.4.43", - "rollup": "^4.20.0" + "lightningcss": "^1.32.0", + "picomatch": "^4.0.4", + "postcss": "^8.5.14", + "rolldown": "1.0.1", + "tinyglobby": "^0.2.16" }, "bin": { "vite": "bin/vite.js" }, "engines": { - "node": "^18.0.0 || >=20.0.0" + "node": "^20.19.0 || >=22.12.0" }, "funding": { "url": "https://github.com/vitejs/vite?sponsor=1" @@ -1547,23 +3815,33 @@ "fsevents": "~2.3.3" }, "peerDependencies": { - "@types/node": "^18.0.0 || >=20.0.0", - "less": "*", - "lightningcss": "^1.21.0", - "sass": "*", - "sass-embedded": "*", - "stylus": "*", - "sugarss": "*", - "terser": "^5.4.0" + "@types/node": "^20.19.0 || >=22.12.0", + "@vitejs/devtools": "^0.1.18", + "esbuild": "^0.27.0 || ^0.28.0", + "jiti": ">=1.21.0", + "less": "^4.0.0", + "sass": "^1.70.0", + "sass-embedded": "^1.70.0", + "stylus": ">=0.54.8", + "sugarss": "^5.0.0", + "terser": "^5.16.0", + "tsx": "^4.8.1", + "yaml": "^2.4.2" }, "peerDependenciesMeta": { "@types/node": { "optional": true }, - "less": { + "@vitejs/devtools": { "optional": true }, - "lightningcss": { + "esbuild": { + "optional": true + }, + "jiti": { + "optional": true + }, + "less": { "optional": true }, "sass": { @@ -1580,8 +3858,283 @@ }, "terser": { "optional": true + }, + "tsx": { + "optional": true + }, + "yaml": { + "optional": true } } + }, + "node_modules/vitest": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/vitest/-/vitest-4.1.6.tgz", + "integrity": "sha512-6lvjbS3p9b4CrdCmguzbh2/4uoXhGE2q71R4OX5sqF9R1bo9Xd6fGrMAfvp5wnCzlBnFVdCOp6onuTQVbo8iUQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/expect": "4.1.6", + "@vitest/mocker": "4.1.6", + "@vitest/pretty-format": "4.1.6", + "@vitest/runner": "4.1.6", + "@vitest/snapshot": "4.1.6", + "@vitest/spy": "4.1.6", + "@vitest/utils": "4.1.6", + "es-module-lexer": "^2.0.0", + "expect-type": "^1.3.0", + "magic-string": "^0.30.21", + "obug": "^2.1.1", + "pathe": "^2.0.3", + "picomatch": "^4.0.3", + "std-env": "^4.0.0-rc.1", + "tinybench": "^2.9.0", + "tinyexec": "^1.0.2", + "tinyglobby": "^0.2.15", + "tinyrainbow": "^3.1.0", + "vite": "^6.0.0 || ^7.0.0 || ^8.0.0", + "why-is-node-running": "^2.3.0" + }, + "bin": { + "vitest": "vitest.mjs" + }, + "engines": { + "node": "^20.0.0 || ^22.0.0 || >=24.0.0" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "@edge-runtime/vm": "*", + "@opentelemetry/api": "^1.9.0", + "@types/node": "^20.0.0 || ^22.0.0 || >=24.0.0", + "@vitest/browser-playwright": "4.1.6", + "@vitest/browser-preview": "4.1.6", + "@vitest/browser-webdriverio": "4.1.6", + "@vitest/coverage-istanbul": "4.1.6", + "@vitest/coverage-v8": "4.1.6", + "@vitest/ui": "4.1.6", + "happy-dom": "*", + "jsdom": "*", + "vite": "^6.0.0 || ^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "@edge-runtime/vm": { + "optional": true + }, + "@opentelemetry/api": { + "optional": true + }, + "@types/node": { + "optional": true + }, + "@vitest/browser-playwright": { + "optional": true + }, + "@vitest/browser-preview": { + "optional": true + }, + "@vitest/browser-webdriverio": { + "optional": true + }, + "@vitest/coverage-istanbul": { + "optional": true + }, + "@vitest/coverage-v8": { + "optional": true + }, + "@vitest/ui": { + "optional": true + }, + "happy-dom": { + "optional": true + }, + "jsdom": { + "optional": true + }, + "vite": { + "optional": false + } + } + }, + "node_modules/vitest/node_modules/@vitest/mocker": { + "version": "4.1.6", + "resolved": "https://registry.npmjs.org/@vitest/mocker/-/mocker-4.1.6.tgz", + "integrity": "sha512-MCFc63czMjEInOlcY2cpQCvCN+KgbAn+60xu9cMgP4sKaLC5JNAKw7JH8QdAnoAC88hW1IiSNZ+GgVXlN1UcMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@vitest/spy": "4.1.6", + "estree-walker": "^3.0.3", + "magic-string": "^0.30.21" + }, + "funding": { + "url": "https://opencollective.com/vitest" + }, + "peerDependencies": { + "msw": "^2.4.9", + "vite": "^6.0.0 || ^7.0.0 || ^8.0.0" + }, + "peerDependenciesMeta": { + "msw": { + "optional": true + }, + "vite": { + "optional": true + } + } + }, + "node_modules/w3c-xmlserializer": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-5.0.0.tgz", + "integrity": "sha512-o8qghlI8NZHU1lLPrpi2+Uq7abh4GGPpYANlalzWxyWteJOCsr/P+oPBA49TOLu5FTZO4d3F9MnWJfiMo4BkmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "xml-name-validator": "^5.0.0" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/webidl-conversions": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-8.0.1.tgz", + "integrity": "sha512-BMhLD/Sw+GbJC21C/UgyaZX41nPt8bUTg+jWyDeg7e7YN4xOM05YPSIXceACnXVtqyEw/LMClUQMtMZ+PGGpqQ==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=20" + } + }, + "node_modules/whatwg-mimetype": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-5.0.0.tgz", + "integrity": "sha512-sXcNcHOC51uPGF0P/D4NVtrkjSU2fNsm9iog4ZvZJsL3rjoDAzXZhkm2MWt1y+PUdggKAYVoMAIYcs78wJ51Cw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=20" + } + }, + "node_modules/whatwg-url": { + "version": "16.0.1", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-16.0.1.tgz", + "integrity": "sha512-1to4zXBxmXHV3IiSSEInrreIlu02vUOvrhxJJH5vcxYTBDAx51cqZiKdyTxlecdKNSjj8EcxGBxNf6Vg+945gw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@exodus/bytes": "^1.11.0", + "tr46": "^6.0.0", + "webidl-conversions": "^8.0.1" + }, + "engines": { + "node": "^20.19.0 || ^22.12.0 || >=24.0.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, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/why-is-node-running": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/why-is-node-running/-/why-is-node-running-2.3.0.tgz", + "integrity": "sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w==", + "dev": true, + "license": "MIT", + "dependencies": { + "siginfo": "^2.0.0", + "stackback": "0.0.2" + }, + "bin": { + "why-is-node-running": "cli.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/xml-name-validator": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-5.0.0.tgz", + "integrity": "sha512-EvGK8EJ3DhaHfbRlETOWAS5pO9MZITeauHKJyb8wyajUfQUenkIg2MvLDTZ4T/TgIcm3HU0TFBgWWboAZ30UHg==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18" + } + }, + "node_modules/xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", + "dev": true, + "license": "MIT" + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/zod": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/zod/-/zod-4.4.3.tgz", + "integrity": "sha512-ytENFjIJFl2UwYglde2jchW2Hwm4GJFLDiSXWdTrJQBIN9Fcyp7n4DhxJEiWNAJMV1/BqWfW/kkg71UDcHJyTQ==", + "dev": true, + "license": "MIT", + "peer": true, + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } + }, + "node_modules/zod-validation-error": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz", + "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.0.0" + }, + "peerDependencies": { + "zod": "^3.25.0 || ^4.0.0" + } } } } diff --git a/package.json b/package.json index d91b5ff..c08dd00 100644 --- a/package.json +++ b/package.json @@ -1,12 +1,19 @@ { "name": "pdf-tools", - "version": "0.2.0", + "version": "0.2.1", "private": true, + "type": "module", "scripts": { "dev": "vite", "build": "vite build", "preview": "vite preview", - "lint": "echo \"no lint configured\"" + "typecheck": "tsc --noEmit", + "lint": "eslint .", + "format": "prettier --write .", + "format:check": "prettier --check .", + "test": "vitest run --environment jsdom", + "test:watch": "vitest --environment jsdom", + "check": "npm run typecheck && npm run lint && npm run test && npm run build" }, "dependencies": { "pdf-lib": "^1.17.1", @@ -15,11 +22,24 @@ "react-dom": "^18.3.1" }, "devDependencies": { + "@eslint/js": "^10.0.1", + "@testing-library/react": "^16.3.2", + "@testing-library/user-event": "^14.6.1", + "@types/node": "^25.8.0", "@types/react": "^18.3.11", "@types/react-dom": "^18.3.2", - "@vitejs/plugin-react-swc": "^3.7.0", + "@vitejs/plugin-react": "^6.0.2", + "eslint": "^10.4.0", + "eslint-config-prettier": "^10.1.8", + "eslint-plugin-react-hooks": "^7.1.1", + "eslint-plugin-react-refresh": "^0.5.2", + "globals": "^17.6.0", + "jsdom": "^29.1.1", + "prettier": "^3.8.3", "typescript": "^5.6.3", - "vite": "^5.4.10" + "typescript-eslint": "^8.59.3", + "vite": "^8.0.13", + "vitest": "^4.1.6" }, "description": "Browser-only, self-hostable PDF workbench for page-level PDF operations." } diff --git a/pdf-tools.zip b/pdf-tools.zip new file mode 100644 index 0000000000000000000000000000000000000000..b93bc3ebc70a80c6b3ecaa0f27c11699a19e67a4 GIT binary patch literal 103974 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 literal 0 HcmV?d00001 diff --git a/prettierignore.txt b/prettierignore.txt new file mode 100644 index 0000000..8336eb0 --- /dev/null +++ b/prettierignore.txt @@ -0,0 +1,4 @@ +node_modules +coverage +dist +*.zip diff --git a/prettierrc.json b/prettierrc.json new file mode 100644 index 0000000..94e30b6 --- /dev/null +++ b/prettierrc.json @@ -0,0 +1,6 @@ +{ + "singleQuote": true, + "semi": true, + "trailingComma": "es5", + "printWidth": 80 +} diff --git a/src/App.tsx b/src/App.tsx index 9acd01d..24e7ae9 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -1,92 +1,42 @@ -import React, { useEffect, useRef, useState } from 'react'; -import Layout from './components/Layout'; -import FileLoader from './components/FileLoader'; -import ReorderPanel from './components/ReorderPanel'; -import ActionsPanel from './components/ActionsPanel'; -import PagePreviewModal from './components/PagePreviewModal'; -import WorkspacePanel from './components/WorkspacePanel'; +import React, { useCallback, useEffect, useState } from "react"; +import Layout from "./components/Layout"; +import FileLoader from "./components/FileLoader"; +import ReorderPanel from "./components/ReorderPanel"; +import ActionsPanel from "./components/ActionsPanel"; +import PagePreviewModal from "./components/PagePreviewModal"; +import WorkspacePanel from "./components/WorkspacePanel"; import ActionDialog, { type ActionDialogAction, -} from './components/ActionDialog'; -import HelpDialog from './components/HelpDialog'; -import { PDFDocument } from 'pdf-lib'; +} from "./components/ActionDialog"; +import HelpDialog from "./components/HelpDialog"; +import { PDFDocument } from "pdf-lib"; import type { StoredWorkspace, WorkspaceSummary, -} from './workspace/workspaceTypes'; -import type { - WorkspaceCommand, - WorkspaceCommandRecord, - WorkspaceCommandState, -} from './workspace/workspaceCommands'; +} from "./workspace/workspaceTypes"; import { - createSnapshotCommand, - reviveWorkspaceCommand, - toWorkspaceCommandRecord, -} from './workspace/workspaceCommands'; + createInitialPageRefs, + createPageRefId, + createWorkspaceId, + defaultWorkspaceNameFromPdfName, + normalizeRotation, + useWorkspaceState, +} from "./workspace/useWorkspaceState"; import { deleteWorkspaceFromIndexedDb, listWorkspaces, loadWorkspaceFromIndexedDb, saveWorkspaceToIndexedDb, -} from './workspace/workspaceDb'; -import type { PageRef, PdfFile, SplitResult } from './pdf/pdfTypes'; +} from "./workspace/workspaceDb"; +import type { PageRef, PdfFile } from "./pdf/pdfTypes"; import { loadPdfFromFile, mergePdfFiles, splitIntoSinglePages, exportPages, -} from './pdf/pdfService'; -import { - generateThumbnailsProgressive, - generateThumbnailsWithRotationsProgressive, -} from './pdf/pdfThumbnailService'; - -const THUMBNAIL_MAX_HEIGHT = 150; -const THUMBNAIL_MAX_WIDTH = 140; -const THUMBNAIL_CONCURRENCY = 3; - -function createId(prefix: string): string { - if (typeof crypto !== 'undefined' && crypto.randomUUID) { - return crypto.randomUUID(); - } - - return `${prefix}_${Date.now()}_${Math.random().toString(36).slice(2)}`; -} - -function defaultWorkspaceNameFromPdfName(pdfName: string): string { - return pdfName.replace(/\.pdf$/i, '') || 'Untitled workspace'; -} - -function createPageRefId(): string { - return createId('page'); -} - -function createInitialPageRefs(pageCount: number): PageRef[] { - return Array.from({ length: pageCount }, (_, sourcePageIndex) => ({ - id: createPageRefId(), - sourcePageIndex, - rotation: 0, - })); -} - -function normalizeRotation(rotation: number | undefined): number { - return (((rotation ?? 0) % 360) + 360) % 360; -} - -function thumbnailCacheKey( - pdfId: string, - sourcePageIndex: number, - rotation: number -): string { - return [ - pdfId, - sourcePageIndex, - normalizeRotation(rotation), - THUMBNAIL_MAX_WIDTH, - THUMBNAIL_MAX_HEIGHT, - ].join(':'); -} +} from "./pdf/pdfService"; +import { usePdfThumbnails } from "./pdf/usePdfThumbnails"; +import { usePdfGeneratedOutputs } from "./hooks/usePdfGeneratedOutputs"; function isEditableKeyboardTarget(target: EventTarget | null): boolean { if (!(target instanceof HTMLElement)) return false; @@ -94,9 +44,9 @@ function isEditableKeyboardTarget(target: EventTarget | null): boolean { const tagName = target.tagName.toLowerCase(); return ( target.isContentEditable || - tagName === 'input' || - tagName === 'textarea' || - tagName === 'select' + tagName === "input" || + tagName === "textarea" || + tagName === "select" ); } @@ -113,49 +63,69 @@ const App: React.FC = () => { const [error, setError] = useState(null); const [workspaces, setWorkspaces] = useState([]); - const [activeWorkspaceId, setActiveWorkspaceId] = useState(null); - const [workspaceName, setWorkspaceName] = useState(''); - const [workspaceDirty, setWorkspaceDirty] = useState(false); - const [workspaceMessage, setWorkspaceMessage] = useState(null); - const [workspaceHistory, setWorkspaceHistory] = useState([]); - const [redoHistory, setRedoHistory] = useState([]); - - const [pages, setPages] = useState([]); - const [reorderThumbnails, setReorderThumbnails] = useState>({}); - - const [selectedPageIds, setSelectedPageIds] = useState([]); - const [lastSelectedVisualIndex, setLastSelectedVisualIndex] = useState(null); - - const [splitResults, setSplitResults] = useState([]); - const [subsetUrl, setSubsetUrl] = useState(null); - const [subsetFilename, setSubsetFilename] = useState(null); - const [exportUrl, setExportUrl] = useState(null); - const [exportFilename, setExportFilename] = useState(null); + const [activeWorkspaceId, setActiveWorkspaceId] = useState( + null, + ); + const [workspaceName, setWorkspaceName] = useState(""); const [previewPageId, setPreviewPageId] = useState(null); const [pendingFile, setPendingFile] = useState(null); const [showMergeOptions, setShowMergeOptions] = useState(false); - const [mergeMode, setMergeMode] = useState<'overwrite' | 'append' | 'insertAt'>('append'); - const [mergeInsertAt, setMergeInsertAt] = useState(''); + const [mergeMode, setMergeMode] = useState< + "overwrite" | "append" | "insertAt" + >("append"); + const [mergeInsertAt, setMergeInsertAt] = useState(""); - const thumbnailCacheRef = useRef>(new Map()); - const latestPagesRef = useRef([]); - const previousPageRotationsRef = useRef>(new Map()); + const { + splitDownloads, + subsetDownload, + exportDownload, + replaceSplitResults, + replaceSubsetResult, + replaceExportResult, + clearAllResults: clearGeneratedOutputs, + } = usePdfGeneratedOutputs(); - const resetGeneratedUrls = () => { - if (subsetUrl) { - URL.revokeObjectURL(subsetUrl); - setSubsetUrl(null); - setSubsetFilename(null); - } + const handleWorkspaceContentChanged = useCallback(() => { + clearGeneratedOutputs(); + }, [clearGeneratedOutputs]); - if (exportUrl) { - URL.revokeObjectURL(exportUrl); - setExportUrl(null); - setExportFilename(null); - } - }; + const { + pages, + selectedPageIds, + setSelectedPageIds, + lastSelectedVisualIndex, + setLastSelectedVisualIndex, + workspaceDirty, + setWorkspaceDirty, + workspaceMessage, + setWorkspaceMessage, + workspaceHistory, + redoHistory, + getCurrentCommandState, + createWorkspaceCommand, + executeWorkspaceCommand, + handleUndo, + handleRedo, + replaceWorkspaceState, + resetWorkspaceState: resetWorkspaceCommandState, + } = useWorkspaceState({ onContentChanged: handleWorkspaceContentChanged }); + + const handleThumbnailError = useCallback( + (message: string, thrown: unknown) => { + console.error(thrown); + setError(message); + }, + [], + ); + + const { thumbnails: reorderThumbnails, clearThumbnailCache } = + usePdfThumbnails({ + pdf, + pages, + onError: handleThumbnailError, + }); const refreshWorkspaces = async () => { try { @@ -163,7 +133,7 @@ const App: React.FC = () => { setWorkspaces(summaries); } catch (e) { console.error(e); - setError('Failed to read saved workspaces from browser storage.'); + setError("Failed to read saved workspaces from browser storage."); } }; @@ -174,93 +144,11 @@ const App: React.FC = () => { const resetWorkspaceState = () => { setPdf(null); setActiveWorkspaceId(null); - setWorkspaceName(''); - setWorkspaceDirty(false); - setWorkspaceMessage(null); - setWorkspaceHistory([]); - setSplitResults([]); - setSelectedPageIds([]); - setLastSelectedVisualIndex(null); - resetGeneratedUrls(); - setReorderThumbnails({}); - thumbnailCacheRef.current.clear(); - previousPageRotationsRef.current.clear(); - latestPagesRef.current = []; - setPages([]); + setWorkspaceName(""); + resetWorkspaceCommandState(); + clearGeneratedOutputs(); + clearThumbnailCache(); setPreviewPageId(null); - setWorkspaceHistory([]); - setRedoHistory([]); - }; - - const getCurrentCommandState = (): WorkspaceCommandState => ({ - pages, - selectedPageIds, - lastSelectedVisualIndex, - }); - - const applyCommandState = (state: WorkspaceCommandState) => { - setPages(state.pages); - latestPagesRef.current = state.pages; - setSelectedPageIds(state.selectedPageIds); - setLastSelectedVisualIndex(state.lastSelectedVisualIndex); - }; - - const invalidateWorkspaceOutputs = () => { - setSplitResults([]); - resetGeneratedUrls(); - setWorkspaceDirty(true); - setWorkspaceMessage(null); - }; - - const executeWorkspaceCommand = (command: WorkspaceCommand) => { - const nextState = command.do(getCurrentCommandState()); - - applyCommandState(nextState); - setWorkspaceHistory((prev) => [...prev, toWorkspaceCommandRecord(command)]); - setRedoHistory([]); - invalidateWorkspaceOutputs(); - }; - - const createWorkspaceCommand = (params: { - type: string; - label: string; - before: WorkspaceCommandState; - after: WorkspaceCommandState; - details?: Record; - }): WorkspaceCommand => - createSnapshotCommand({ - id: createId('command'), - type: params.type, - label: params.label, - before: params.before, - after: params.after, - details: params.details, - }); - - const handleUndo = () => { - const record = workspaceHistory[workspaceHistory.length - 1]; - if (!record) return; - - const command = reviveWorkspaceCommand(record); - const previousState = command.undo(getCurrentCommandState()); - - applyCommandState(previousState); - setWorkspaceHistory((prev) => prev.slice(0, -1)); - setRedoHistory((prev) => [...prev, record]); - invalidateWorkspaceOutputs(); - }; - - const handleRedo = () => { - const record = redoHistory[redoHistory.length - 1]; - if (!record) return; - - const command = reviveWorkspaceCommand(record); - const nextState = command.do(getCurrentCommandState()); - - applyCommandState(nextState); - setRedoHistory((prev) => prev.slice(0, -1)); - setWorkspaceHistory((prev) => [...prev, record]); - invalidateWorkspaceOutputs(); }; const handleSaveWorkspace = async (): Promise => { @@ -269,10 +157,13 @@ const App: React.FC = () => { setError(null); const now = new Date().toISOString(); - const name = workspaceName.trim() || defaultWorkspaceNameFromPdfName(pdf.name); - const workspaceId = activeWorkspaceId ?? createId('workspace'); + const name = + workspaceName.trim() || defaultWorkspaceNameFromPdfName(pdf.name); + const workspaceId = activeWorkspaceId ?? createWorkspaceId(); - const existing = workspaces.find((workspace) => workspace.id === workspaceId); + const existing = workspaces.find( + (workspace) => workspace.id === workspaceId, + ); const workspace: StoredWorkspace = { schemaVersion: 1, @@ -309,7 +200,7 @@ const App: React.FC = () => { } catch (e) { console.error(e); setError( - 'Failed to save workspace. The browser storage quota may be full.' + "Failed to save workspace. The browser storage quota may be full.", ); return false; } finally { @@ -330,12 +221,10 @@ const App: React.FC = () => { } openActionDialog({ - title: 'Reset workspace?', + title: "Reset workspace?", content: ( <> -

- This workspace has unsaved changes. -

+

This workspace has unsaved changes.

Do you want to save it before resetting?

@@ -343,21 +232,21 @@ const App: React.FC = () => { ), actions: [ { - label: 'Cancel', - variant: 'secondary', + label: "Cancel", + variant: "secondary", onClick: closeActionDialog, }, { - label: 'Reset without saving', - variant: 'danger', + label: "Reset without saving", + variant: "danger", onClick: () => { closeActionDialog(); performResetWorkspace(); }, }, { - label: 'Save and reset', - variant: 'primary', + label: "Save and reset", + variant: "primary", autoFocus: true, onClick: async () => { closeActionDialog(); @@ -379,12 +268,12 @@ const App: React.FC = () => { const loaded = await loadWorkspaceFromIndexedDb(workspaceId); if (!loaded) { - setError('Workspace not found.'); + setError("Workspace not found."); await refreshWorkspaces(); return; } - resetGeneratedUrls(); + clearGeneratedOutputs(); const doc = await PDFDocument.load(loaded.pdfArrayBuffer); @@ -397,27 +286,24 @@ const App: React.FC = () => { }; setPdf(loadedPdf); - setPages(loaded.workspace.pages); - latestPagesRef.current = loaded.workspace.pages; - - setSelectedPageIds(loaded.workspace.selectedPageIds ?? []); - setLastSelectedVisualIndex(null); - setSplitResults([]); + replaceWorkspaceState({ + pages: loaded.workspace.pages, + selectedPageIds: loaded.workspace.selectedPageIds ?? [], + lastSelectedVisualIndex: null, + history: loaded.workspace.history ?? [], + redoHistory: loaded.workspace.redoHistory ?? [], + dirty: false, + message: `Workspace "${loaded.workspace.name}" loaded.`, + }); setPreviewPageId(null); - setReorderThumbnails({}); - thumbnailCacheRef.current.clear(); - previousPageRotationsRef.current.clear(); + clearThumbnailCache(); setActiveWorkspaceId(loaded.workspace.id); setWorkspaceName(loaded.workspace.name); - setWorkspaceHistory(loaded.workspace.history ?? []); - setRedoHistory(loaded.workspace.redoHistory ?? []); - setWorkspaceDirty(false); - setWorkspaceMessage(`Workspace "${loaded.workspace.name}" loaded.`); } catch (e) { console.error(e); - setError('Failed to load workspace from browser storage.'); + setError("Failed to load workspace from browser storage."); } finally { setIsBusy(false); } @@ -425,30 +311,31 @@ const App: React.FC = () => { const handleDeleteWorkspace = (workspaceId: string) => { const workspace = workspaces.find((item) => item.id === workspaceId); - const name = workspace?.name ?? 'this workspace'; + const name = workspace?.name ?? "this workspace"; openActionDialog({ - title: 'Delete workspace?', + title: "Delete workspace?", content: ( <>

- Delete the saved workspace {name} from this browser? + Delete the saved workspace {name} from this + browser?

- The currently open in-memory document will not be closed, but the saved - workspace entry will be removed. + The currently open in-memory document will not be closed, but the + saved workspace entry will be removed.

), actions: [ { - label: 'Cancel', - variant: 'secondary', + label: "Cancel", + variant: "secondary", onClick: closeActionDialog, }, { - label: 'Delete workspace', - variant: 'danger', + label: "Delete workspace", + variant: "danger", autoFocus: true, onClick: () => { closeActionDialog(); @@ -469,14 +356,14 @@ const App: React.FC = () => { setActiveWorkspaceId(null); setWorkspaceDirty(true); setWorkspaceMessage( - 'Saved workspace deleted. Current in-memory document remains open.' + "Saved workspace deleted. Current in-memory document remains open.", ); } await refreshWorkspaces(); } catch (e) { console.error(e); - setError('Failed to delete workspace.'); + setError("Failed to delete workspace."); } }; @@ -490,17 +377,20 @@ const App: React.FC = () => { const initialPages = createInitialPageRefs(loaded.pageCount); setPdf(loaded); - setPages(initialPages); - latestPagesRef.current = initialPages; + replaceWorkspaceState({ + pages: initialPages, + selectedPageIds: [], + lastSelectedVisualIndex: null, + history: [], + redoHistory: [], + dirty: true, + message: null, + }); setWorkspaceName(defaultWorkspaceNameFromPdfName(loaded.name)); - setWorkspaceHistory([]); - setRedoHistory([]); - setWorkspaceDirty(true); - setWorkspaceMessage(null); } catch (e) { console.error(e); - setError('Failed to load PDF (see console).'); + setError("Failed to load PDF (see console)."); } finally { setIsBusy(false); } @@ -512,7 +402,7 @@ const App: React.FC = () => { } else { setPendingFile(file); setShowMergeOptions(true); - setMergeMode('append'); + setMergeMode("append"); setMergeInsertAt(String(pages.length + 1)); } }; @@ -525,7 +415,7 @@ const App: React.FC = () => { const handleMergeConfirm = async () => { if (!pendingFile) return; - if (!pdf || mergeMode === 'overwrite') { + if (!pdf || mergeMode === "overwrite") { await loadFileAsNew(pendingFile); setPendingFile(null); setShowMergeOptions(false); @@ -553,12 +443,12 @@ const App: React.FC = () => { // 3) Determine insert position (0-based) let insertAt = pages.length; // default: append at end - if (mergeMode === 'insertAt') { + if (mergeMode === "insertAt") { const parsed = parseInt(mergeInsertAt, 10); if (Number.isFinite(parsed)) { insertAt = Math.min(Math.max(parsed - 1, 0), pages.length); } - } else if (mergeMode === 'append') { + } else if (mergeMode === "append") { insertAt = pages.length; } @@ -568,26 +458,23 @@ const App: React.FC = () => { // 5) Reset state to the merged document setPdf(mergedPdf); - - setPages(mergedPages); - latestPagesRef.current = mergedPages; - setSelectedPageIds([]); - setLastSelectedVisualIndex(null); - setSplitResults([]); - resetGeneratedUrls(); - setReorderThumbnails({}); - thumbnailCacheRef.current.clear(); - previousPageRotationsRef.current.clear(); + replaceWorkspaceState({ + pages: mergedPages, + selectedPageIds: [], + lastSelectedVisualIndex: null, + history: [], + redoHistory: [], + dirty: true, + message: null, + }); + clearGeneratedOutputs(); + clearThumbnailCache(); setPreviewPageId(null); setWorkspaceName(defaultWorkspaceNameFromPdfName(mergedPdf.name)); - setWorkspaceHistory([]); - setRedoHistory([]); - setWorkspaceDirty(true); setActiveWorkspaceId(null); - setWorkspaceMessage(null); } catch (e) { console.error(e); - setError('Failed to merge PDF (see console).'); + setError("Failed to merge PDF (see console)."); } finally { setIsBusy(false); setPendingFile(null); @@ -596,191 +483,10 @@ const App: React.FC = () => { }; useEffect(() => { - latestPagesRef.current = pages; - }, [pages]); - - useEffect(() => { - if (!pdf) { - setReorderThumbnails({}); - thumbnailCacheRef.current.clear(); - previousPageRotationsRef.current.clear(); - return; - } - - const controller = new AbortController(); - - latestPagesRef.current = pages; - thumbnailCacheRef.current.clear(); - previousPageRotationsRef.current = new Map( - pages.map((page) => [page.id, normalizeRotation(page.rotation)]) - ); - setReorderThumbnails({}); - - void generateThumbnailsProgressive(pdf.arrayBuffer, { - maxHeight: THUMBNAIL_MAX_HEIGHT, - maxWidth: THUMBNAIL_MAX_WIDTH, - concurrency: THUMBNAIL_CONCURRENCY, - signal: controller.signal, - onThumbnail: ({ pageIndex, dataUrl }) => { - if (controller.signal.aborted) return; - - thumbnailCacheRef.current.set( - thumbnailCacheKey(pdf.id, pageIndex, 0), - dataUrl - ); - - const currentPages = latestPagesRef.current; - const updates: Record = {}; - - for (const page of currentPages) { - if ( - page.sourcePageIndex === pageIndex && - normalizeRotation(page.rotation) === 0 - ) { - updates[page.id] = dataUrl; - } - } - - if (Object.keys(updates).length === 0) return; - - setReorderThumbnails((prev) => ({ - ...prev, - ...updates, - })); - }, - }).catch((e) => { - if (!controller.signal.aborted) { - console.error(e); - setError('Failed to generate thumbnails (see console).'); - } - }); - - return () => { - controller.abort(); - }; - }, [pdf]); - - useEffect(() => { - if (!pdf) { - previousPageRotationsRef.current.clear(); - return; - } - - const previousRotations = previousPageRotationsRef.current; - const changedPages = pages.filter( - (page) => - normalizeRotation(previousRotations.get(page.id)) !== - normalizeRotation(page.rotation) - ); - - previousPageRotationsRef.current = new Map( - pages.map((page) => [page.id, normalizeRotation(page.rotation)]) - ); - - if (changedPages.length === 0) return; - - const cachedUpdates: Record = {}; - const pagesToRender: PageRef[] = []; - - for (const page of changedPages) { - const rotation = normalizeRotation(page.rotation); - const cached = thumbnailCacheRef.current.get( - thumbnailCacheKey(pdf.id, page.sourcePageIndex, rotation) - ); - - if (cached) { - cachedUpdates[page.id] = cached; - } else { - pagesToRender.push(page); - } - } - - if (Object.keys(cachedUpdates).length > 0) { - setReorderThumbnails((prev) => ({ - ...prev, - ...cachedUpdates, - })); - } - - if (pagesToRender.length === 0) return; - - const controller = new AbortController(); - const groups = new Map(); - - for (const page of pagesToRender) { - const rotation = normalizeRotation(page.rotation); - const group = groups.get(rotation) ?? []; - group.push(page); - groups.set(rotation, group); - } - - const renderGroups = async () => { - for (const [rotation, groupPages] of groups) { - if (controller.signal.aborted) return; - - const pageIndices = Array.from( - new Set(groupPages.map((page) => page.sourcePageIndex)) - ); - const rotationsBySourcePage: Record = {}; - - for (const pageIndex of pageIndices) { - rotationsBySourcePage[pageIndex] = rotation; - } - - await generateThumbnailsWithRotationsProgressive( - pdf.arrayBuffer, - rotationsBySourcePage, - { - maxHeight: THUMBNAIL_MAX_HEIGHT, - maxWidth: THUMBNAIL_MAX_WIDTH, - concurrency: Math.min(THUMBNAIL_CONCURRENCY, pageIndices.length), - pageIndices, - signal: controller.signal, - onThumbnail: ({ pageIndex, dataUrl }) => { - if (controller.signal.aborted) return; - - thumbnailCacheRef.current.set( - thumbnailCacheKey(pdf.id, pageIndex, rotation), - dataUrl - ); - - const updates: Record = {}; - - for (const page of latestPagesRef.current) { - if ( - page.sourcePageIndex === pageIndex && - normalizeRotation(page.rotation) === rotation - ) { - updates[page.id] = dataUrl; - } - } - - if (Object.keys(updates).length === 0) return; - - setReorderThumbnails((prev) => ({ - ...prev, - ...updates, - })); - }, - } - ); - } - }; - - void renderGroups().catch((e) => { - if (!controller.signal.aborted) { - console.error(e); - setError('Failed to generate rotated thumbnails (see console).'); - } - }); - - return () => { - controller.abort(); - }; - }, [pdf, pages]); - - useEffect(() => { - if (previewPageId != null && !pages.some((page) => page.id === previewPageId)) { + if ( + previewPageId != null && + !pages.some((page) => page.id === previewPageId) + ) { setPreviewPageId(null); } }, [previewPageId, pages]); @@ -791,16 +497,16 @@ const App: React.FC = () => { const handleKeyDown = (e: KeyboardEvent) => { if (isEditableKeyboardTarget(e.target)) return; - if (e.key === 'F1' || e.key === '?') { + if (e.key === "F1" || e.key === "?") { e.preventDefault(); setHelpOpen(true); } }; - window.addEventListener('keydown', handleKeyDown); + window.addEventListener("keydown", handleKeyDown); return () => { - window.removeEventListener('keydown', handleKeyDown); + window.removeEventListener("keydown", handleKeyDown); }; }, []); @@ -810,13 +516,13 @@ const App: React.FC = () => { const afterPages = pages.map((page) => page.id === pageId ? { ...page, rotation: (normalizeRotation(page.rotation) + 90) % 360 } - : page + : page, ); executeWorkspaceCommand( createWorkspaceCommand({ - type: 'page.rotate', - label: 'Rotated page clockwise', + type: "page.rotate", + label: "Rotated page clockwise", before, after: { ...before, @@ -826,7 +532,7 @@ const App: React.FC = () => { pageId, degrees: 90, }, - }) + }), ); }; @@ -835,13 +541,13 @@ const App: React.FC = () => { const afterPages = pages.map((page) => page.id === pageId ? { ...page, rotation: (normalizeRotation(page.rotation) + 270) % 360 } - : page + : page, ); executeWorkspaceCommand( createWorkspaceCommand({ - type: 'page.rotate', - label: 'Rotated page counterclockwise', + type: "page.rotate", + label: "Rotated page counterclockwise", before, after: { ...before, @@ -851,7 +557,7 @@ const App: React.FC = () => { pageId, degrees: -90, }, - }) + }), ); }; @@ -859,10 +565,10 @@ const App: React.FC = () => { const page = pages.find((item) => item.id === pageId); const visualIndex = page ? pages.indexOf(page) : -1; const pageLabel = - visualIndex >= 0 ? `page at position ${visualIndex + 1}` : 'this page'; + visualIndex >= 0 ? `page at position ${visualIndex + 1}` : "this page"; openActionDialog({ - title: 'Delete page?', + title: "Delete page?", content: (

Delete {pageLabel} from the current workspace? @@ -870,13 +576,13 @@ const App: React.FC = () => { ), actions: [ { - label: 'Cancel', - variant: 'secondary', + label: "Cancel", + variant: "secondary", onClick: closeActionDialog, }, { - label: 'Delete page', - variant: 'danger', + label: "Delete page", + variant: "danger", autoFocus: true, onClick: () => { closeActionDialog(); @@ -892,8 +598,8 @@ const App: React.FC = () => { executeWorkspaceCommand( createWorkspaceCommand({ - type: 'page.delete', - label: 'Deleted page', + type: "page.delete", + label: "Deleted page", before, after: { pages: pages.filter((page) => page.id !== pageId), @@ -903,7 +609,7 @@ const App: React.FC = () => { details: { pageId, }, - }) + }), ); }; @@ -912,8 +618,8 @@ const App: React.FC = () => { executeWorkspaceCommand( createWorkspaceCommand({ - type: 'pages.reorder', - label: 'Reordered pages', + type: "pages.reorder", + label: "Reordered pages", before, after: { ...before, @@ -922,14 +628,14 @@ const App: React.FC = () => { details: { pageCount: newPages.length, }, - }) + }), ); }; const handleToggleSelect = ( pageId: string, visualIndex: number, - e: React.MouseEvent + e: React.MouseEvent, ) => { setSelectedPageIds((prev) => { if (e.shiftKey && lastSelectedVisualIndex !== null && pages.length > 0) { @@ -961,7 +667,7 @@ const App: React.FC = () => { setSelectedPageIds([]); setLastSelectedVisualIndex(null); }; - + const handleDeleteSelected = () => { if (selectedPageIds.length === 0) return; @@ -970,29 +676,28 @@ const App: React.FC = () => { openActionDialog({ title: idsToDelete.length === 1 - ? 'Delete selected page?' - : 'Delete selected pages?', + ? "Delete selected page?" + : "Delete selected pages?", content: (

- Delete{' '} + Delete{" "} {idsToDelete.length === 1 - ? '1 selected page' + ? "1 selected page" : `${idsToDelete.length} selected pages`} - {' '} + {" "} from the current workspace?

), actions: [ { - label: 'Cancel', - variant: 'secondary', + label: "Cancel", + variant: "secondary", onClick: closeActionDialog, }, { - label: - idsToDelete.length === 1 ? 'Delete page' : 'Delete pages', - variant: 'danger', + label: idsToDelete.length === 1 ? "Delete page" : "Delete pages", + variant: "danger", autoFocus: true, onClick: () => { closeActionDialog(); @@ -1011,10 +716,10 @@ const App: React.FC = () => { executeWorkspaceCommand( createWorkspaceCommand({ - type: 'pages.delete', + type: "pages.delete", label: pageIdsToDelete.length === 1 - ? 'Deleted selected page' + ? "Deleted selected page" : `Deleted ${pageIdsToDelete.length} selected pages`, before, after: { @@ -1025,7 +730,7 @@ const App: React.FC = () => { details: { count: pageIdsToDelete.length, }, - }) + }), ); }; @@ -1052,34 +757,14 @@ const App: React.FC = () => { ...pages.slice(clampedSlot), ]; - const thumbnailUpdates: Record = {}; - - sourcePages.forEach((sourcePage, index) => { - const copiedPage = copiedPages[index]; - - const thumbnail = - reorderThumbnails[sourcePage.id] ?? - thumbnailCacheRef.current.get( - thumbnailCacheKey( - pdf.id, - sourcePage.sourcePageIndex, - sourcePage.rotation - ) - ); - - if (thumbnail) { - thumbnailUpdates[copiedPage.id] = thumbnail; - } - }); - const before = getCurrentCommandState(); executeWorkspaceCommand( createWorkspaceCommand({ - type: 'pages.copy', + type: "pages.copy", label: copiedPages.length === 1 - ? 'Copied page' + ? "Copied page" : `Copied ${copiedPages.length} pages`, before, after: { @@ -1091,15 +776,8 @@ const App: React.FC = () => { count: copiedPages.length, insertSlot: clampedSlot, }, - }) + }), ); - - if (Object.keys(thumbnailUpdates).length > 0) { - setReorderThumbnails((prev) => ({ - ...prev, - ...thumbnailUpdates, - })); - } }; const closeActionDialog = () => { @@ -1152,7 +830,7 @@ const App: React.FC = () => { const key = e.key.toLowerCase(); - if ((e.ctrlKey || e.metaKey) && key === 'z') { + if ((e.ctrlKey || e.metaKey) && key === "z") { e.preventDefault(); if (e.shiftKey) { handleRedo(); @@ -1162,38 +840,53 @@ const App: React.FC = () => { return; } - if ((e.ctrlKey || e.metaKey) && key === 'y') { + if ((e.ctrlKey || e.metaKey) && key === "y") { e.preventDefault(); handleRedo(); return; } - if ((e.ctrlKey || e.metaKey) && key === 'a') { + if ((e.ctrlKey || e.metaKey) && key === "a") { e.preventDefault(); setSelectedPageIds(pages.map((page) => page.id)); setLastSelectedVisualIndex(null); return; } - if ((e.key === 'Delete' || e.key === 'Backspace') && selectedPageIds.length > 0) { + if ( + (e.key === "Delete" || e.key === "Backspace") && + selectedPageIds.length > 0 + ) { e.preventDefault(); handleDeleteSelected(); return; } - if (e.key === 'Escape' && selectedPageIds.length > 0) { + if (e.key === "Escape" && selectedPageIds.length > 0) { e.preventDefault(); setSelectedPageIds([]); setLastSelectedVisualIndex(null); } }; - window.addEventListener('keydown', handleKeyDown); + window.addEventListener("keydown", handleKeyDown); return () => { - window.removeEventListener('keydown', handleKeyDown); + window.removeEventListener("keydown", handleKeyDown); }; - }, [hasPdf, previewPageId, pages, selectedPageIds, workspaceHistory, redoHistory]); + }, [ + hasPdf, + previewPageId, + pages, + selectedPageIds, + workspaceHistory, + redoHistory, + handleUndo, + handleRedo, + handleDeleteSelected, + setSelectedPageIds, + setLastSelectedVisualIndex, + ]); const handleSplit = async () => { if (!pdf) return; @@ -1201,38 +894,30 @@ const App: React.FC = () => { setIsBusy(true); try { const result = await splitIntoSinglePages(pdf); - setSplitResults(result); + replaceSplitResults(result); } catch (e) { console.error(e); - setError('Error while splitting PDF (see console).'); + setError("Error while splitting PDF (see console)."); } finally { setIsBusy(false); } }; const handleExtractSelected = async () => { - if (!pdf || selectedPageIds.length === 0) return + if (!pdf || selectedPageIds.length === 0) return; setError(null); setIsBusy(true); - if (subsetUrl) { - URL.revokeObjectURL(subsetUrl); - setSubsetUrl(null); - setSubsetFilename(null); - } - try { const selectedSet = new Set(selectedPageIds); const selectedPages = pages.filter((page) => selectedSet.has(page.id)); const blob = await exportPages(pdf, selectedPages); - const url = URL.createObjectURL(blob); - const base = pdf.name.replace(/\.pdf$/i, ''); + const base = pdf.name.replace(/\.pdf$/i, ""); const filename = `${base}_selected.pdf`; - setSubsetUrl(url); - setSubsetFilename(filename); + replaceSubsetResult(blob, filename); } catch (e) { console.error(e); - setError('Error while extracting selected pages (see console).'); + setError("Error while extracting selected pages (see console)."); } finally { setIsBusy(false); } @@ -1243,22 +928,14 @@ const App: React.FC = () => { setError(null); setIsBusy(true); - if (exportUrl) { - URL.revokeObjectURL(exportUrl); - setExportUrl(null); - setExportFilename(null); - } - try { const blob = await exportPages(pdf, pages); - const url = URL.createObjectURL(blob); - const base = pdf.name.replace(/\.pdf$/i, ''); + const base = pdf.name.replace(/\.pdf$/i, ""); const filename = `${base}_reordered.pdf`; - setExportUrl(url); - setExportFilename(filename); + replaceExportResult(blob, filename); } catch (e) { console.error(e); - setError('Error while exporting reordered PDF (see console).'); + setError("Error while exporting reordered PDF (see console)."); } finally { setIsBusy(false); } @@ -1305,62 +982,62 @@ const App: React.FC = () => { {showMergeOptions && pendingFile && pdf && pages.length > 0 && (

Open file: merge or replace?

-

- You already have {pdf.name} with {pages.length}{' '} - pages open. What should happen with{' '} +

+ You already have {pdf.name} with {pages.length}{" "} + pages open. What should happen with{" "} {pendingFile.name}?

-
+
@@ -1419,7 +1096,6 @@ const App: React.FC = () => { onDeleteSelected={handleDeleteSelected} /> - { onSplit={handleSplit} onExtractSelected={handleExtractSelected} onExportReordered={handleExportReordered} - splitResults={splitResults} - subsetDownloadUrl={subsetUrl} - subsetFilename={subsetFilename} - exportDownloadUrl={exportUrl} - exportFilename={exportFilename} + splitDownloads={splitDownloads} + subsetDownload={subsetDownload} + exportDownload={exportDownload} /> {error && (
Error: {error}
@@ -1459,7 +1133,7 @@ const App: React.FC = () => { diff --git a/src/components/ActionDialog.tsx b/src/components/ActionDialog.tsx index b4a399c..bb50f8c 100644 --- a/src/components/ActionDialog.tsx +++ b/src/components/ActionDialog.tsx @@ -1,9 +1,9 @@ -import React, { useEffect } from 'react'; +import React, { useEffect } from "react"; export interface ActionDialogAction { label: string; onClick: () => void | Promise; - variant?: 'primary' | 'secondary' | 'danger'; + variant?: "primary" | "secondary" | "danger"; disabled?: boolean; autoFocus?: boolean; title?: string; @@ -18,21 +18,21 @@ interface ActionDialogProps { } const backgroundByVariant: Record< - NonNullable, + NonNullable, string > = { - primary: '#2563eb', - secondary: '#e5e7eb', - danger: '#dc2626', + primary: "#2563eb", + secondary: "#e5e7eb", + danger: "#dc2626", }; const colorByVariant: Record< - NonNullable, + NonNullable, string > = { - primary: 'white', - secondary: '#111827', - danger: 'white', + primary: "white", + secondary: "#111827", + danger: "white", }; const ActionDialog: React.FC = ({ @@ -46,16 +46,16 @@ const ActionDialog: React.FC = ({ if (!open) return; const handleKeyDown = (e: KeyboardEvent) => { - if (e.key === 'Escape') { + if (e.key === "Escape") { e.preventDefault(); onClose(); } }; - window.addEventListener('keydown', handleKeyDown); + window.addEventListener("keydown", handleKeyDown); return () => { - window.removeEventListener('keydown', handleKeyDown); + window.removeEventListener("keydown", handleKeyDown); }; }, [open, onClose]); @@ -72,42 +72,42 @@ const ActionDialog: React.FC = ({ } }} style={{ - position: 'fixed', + position: "fixed", inset: 0, zIndex: 70, - background: 'rgba(15, 23, 42, 0.55)', - display: 'flex', - alignItems: 'center', - justifyContent: 'center', - padding: '1rem', + background: "rgba(15, 23, 42, 0.55)", + display: "flex", + alignItems: "center", + justifyContent: "center", + padding: "1rem", }} >

{title} @@ -117,18 +117,18 @@ const ActionDialog: React.FC = ({ type="button" onClick={onClose} style={{ - border: 'none', - borderRadius: '999px', - width: '1.8rem', - height: '1.8rem', - background: '#e5e7eb', - color: '#111827', - cursor: 'pointer', - fontSize: '1.1rem', + border: "none", + borderRadius: "999px", + width: "1.8rem", + height: "1.8rem", + background: "#e5e7eb", + color: "#111827", + cursor: "pointer", + fontSize: "1.1rem", lineHeight: 1, - display: 'flex', - alignItems: 'center', - justifyContent: 'center', + display: "flex", + alignItems: "center", + justifyContent: "center", }} aria-label="Close dialog" > @@ -138,8 +138,8 @@ const ActionDialog: React.FC = ({
@@ -148,15 +148,15 @@ const ActionDialog: React.FC = ({
{actions.map((action) => { - const variant = action.variant ?? 'secondary'; + const variant = action.variant ?? "secondary"; return ( @@ -63,11 +62,11 @@ const ActionsPanel: React.FC = ({ className="secondary" disabled={disabled || selectedCount === 0} onClick={handleExtractSelectedClick} - style={{ flex: '1 1 45%' }} + style={{ flex: "1 1 45%" }} title={ selectedCount === 0 - ? 'Select at least one page' - : 'Create a PDF from selected pages' + ? "Select at least one page" + : "Create a PDF from selected pages" } > 📤 Extract selected ({selectedCount}) @@ -77,58 +76,52 @@ const ActionsPanel: React.FC = ({ className="secondary" disabled={disabled} onClick={onSplit} - style={{ flex: '1 1 45%' }} + style={{ flex: "1 1 45%" }} > 📂 Split into single PDFs
- {subsetDownloadUrl && subsetFilename && ( -
- Subset result:{' '} + {subsetDownload && ( + )} - {exportDownloadUrl && exportFilename && ( -
- Exported document:{' '} + {exportDownload && ( + )} - {splitResults.length > 0 && ( -
+ {splitDownloads.length > 0 && ( +
Single-page PDFs:
- {splitResults.map((r) => { - const url = URL.createObjectURL(r.blob); - return ( - { - setTimeout(() => URL.revokeObjectURL(url), 5000); - }} - > - {r.filename} - - ); - })} + {splitDownloads.map((download) => ( + + {download.filename} + + ))}
)} diff --git a/src/components/FileLoader.tsx b/src/components/FileLoader.tsx index 08d2af8..b18793e 100644 --- a/src/components/FileLoader.tsx +++ b/src/components/FileLoader.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import type { PdfFile } from '../pdf/pdfTypes'; +import React from "react"; +import type { PdfFile } from "../pdf/pdfTypes"; interface FileLoaderProps { pdf: PdfFile | null; @@ -11,7 +11,7 @@ const FileLoader: React.FC = ({ pdf, onFileLoaded }) => { const file = e.target.files?.[0]; if (file) { onFileLoaded(file); - e.target.value = ''; + e.target.value = ""; } }; @@ -22,7 +22,7 @@ const FileLoader: React.FC = ({ pdf, onFileLoaded }) => { {pdf && ( -
+
Loaded: {pdf.name}
diff --git a/src/components/HelpDialog.tsx b/src/components/HelpDialog.tsx index abbbad2..9e7afa7 100644 --- a/src/components/HelpDialog.tsx +++ b/src/components/HelpDialog.tsx @@ -1,4 +1,4 @@ -import React, { useEffect } from 'react'; +import React, { useEffect } from "react"; interface HelpDialogProps { open: boolean; @@ -6,36 +6,51 @@ interface HelpDialogProps { } const shortcuts = [ - { keys: 'F1 / ?', description: 'Open this help and tutorial dialog' }, - { keys: 'Ctrl/⌘ + A', description: 'Select all pages in the current workspace' }, - { keys: 'Delete / Backspace', description: 'Delete the selected pages after confirmation' }, - { keys: 'Esc', description: 'Clear the page selection or close an open dialog' }, - { keys: 'Ctrl/⌘ + Z', description: 'Undo the latest workspace command' }, - { keys: 'Ctrl/⌘ + Shift + Z', description: 'Redo the next workspace command' }, - { keys: 'Ctrl/⌘ + Y', description: 'Redo the next workspace command' }, - { keys: '← / → in preview', description: 'Move to the previous or next page in the preview overlay' }, + { keys: "F1 / ?", description: "Open this help and tutorial dialog" }, + { + keys: "Ctrl/⌘ + A", + description: "Select all pages in the current workspace", + }, + { + keys: "Delete / Backspace", + description: "Delete the selected pages after confirmation", + }, + { + keys: "Esc", + description: "Clear the page selection or close an open dialog", + }, + { keys: "Ctrl/⌘ + Z", description: "Undo the latest workspace command" }, + { + keys: "Ctrl/⌘ + Shift + Z", + description: "Redo the next workspace command", + }, + { keys: "Ctrl/⌘ + Y", description: "Redo the next workspace command" }, + { + keys: "← / → in preview", + description: "Move to the previous or next page in the preview overlay", + }, ]; const tutorialSteps = [ { - title: '1. Open a PDF or load a workspace', - body: 'Start by selecting a local PDF file. If you saved workspaces before, you can restore one from browser storage instead.', + title: "1. Open a PDF or load a workspace", + body: "Start by selecting a local PDF file. If you saved workspaces before, you can restore one from browser storage instead.", }, { - title: '2. Arrange pages visually', - body: 'Drag page cards to reorder them. Rotate single pages, open the large preview with a click, or remove pages you do not want in the export.', + title: "2. Arrange pages visually", + body: "Drag page cards to reorder them. Rotate single pages, open the large preview with a click, or remove pages you do not want in the export.", }, { - title: '3. Select, copy, and delete pages', - body: 'Use the checkbox on a page card to select it. Shift-click extends a range. Dragging a selected page moves the whole selection; the copy controls duplicate selected pages into a chosen slot.', + title: "3. Select, copy, and delete pages", + body: "Use the checkbox on a page card to select it. Shift-click extends a range. Dragging a selected page moves the whole selection; the copy controls duplicate selected pages into a chosen slot.", }, { - title: '4. Save your workspace or export a PDF', - body: 'Saving a workspace keeps the current working state in this browser. Exporting creates a new PDF file for download.', + title: "4. Save your workspace or export a PDF", + body: "Saving a workspace keeps the current working state in this browser. Exporting creates a new PDF file for download.", }, { - title: '5. Use history deliberately', - body: 'Each workspace operation is stored as a command with label and timestamp. Undo and redo walk through that command history.', + title: "5. Use history deliberately", + body: "Each workspace operation is stored as a command with label and timestamp. Undo and redo walk through that command history.", }, ]; @@ -44,7 +59,7 @@ const HelpDialog: React.FC = ({ open, onClose }) => { if (!open) return; const handleKeyDown = (e: KeyboardEvent) => { - if (e.key !== 'Escape') return; + if (e.key !== "Escape") return; e.preventDefault(); e.stopPropagation(); @@ -52,10 +67,10 @@ const HelpDialog: React.FC = ({ open, onClose }) => { onClose(); }; - window.addEventListener('keydown', handleKeyDown, { capture: true }); + window.addEventListener("keydown", handleKeyDown, { capture: true }); return () => { - window.removeEventListener('keydown', handleKeyDown, { capture: true }); + window.removeEventListener("keydown", handleKeyDown, { capture: true }); }; }, [open, onClose]); @@ -79,8 +94,8 @@ const HelpDialog: React.FC = ({ open, onClose }) => {

Help & tutorial

PDF Workbench is a browser-only page workspace. Use it to quickly - rearrange, split, merge, rotate, duplicate, and export PDFs without - uploading documents to a server. + rearrange, split, merge, rotate, duplicate, and export PDFs + without uploading documents to a server.

@@ -119,8 +134,8 @@ const HelpDialog: React.FC = ({ open, onClose }) => { ))}

- Shortcuts are ignored while typing in text fields or other editable - controls. + Shortcuts are ignored while typing in text fields or other + editable controls.

diff --git a/src/components/Layout.tsx b/src/components/Layout.tsx index 5d51fe1..23e745e 100644 --- a/src/components/Layout.tsx +++ b/src/components/Layout.tsx @@ -1,5 +1,5 @@ -import React from 'react'; -import { APP_VERSION } from '../version'; +import React from "react"; +import { APP_VERSION } from "../version"; interface LayoutProps { children: React.ReactNode; @@ -44,4 +44,4 @@ const Layout: React.FC = ({ children, onOpenHelp }) => { ); }; -export default Layout; \ No newline at end of file +export default Layout; diff --git a/src/components/PagePreviewModal.tsx b/src/components/PagePreviewModal.tsx index 11417ee..2325467 100644 --- a/src/components/PagePreviewModal.tsx +++ b/src/components/PagePreviewModal.tsx @@ -1,7 +1,7 @@ -import React, { useEffect, useRef } from 'react'; -import type { PdfFile } from '../pdf/pdfTypes'; -import * as pdfjsLib from 'pdfjs-dist'; -import pdfjsWorker from 'pdfjs-dist/build/pdf.worker?worker&url'; +import React, { useEffect, useRef } from "react"; +import type { PdfFile } from "../pdf/pdfTypes"; +import * as pdfjsLib from "pdfjs-dist"; +import pdfjsWorker from "pdfjs-dist/build/pdf.worker?worker&url"; // pdf.js worker setup // eslint-disable-next-line @typescript-eslint/no-explicit-any @@ -10,10 +10,10 @@ import pdfjsWorker from 'pdfjs-dist/build/pdf.worker?worker&url'; interface PagePreviewModalProps { isOpen: boolean; pdf: PdfFile | null; - pageIndex: number | null; // original page index, 0-based - rotation: number; // degrees + pageIndex: number | null; // original page index, 0-based + rotation: number; // degrees - visualIndex: number | null; // current position in order, 0-based + visualIndex: number | null; // current position in order, 0-based totalPages: number; canGoPrevious: boolean; @@ -43,28 +43,28 @@ const PagePreviewModal: React.FC = ({ if (!isOpen) return; const handleKeyDown = (e: KeyboardEvent) => { - if (e.key === 'Escape') { + if (e.key === "Escape") { e.preventDefault(); onClose(); return; } - if (e.key === 'ArrowLeft' && canGoPrevious) { + if (e.key === "ArrowLeft" && canGoPrevious) { e.preventDefault(); onPrevious(); return; } - if (e.key === 'ArrowRight' && canGoNext) { + if (e.key === "ArrowRight" && canGoNext) { e.preventDefault(); onNext(); } }; - window.addEventListener('keydown', handleKeyDown); + window.addEventListener("keydown", handleKeyDown); return () => { - window.removeEventListener('keydown', handleKeyDown); + window.removeEventListener("keydown", handleKeyDown); }; }, [isOpen, canGoPrevious, canGoNext, onPrevious, onNext, onClose]); @@ -77,7 +77,7 @@ const PagePreviewModal: React.FC = ({ try { const canvas = canvasRef.current; if (canvas) { - const ctx = canvas.getContext('2d'); + const ctx = canvas.getContext("2d"); if (ctx) { ctx.clearRect(0, 0, canvas.width, canvas.height); } @@ -102,7 +102,7 @@ const PagePreviewModal: React.FC = ({ const scale = Math.min( maxWidth / viewport.width, - maxHeight / viewport.height + maxHeight / viewport.height, ); const scaledViewport = page.getViewport({ scale }); @@ -110,7 +110,7 @@ const PagePreviewModal: React.FC = ({ const visibleCanvas = canvasRef.current; if (!visibleCanvas) return; - const visibleCtx = visibleCanvas.getContext('2d'); + const visibleCtx = visibleCanvas.getContext("2d"); if (!visibleCtx) return; let canvasWidth = scaledViewport.width; @@ -126,8 +126,8 @@ const PagePreviewModal: React.FC = ({ visibleCanvas.width = canvasWidth; visibleCanvas.height = canvasHeight; - const baseCanvas = document.createElement('canvas'); - const baseCtx = baseCanvas.getContext('2d'); + const baseCanvas = document.createElement("canvas"); + const baseCtx = baseCanvas.getContext("2d"); if (!baseCtx) return; baseCanvas.width = scaledViewport.width; @@ -161,7 +161,7 @@ const PagePreviewModal: React.FC = ({ visibleCtx.drawImage(baseCanvas, 0, 0); visibleCtx.restore(); } catch (e) { - console.error('Error rendering preview', e); + console.error("Error rendering preview", e); } })(); @@ -181,30 +181,30 @@ const PagePreviewModal: React.FC = ({
e.stopPropagation()} style={{ - position: 'relative', - background: '#111827', - borderRadius: '0.75rem', - padding: '0.75rem', - maxWidth: '90vw', - maxHeight: '90vh', - display: 'flex', - flexDirection: 'column', - alignItems: 'center', - gap: '0.5rem', - overflow: 'visible', + position: "relative", + background: "#111827", + borderRadius: "0.75rem", + padding: "0.75rem", + maxWidth: "90vw", + maxHeight: "90vh", + display: "flex", + flexDirection: "column", + alignItems: "center", + gap: "0.5rem", + overflow: "visible", }} > {/* Previous page */} @@ -216,22 +216,22 @@ const PagePreviewModal: React.FC = ({ }} disabled={!canGoPrevious} style={{ - position: 'absolute', + position: "absolute", left: 0, - top: '50%', - transform: 'translate(-50%, -50%)', - width: '2.5rem', - height: '2.5rem', - borderRadius: '999px', - border: 'none', - background: canGoPrevious ? '#374151' : '#1f2937', - color: canGoPrevious ? '#e5e7eb' : '#6b7280', - cursor: canGoPrevious ? 'pointer' : 'default', - fontSize: '1.35rem', + top: "50%", + transform: "translate(-50%, -50%)", + width: "2.5rem", + height: "2.5rem", + borderRadius: "999px", + border: "none", + background: canGoPrevious ? "#374151" : "#1f2937", + color: canGoPrevious ? "#e5e7eb" : "#6b7280", + cursor: canGoPrevious ? "pointer" : "default", + fontSize: "1.35rem", lineHeight: 1, - display: 'flex', - alignItems: 'center', - justifyContent: 'center', + display: "flex", + alignItems: "center", + justifyContent: "center", zIndex: 2, }} title="Previous page (←)" @@ -249,22 +249,22 @@ const PagePreviewModal: React.FC = ({ }} disabled={!canGoNext} style={{ - position: 'absolute', + position: "absolute", right: 0, - top: '50%', - transform: 'translate(50%, -50%)', - width: '2.5rem', - height: '2.5rem', - borderRadius: '999px', - border: 'none', - background: canGoNext ? '#374151' : '#1f2937', - color: canGoNext ? '#e5e7eb' : '#6b7280', - cursor: canGoNext ? 'pointer' : 'default', - fontSize: '1.35rem', + top: "50%", + transform: "translate(50%, -50%)", + width: "2.5rem", + height: "2.5rem", + borderRadius: "999px", + border: "none", + background: canGoNext ? "#374151" : "#1f2937", + color: canGoNext ? "#e5e7eb" : "#6b7280", + cursor: canGoNext ? "pointer" : "default", + fontSize: "1.35rem", lineHeight: 1, - display: 'flex', - alignItems: 'center', - justifyContent: 'center', + display: "flex", + alignItems: "center", + justifyContent: "center", zIndex: 2, }} title="Next page (→)" @@ -281,22 +281,22 @@ const PagePreviewModal: React.FC = ({ onClose(); }} style={{ - position: 'absolute', + position: "absolute", top: 0, right: 0, - transform: 'translate(50%, -50%)', - width: '2.25rem', - height: '2.25rem', - borderRadius: '999px', - border: 'none', - background: '#374151', - color: '#e5e7eb', - cursor: 'pointer', - fontSize: '1.2rem', + transform: "translate(50%, -50%)", + width: "2.25rem", + height: "2.25rem", + borderRadius: "999px", + border: "none", + background: "#374151", + color: "#e5e7eb", + cursor: "pointer", + fontSize: "1.2rem", lineHeight: 1, - display: 'flex', - alignItems: 'center', - justifyContent: 'center', + display: "flex", + alignItems: "center", + justifyContent: "center", zIndex: 3, }} title="Close preview (Esc)" @@ -308,14 +308,14 @@ const PagePreviewModal: React.FC = ({ -
+
{positionLabel} · Original page {pageIndex + 1} · Rot {rotation}°
@@ -323,4 +323,4 @@ const PagePreviewModal: React.FC = ({ ); }; -export default PagePreviewModal; \ No newline at end of file +export default PagePreviewModal; diff --git a/src/components/PageWorkspace/CopyPagesDialog.tsx b/src/components/PageWorkspace/CopyPagesDialog.tsx new file mode 100644 index 0000000..6b4275a --- /dev/null +++ b/src/components/PageWorkspace/CopyPagesDialog.tsx @@ -0,0 +1,224 @@ +import React, { useEffect } from "react"; + +interface CopyPagesDialogProps { + selectedCount: number; + pageCount: number; + targetPosition: string; + error: string | null; + onTargetPositionChange: (value: string) => void; + onCancel: () => void; + onConfirm: (e?: React.FormEvent) => void; +} + +const CopyPagesDialog: React.FC = ({ + selectedCount, + pageCount, + targetPosition, + error, + onTargetPositionChange, + onCancel, + onConfirm, +}) => { + useEffect(() => { + const handleKeyDown = (e: KeyboardEvent) => { + if (e.key === "Escape") { + e.preventDefault(); + onCancel(); + } + }; + + window.addEventListener("keydown", handleKeyDown); + + return () => { + window.removeEventListener("keydown", handleKeyDown); + }; + }, [onCancel]); + + return ( +
{ + if (e.target === e.currentTarget) { + onCancel(); + } + }} + style={{ + position: "fixed", + inset: 0, + zIndex: 60, + background: "rgba(15, 23, 42, 0.55)", + display: "flex", + alignItems: "center", + justifyContent: "center", + padding: "1rem", + }} + > +
+
+

+ Copy selected pages +

+ + +
+ +

+ Copy{" "} + + {selectedCount === 1 + ? "1 selected page" + : `${selectedCount} selected pages`} + {" "} + to a new position. +

+ + + +
+
1 = before the first page
+
{pageCount + 1} = after the last page
+
+ + {error && ( +
+ {error} +
+ )} + +
+ + + +
+
+
+ ); +}; + +export default CopyPagesDialog; diff --git a/src/components/PageWorkspace/DropIndicator.tsx b/src/components/PageWorkspace/DropIndicator.tsx new file mode 100644 index 0000000..2437960 --- /dev/null +++ b/src/components/PageWorkspace/DropIndicator.tsx @@ -0,0 +1,27 @@ +import React from "react"; + +interface DropIndicatorProps { + side: "left" | "right" | "end"; + color: string; +} + +const DropIndicator: React.FC = ({ side, color }) => { + const isEnd = side === "end"; + + return ( +
+ ); +}; + +export default DropIndicator; diff --git a/src/components/PageWorkspace/PageCard.tsx b/src/components/PageWorkspace/PageCard.tsx new file mode 100644 index 0000000..7592616 --- /dev/null +++ b/src/components/PageWorkspace/PageCard.tsx @@ -0,0 +1,213 @@ +import React from "react"; +import type { PageRef } from "../../pdf/pdfTypes"; +import DropIndicator from "./DropIndicator"; + +interface PageCardProps { + page: PageRef; + visualIndex: number; + thumbnail?: string; + selected: boolean; + isDraggingCard: boolean; + isBusy: boolean; + isCopyDragging: boolean; + showLeftLine: boolean; + showRightLine: boolean; + dropIndicatorColor: string; + onDragStart: React.DragEventHandler; + onDragEnd: React.DragEventHandler; + onDragOver: React.DragEventHandler; + onOpenPreview: () => void; + onToggleSelect: React.MouseEventHandler; + onRotateClockwise: () => void; + onRotateCounterclockwise: () => void; + onDelete: () => void; +} + +const pageActionButtonStyle: React.CSSProperties = { + border: "none", + borderRadius: "999px", + padding: "0.15rem 0.4rem", + fontSize: "0.75rem", + cursor: "pointer", +}; + +const PageCard: React.FC = ({ + page, + visualIndex, + thumbnail, + selected, + isDraggingCard, + isBusy, + isCopyDragging, + showLeftLine, + showRightLine, + dropIndicatorColor, + onDragStart, + onDragEnd, + onDragOver, + onOpenPreview, + onToggleSelect, + onRotateClockwise, + onRotateCounterclockwise, + onDelete, +}) => { + const background = isDraggingCard + ? isCopyDragging + ? "#dcfce7" + : "#dbeafe" + : selected + ? "#eff6ff" + : "#f9fafb"; + + return ( +
+ + + {showLeftLine && } + {showRightLine && ( + + )} + +
+ {thumbnail ? ( + {`Page + ) : ( +
+ )} +
+ + + Page {page.sourcePageIndex + 1} + + + Pos {visualIndex + 1} · Rot {page.rotation}° + + +
+ + + + + +
+
+ ); +}; + +export default PageCard; diff --git a/src/components/PageWorkspace/PageGrid.tsx b/src/components/PageWorkspace/PageGrid.tsx new file mode 100644 index 0000000..2c23312 --- /dev/null +++ b/src/components/PageWorkspace/PageGrid.tsx @@ -0,0 +1,130 @@ +import React from "react"; +import type { PageRef } from "../../pdf/pdfTypes"; +import DropIndicator from "./DropIndicator"; +import PageCard from "./PageCard"; + +interface PageGridProps { + pages: PageRef[]; + thumbnails: Record; + selectedPageIds: string[]; + isBusy: boolean; + draggingIndex: number | null; + dropIndex: number | null; + draggingSelectionActive: boolean; + isCopyDragging: boolean; + dropIndicatorColor: string; + onDragStart: (visualIndex: number) => React.DragEventHandler; + onDragEnd: React.DragEventHandler; + onCardDragOver: ( + visualIndex: number, + ) => React.DragEventHandler; + onEndSlotDragOver: React.DragEventHandler; + onDrop: React.DragEventHandler; + onOpenPreview: (pageId: string) => void; + onToggleSelect: ( + pageId: string, + visualIndex: number, + ) => React.MouseEventHandler; + onRotateClockwise: (pageId: string) => void; + onRotateCounterclockwise: (pageId: string) => void; + onDelete: (pageId: string) => void; +} + +const PageGrid: React.FC = ({ + pages, + thumbnails, + selectedPageIds, + isBusy, + draggingIndex, + dropIndex, + draggingSelectionActive, + isCopyDragging, + dropIndicatorColor, + onDragStart, + onDragEnd, + onCardDragOver, + onEndSlotDragOver, + onDrop, + onOpenPreview, + onToggleSelect, + onRotateClockwise, + onRotateCounterclockwise, + onDelete, +}) => { + const isSelected = (pageId: string) => selectedPageIds.includes(pageId); + + const showLeftLine = (visualIndex: number) => + dropIndex !== null && dropIndex === visualIndex && draggingIndex !== null; + + const showRightLine = (visualIndex: number) => + dropIndex !== null && + dropIndex === visualIndex + 1 && + draggingIndex !== null; + + const showEndLine = () => + dropIndex !== null && dropIndex === pages.length && draggingIndex !== null; + + return ( +
+ {pages.map((page, visualIndex) => { + const selected = isSelected(page.id); + const isDraggingCard = + draggingIndex != null && + ((draggingSelectionActive && selected) || + (!draggingSelectionActive && visualIndex === draggingIndex)); + + return ( + onOpenPreview(page.id)} + onToggleSelect={onToggleSelect(page.id, visualIndex)} + onRotateClockwise={() => onRotateClockwise(page.id)} + onRotateCounterclockwise={() => onRotateCounterclockwise(page.id)} + onDelete={() => onDelete(page.id)} + /> + ); + })} + + {pages.length > 0 && ( +
+ {showEndLine() && ( + + )} +
+ )} +
+ ); +}; + +export default PageGrid; diff --git a/src/components/PageWorkspace/PageSelectionToolbar.tsx b/src/components/PageWorkspace/PageSelectionToolbar.tsx new file mode 100644 index 0000000..68ec869 --- /dev/null +++ b/src/components/PageWorkspace/PageSelectionToolbar.tsx @@ -0,0 +1,112 @@ +import React from "react"; + +interface PageSelectionToolbarProps { + selectedCount: number; + onCopySelected: () => void; + onDeleteSelected: () => void; + onSelectAll: () => void; + onClearSelection: () => void; +} + +const pillButtonStyle: React.CSSProperties = { + border: "none", + borderRadius: "999px", + padding: "0.15rem 0.6rem", + fontSize: "0.8rem", +}; + +const PageSelectionToolbar: React.FC = ({ + selectedCount, + onCopySelected, + onDeleteSelected, + onSelectAll, + onClearSelection, +}) => { + const hasSelection = selectedCount > 0; + + return ( +
+ + Selected: {selectedCount} + + +
+ {hasSelection && ( + + )} + + {hasSelection && ( + + )} + + + + +
+
+ ); +}; + +export default PageSelectionToolbar; diff --git a/src/components/ReorderPanel.tsx b/src/components/ReorderPanel.tsx index d26e839..a109cc8 100644 --- a/src/components/ReorderPanel.tsx +++ b/src/components/ReorderPanel.tsx @@ -1,5 +1,8 @@ -import React, { useEffect, useState, useRef } from 'react'; -import type { PageRef } from '../pdf/pdfTypes'; +import React, { useRef, useState } from "react"; +import type { PageRef } from "../pdf/pdfTypes"; +import CopyPagesDialog from "./PageWorkspace/CopyPagesDialog"; +import PageGrid from "./PageWorkspace/PageGrid"; +import PageSelectionToolbar from "./PageWorkspace/PageSelectionToolbar"; interface ReorderPanelProps { pages: PageRef[]; @@ -17,7 +20,7 @@ interface ReorderPanelProps { onToggleSelect: ( pageId: string, visualIndex: number, - e: React.MouseEvent + e: React.MouseEvent, ) => void; onSelectAll: () => void; @@ -48,13 +51,11 @@ const ReorderPanel: React.FC = ({ const [isCopyDragging, setIsCopyDragging] = useState(false); const [copyDialogOpen, setCopyDialogOpen] = useState(false); - const [copyTargetPosition, setCopyTargetPosition] = useState(''); + const [copyTargetPosition, setCopyTargetPosition] = useState(""); const [copyDialogError, setCopyDialogError] = useState(null); const dragGhostRef = useRef(null); - const isSelected = (pageId: string) => selectedPageIds.includes(pageId); - const cleanupDragGhost = () => { if (dragGhostRef.current && dragGhostRef.current.parentNode) { dragGhostRef.current.parentNode.removeChild(dragGhostRef.current); @@ -71,7 +72,7 @@ const ReorderPanel: React.FC = ({ if (!draggedPage) return []; const selectedInVisualOrder = pages.filter((page) => - selectedPageIds.includes(page.id) + selectedPageIds.includes(page.id), ); const draggingIsSelected = @@ -84,20 +85,20 @@ const ReorderPanel: React.FC = ({ const createDragGhost = (e: React.DragEvent, count: number) => { cleanupDragGhost(); - const ghost = document.createElement('div'); - ghost.textContent = count === 1 ? '1 page' : `${count} pages`; + const ghost = document.createElement("div"); + ghost.textContent = count === 1 ? "1 page" : `${count} pages`; - ghost.style.position = 'fixed'; - ghost.style.top = '0'; - ghost.style.left = '0'; - ghost.style.padding = '4px 8px'; - ghost.style.borderRadius = '999px'; - ghost.style.background = '#111827'; - ghost.style.color = '#e5e7eb'; - ghost.style.fontSize = '12px'; + ghost.style.position = "fixed"; + ghost.style.top = "0"; + ghost.style.left = "0"; + ghost.style.padding = "4px 8px"; + ghost.style.borderRadius = "999px"; + ghost.style.background = "#111827"; + ghost.style.color = "#e5e7eb"; + ghost.style.fontSize = "12px"; ghost.style.fontFamily = 'system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif'; - ghost.style.zIndex = '9999'; + ghost.style.zIndex = "9999"; document.body.appendChild(ghost); dragGhostRef.current = ghost; @@ -106,6 +107,13 @@ const ReorderPanel: React.FC = ({ e.dataTransfer.setDragImage(ghost, rect.width / 2, rect.height / 2); }; + const resetDragState = () => { + cleanupDragGhost(); + setDraggingIndex(null); + setDropIndex(null); + setIsCopyDragging(false); + }; + const handleDragStart = (visualIndex: number) => (e: React.DragEvent) => { setDraggingIndex(visualIndex); setDropIndex(visualIndex); @@ -113,19 +121,16 @@ const ReorderPanel: React.FC = ({ const copying = isCopyModifierPressed(e); setIsCopyDragging(copying); - e.dataTransfer.effectAllowed = 'copyMove'; - e.dataTransfer.dropEffect = copying ? 'copy' : 'move'; - e.dataTransfer.setData('text/plain', String(visualIndex)); + e.dataTransfer.effectAllowed = "copyMove"; + e.dataTransfer.dropEffect = copying ? "copy" : "move"; + e.dataTransfer.setData("text/plain", String(visualIndex)); const draggedPages = getDraggedPages(visualIndex); createDragGhost(e, draggedPages.length); }; const handleDragEnd = () => { - cleanupDragGhost(); - setDraggingIndex(null); - setDropIndex(null); - setIsCopyDragging(false); + resetDragState(); }; const handleCardDragOver = (visualIndex: number) => (e: React.DragEvent) => { @@ -136,7 +141,7 @@ const ReorderPanel: React.FC = ({ const copying = isCopyModifierPressed(e); setIsCopyDragging(copying); - e.dataTransfer.dropEffect = copying ? 'copy' : 'move'; + e.dataTransfer.dropEffect = copying ? "copy" : "move"; const rect = (e.currentTarget as HTMLDivElement).getBoundingClientRect(); const x = e.clientX - rect.left; @@ -153,7 +158,7 @@ const ReorderPanel: React.FC = ({ const copying = isCopyModifierPressed(e); setIsCopyDragging(copying); - e.dataTransfer.dropEffect = copying ? 'copy' : 'move'; + e.dataTransfer.dropEffect = copying ? "copy" : "move"; setDropIndex(pages.length); }; @@ -172,7 +177,7 @@ const ReorderPanel: React.FC = ({ if (shouldCopy) { onCopyPagesToSlot( draggedPages.map((page) => page.id), - dropIndex + dropIndex, ); setDraggingIndex(null); @@ -207,35 +212,23 @@ const ReorderPanel: React.FC = ({ setIsCopyDragging(false); }; - const handleDeleteClick = (pageId: string) => () => { + const handleDeleteClick = (pageId: string) => { onDelete(pageId); setDraggingIndex(null); setDropIndex(null); }; - const handleRotateClickClockwise = (pageId: string) => () => { - onRotateClockwise(pageId); - }; - - const handleRotateClickCounterclockwise = (pageId: string) => () => { - onRotateCounterclockwise(pageId); - }; - - const handleCardClick = (pageId: string) => () => { - onOpenPreview(pageId); - }; - const handleCheckboxClick = (pageId: string, visualIndex: number) => - (e: React.MouseEvent) => { - e.stopPropagation(); // don't trigger preview - onToggleSelect(pageId, visualIndex, e); - }; + (e: React.MouseEvent) => { + e.stopPropagation(); + onToggleSelect(pageId, visualIndex, e); + }; const handleCopySelectedClick = () => { if (selectedPageIds.length === 0) return; - setCopyTargetPosition(String(pages.length + 1)); // default: after last page + setCopyTargetPosition(String(pages.length + 1)); setCopyDialogError(null); setCopyDialogOpen(true); }; @@ -245,11 +238,16 @@ const ReorderPanel: React.FC = ({ setCopyDialogError(null); }; + const handleCopyTargetPositionChange = (value: string) => { + setCopyTargetPosition(value); + setCopyDialogError(null); + }; + const handleCopyDialogConfirm = (e?: React.FormEvent) => { e?.preventDefault(); if (selectedPageIds.length === 0) { - setCopyDialogError('No pages selected.'); + setCopyDialogError("No pages selected."); return; } @@ -267,23 +265,6 @@ const ReorderPanel: React.FC = ({ setCopyDialogError(null); }; - useEffect(() => { - if (!copyDialogOpen) return; - - const handleKeyDown = (e: KeyboardEvent) => { - if (e.key === 'Escape') { - e.preventDefault(); - handleCopyDialogCancel(); - } - }; - - window.addEventListener('keydown', handleKeyDown); - - return () => { - window.removeEventListener('keydown', handleKeyDown); - }; - }, [copyDialogOpen]); - if (!hasPdf) { return (
@@ -293,554 +274,66 @@ const ReorderPanel: React.FC = ({ ); } - const showLeftLine = (visualIndex: number) => - dropIndex !== null && dropIndex === visualIndex && draggingIndex !== null; - - const showRightLine = (visualIndex: number) => - dropIndex !== null && - dropIndex === visualIndex + 1 && - draggingIndex !== null; - - const showEndLine = () => - dropIndex !== null && dropIndex === pages.length && draggingIndex !== null; - - // For highlighting the whole selection while dragging it const draggingPage = draggingIndex != null ? pages[draggingIndex] : null; const draggingSelectionActive = draggingPage != null && selectedPageIds.length > 0 && selectedPageIds.includes(draggingPage.id); - const dropIndicatorColor = isCopyDragging ? '#16a34a' : '#2563eb'; + const dropIndicatorColor = isCopyDragging ? "#16a34a" : "#2563eb"; return ( <>

Pages

-

+

Tap/click a page to preview it. Use the checkbox to select pages - (Shift for ranges). Drag to reorder; dragging a selected page moves the - whole selection. Hold Ctrl/⌘ while dropping to copy instead of move. - Shortcuts: Ctrl/⌘+A selects all, Delete removes selected pages, Esc clears - selection. + (Shift for ranges). Drag to reorder; dragging a selected page moves + the whole selection. Hold Ctrl/⌘ while dropping to copy instead of + move. Shortcuts: Ctrl/⌘+A selects all, Delete removes selected pages, + Esc clears selection.

-
- - Selected: {selectedPageIds.length} - -
- {selectedPageIds.length > 0 && ( - - )} - {selectedPageIds.length > 0 && ( - - )} - - -
-
+ -
- {pages.map((page, visualIndex) => { - const thumb = thumbnails[page.id]; - const rotation = page.rotation; - const selected = isSelected(page.id); - - const isDraggingCard = - draggingIndex != null && - ((draggingSelectionActive && selected) || - (!draggingSelectionActive && visualIndex === draggingIndex)); - - return ( -
- {/* selection checkbox */} - - - {/* left drop indicator */} - {showLeftLine(visualIndex) && ( -
- )} - - {/* right drop indicator */} - {showRightLine(visualIndex) && ( -
- )} - -
- {thumb ? ( - {`Page - ) : ( -
- )} -
- - Page {page.sourcePageIndex + 1} - - Pos {visualIndex + 1} · Rot {rotation}° - - -
- - - -
-
- ); - })} - - {/* end slot for dropping after the last card */} - {pages.length > 0 && ( -
- {showEndLine() && ( -
- )} -
- )} -
+ onOpenPreview={onOpenPreview} + onToggleSelect={handleCheckboxClick} + onRotateClockwise={onRotateClockwise} + onRotateCounterclockwise={onRotateCounterclockwise} + onDelete={handleDeleteClick} + />
{copyDialogOpen && ( -
{ - if (e.target === e.currentTarget) { - handleCopyDialogCancel(); - } - }} - style={{ - position: 'fixed', - inset: 0, - zIndex: 60, - background: 'rgba(15, 23, 42, 0.55)', - display: 'flex', - alignItems: 'center', - justifyContent: 'center', - padding: '1rem', - }} - > -
-
-

- Copy selected pages -

- - -
- -

- Copy{' '} - - {selectedPageIds.length === 1 - ? '1 selected page' - : `${selectedPageIds.length} selected pages`} - {' '} - to a new position. -

- - - -
-
1 = before the first page
-
{pages.length + 1} = after the last page
-
- - {copyDialogError && ( -
- {copyDialogError} -
- )} - -
- - - -
-
-
+ )} ); diff --git a/src/components/WorkspacePanel.tsx b/src/components/WorkspacePanel.tsx index 3759c23..bb1e2ae 100644 --- a/src/components/WorkspacePanel.tsx +++ b/src/components/WorkspacePanel.tsx @@ -1,327 +1,327 @@ -import React from 'react'; -import type { WorkspaceSummary } from '../workspace/workspaceTypes'; -import type { WorkspaceCommandRecord } from '../workspace/workspaceCommands'; +import React from "react"; +import type { WorkspaceSummary } from "../workspace/workspaceTypes"; +import type { WorkspaceCommandRecord } from "../workspace/workspaceCommands"; interface WorkspacePanelProps { - hasPdf: boolean; - isBusy: boolean; + hasPdf: boolean; + isBusy: boolean; - activeWorkspaceId: string | null; - workspaceName: string; - workspaceDirty: boolean; - workspaceMessage: string | null; + activeWorkspaceId: string | null; + workspaceName: string; + workspaceDirty: boolean; + workspaceMessage: string | null; - workspaces: WorkspaceSummary[]; - history: WorkspaceCommandRecord[]; - redoHistory: WorkspaceCommandRecord[]; + workspaces: WorkspaceSummary[]; + history: WorkspaceCommandRecord[]; + redoHistory: WorkspaceCommandRecord[]; - onWorkspaceNameChange: (value: string) => void; - onSaveWorkspace: () => void; - onLoadWorkspace: (workspaceId: string) => void; - onDeleteWorkspace: (workspaceId: string) => void; - onRefreshWorkspaces: () => void; - onResetWorkspace: () => void; - onUndo: () => void; - onRedo: () => void; + onWorkspaceNameChange: (value: string) => void; + onSaveWorkspace: () => void; + onLoadWorkspace: (workspaceId: string) => void; + onDeleteWorkspace: (workspaceId: string) => void; + onRefreshWorkspaces: () => void; + onResetWorkspace: () => void; + onUndo: () => void; + onRedo: () => void; } const WorkspacePanel: React.FC = ({ - hasPdf, - isBusy, - activeWorkspaceId, - workspaceName, - workspaceDirty, - workspaceMessage, - workspaces, - history, - redoHistory, - onWorkspaceNameChange, - onSaveWorkspace, - onLoadWorkspace, - onDeleteWorkspace, - onRefreshWorkspaces, - onResetWorkspace, - onUndo, - onRedo, + hasPdf, + isBusy, + activeWorkspaceId, + workspaceName, + workspaceDirty, + workspaceMessage, + workspaces, + history, + redoHistory, + onWorkspaceNameChange, + onSaveWorkspace, + onLoadWorkspace, + onDeleteWorkspace, + onRefreshWorkspaces, + onResetWorkspace, + onUndo, + onRedo, }) => { - const canUndo = history.length > 0; - const canRedo = redoHistory.length > 0; + const canUndo = history.length > 0; + const canRedo = redoHistory.length > 0; - const latestUndo = history[history.length - 1]; - const latestRedo = redoHistory[redoHistory.length - 1]; + const latestUndo = history[history.length - 1]; + const latestRedo = redoHistory[redoHistory.length - 1]; - return ( -
-

Workspace

+ return ( +
+

Workspace

-

- Save named workspaces in this browser. PDF binaries are stored in - IndexedDB; nothing is uploaded. -

+

+ Save named workspaces in this browser. PDF binaries are stored in + IndexedDB; nothing is uploaded. +

+ +
+ onWorkspaceNameChange(e.target.value)} + placeholder="Workspace name" + disabled={!hasPdf || isBusy} + style={{ + flex: "1 1 220px", + minWidth: 0, + padding: "0.45rem 0.55rem", + borderRadius: "0.5rem", + border: "1px solid #d1d5db", + fontSize: "0.9rem", + }} + /> + + + + + + + + + + +
+ + {workspaceDirty && hasPdf && ( +
+ Unsaved workspace changes. +
+ )} + + {workspaceMessage && ( +
+ {workspaceMessage} +
+ )} + + {workspaces.length > 0 && ( +
+ Saved workspaces + +
+ {workspaces.map((workspace) => { + const active = workspace.id === activeWorkspaceId; + + return ( +
+
+
+ {workspace.name} + {active && ( + · active + )} +
+ +
+ {workspace.pdfName} · source pages:{" "} + {workspace.sourcePageCount} · workspace pages:{" "} + {workspace.workspacePageCount} · undo:{" "} + {workspace.historyCount} · redo: {workspace.redoCount} · + updated {new Date(workspace.updatedAt).toLocaleString()} +
+
+ +
+ + + +
+
+ ); + })} +
+
+ )} + + {(history.length > 0 || redoHistory.length > 0) && ( +
+ + Command history ({history.length} undo / {redoHistory.length} redo) + + +
+ {history.map((entry, index) => ( +
+ + Undo {history.length - index}. {entry.label} + +
+ + {new Date(entry.timestamp).toLocaleString()} + +
+ ))}
- onWorkspaceNameChange(e.target.value)} - placeholder="Workspace name" - disabled={!hasPdf || isBusy} - style={{ - flex: '1 1 220px', - minWidth: 0, - padding: '0.45rem 0.55rem', - borderRadius: '0.5rem', - border: '1px solid #d1d5db', - fontSize: '0.9rem', - }} - /> + style={{ + margin: "0.25rem 0", + borderRadius: "999px", + background: "#ecfdf5", + color: "#166534", + fontSize: "0.8rem", + fontWeight: 600, + alignSelf: "flex-start", + border: "2px solid #166534", + width: "100%", + }} + >
- - - - - - - - - -
- - {workspaceDirty && hasPdf && ( + {redoHistory + .slice() + .reverse() + .map((entry, index) => (
- Unsaved workspace changes. + + Redo {index + 1}. {entry.label} + +
+ + {new Date(entry.timestamp).toLocaleString()} +
- )} - - {workspaceMessage && ( -
- {workspaceMessage} -
- )} - - {workspaces.length > 0 && ( -
- Saved workspaces - -
- {workspaces.map((workspace) => { - const active = workspace.id === activeWorkspaceId; - - return ( -
-
-
- {workspace.name} - {active && ( - · active - )} -
- -
- {workspace.pdfName} · source pages:{' '} - {workspace.sourcePageCount} · workspace pages:{' '} - {workspace.workspacePageCount} · undo:{' '} - {workspace.historyCount} · redo: {workspace.redoCount} · updated{' '} - {new Date(workspace.updatedAt).toLocaleString()} -
-
- -
- - - -
-
- ); - })} -
-
- )} - - {(history.length > 0 || redoHistory.length > 0) && ( -
- - Command history ({history.length} undo / {redoHistory.length} redo) - - -
- {history.map((entry, index) => ( -
- - Undo {history.length - index}. {entry.label} - -
- - {new Date(entry.timestamp).toLocaleString()} - -
- ))} - -
- -
- - {redoHistory - .slice() - .reverse() - .map((entry, index) => ( -
- - Redo {index + 1}. {entry.label} - -
- - {new Date(entry.timestamp).toLocaleString()} - -
- ))} -
-
- )} -
- ); + ))} +
+ + )} +
+ ); }; -export default WorkspacePanel; \ No newline at end of file +export default WorkspacePanel; diff --git a/src/hooks/usePdfGeneratedOutputs.ts b/src/hooks/usePdfGeneratedOutputs.ts new file mode 100644 index 0000000..20bb370 --- /dev/null +++ b/src/hooks/usePdfGeneratedOutputs.ts @@ -0,0 +1,128 @@ +import { useCallback, useEffect, useRef, useState } from "react"; +import type { SplitResult } from "../pdf/pdfTypes"; + +export interface PdfDownload { + id: string; + filename: string; + url: string; +} + +export interface SplitPdfDownload extends PdfDownload { + pageIndex: number; +} + +function revokeDownload(download: PdfDownload | null): void { + if (download) { + URL.revokeObjectURL(download.url); + } +} + +function revokeDownloads(downloads: PdfDownload[]): void { + downloads.forEach(revokeDownload); +} + +function createDownload(id: string, filename: string, blob: Blob): PdfDownload { + return { + id, + filename, + url: URL.createObjectURL(blob), + }; +} + +export function usePdfGeneratedOutputs() { + const [splitDownloads, setSplitDownloads] = useState([]); + const [subsetDownload, setSubsetDownload] = useState( + null, + ); + const [exportDownload, setExportDownload] = useState( + null, + ); + + const splitDownloadsRef = useRef([]); + const subsetDownloadRef = useRef(null); + const exportDownloadRef = useRef(null); + + const replaceSplitResults = useCallback((results: SplitResult[]) => { + const nextDownloads: SplitPdfDownload[] = results.map((result) => ({ + ...createDownload( + `split-${result.pageIndex}-${result.filename}`, + result.filename, + result.blob, + ), + pageIndex: result.pageIndex, + })); + + revokeDownloads(splitDownloadsRef.current); + splitDownloadsRef.current = nextDownloads; + setSplitDownloads(nextDownloads); + }, []); + + const clearSplitResults = useCallback(() => { + revokeDownloads(splitDownloadsRef.current); + splitDownloadsRef.current = []; + setSplitDownloads([]); + }, []); + + const replaceSubsetResult = useCallback((blob: Blob, filename: string) => { + const nextDownload = createDownload("subset", filename, blob); + + revokeDownload(subsetDownloadRef.current); + subsetDownloadRef.current = nextDownload; + setSubsetDownload(nextDownload); + }, []); + + const clearSubsetResult = useCallback(() => { + revokeDownload(subsetDownloadRef.current); + subsetDownloadRef.current = null; + setSubsetDownload(null); + }, []); + + const replaceExportResult = useCallback((blob: Blob, filename: string) => { + const nextDownload = createDownload("export", filename, blob); + + revokeDownload(exportDownloadRef.current); + exportDownloadRef.current = nextDownload; + setExportDownload(nextDownload); + }, []); + + const clearExportResult = useCallback(() => { + revokeDownload(exportDownloadRef.current); + exportDownloadRef.current = null; + setExportDownload(null); + }, []); + + const clearAllResults = useCallback(() => { + revokeDownloads(splitDownloadsRef.current); + revokeDownload(subsetDownloadRef.current); + revokeDownload(exportDownloadRef.current); + + splitDownloadsRef.current = []; + subsetDownloadRef.current = null; + exportDownloadRef.current = null; + + setSplitDownloads([]); + setSubsetDownload(null); + setExportDownload(null); + }, []); + + useEffect(() => { + return () => { + revokeDownloads(splitDownloadsRef.current); + revokeDownload(subsetDownloadRef.current); + revokeDownload(exportDownloadRef.current); + }; + }, []); + + return { + splitDownloads, + subsetDownload, + exportDownload, + replaceSplitResults, + clearSplitResults, + replaceSubsetResult, + clearSubsetResult, + replaceExportResult, + clearExportResult, + clearAllResults, + }; +} diff --git a/src/main.tsx b/src/main.tsx index 5610739..81da346 100644 --- a/src/main.tsx +++ b/src/main.tsx @@ -1,10 +1,10 @@ -import React from 'react'; -import ReactDOM from 'react-dom/client'; -import App from './App'; -import './styles.css'; +import React from "react"; +import ReactDOM from "react-dom/client"; +import App from "./App"; +import "./styles.css"; -ReactDOM.createRoot(document.getElementById('root') as HTMLElement).render( +ReactDOM.createRoot(document.getElementById("root") as HTMLElement).render( - + , ); diff --git a/src/pdf/pdfService.ts b/src/pdf/pdfService.ts index 3891406..2bb1fd2 100644 --- a/src/pdf/pdfService.ts +++ b/src/pdf/pdfService.ts @@ -1,10 +1,20 @@ -import { PDFDocument, degrees } from 'pdf-lib'; -import type { PdfFile, PageRef, SplitResult, Range } from './pdfTypes'; +import { PDFDocument, degrees } from "pdf-lib"; +import type { PdfFile, PageRef, SplitResult, Range } from "./pdfTypes"; function createId() { return Math.random().toString(36).slice(2); } +function pdfBytesToArrayBuffer(bytes: Uint8Array): ArrayBuffer { + const buffer = new ArrayBuffer(bytes.byteLength); + new Uint8Array(buffer).set(bytes); + return buffer; +} + +function pdfBytesToBlob(bytes: Uint8Array): Blob { + return new Blob([pdfBytesToArrayBuffer(bytes)], { type: "application/pdf" }); +} + export async function loadPdfFromFile(file: File): Promise { const arrayBuffer = await file.arrayBuffer(); const doc = await PDFDocument.load(arrayBuffer); @@ -21,10 +31,10 @@ export async function loadPdfFromFile(file: File): Promise { export async function mergePdfFiles( basePdf: PdfFile, newPdf: PdfFile, - insertAt: number + insertAt: number, ): Promise { - const baseDoc = basePdf.doc ?? await PDFDocument.load(basePdf.arrayBuffer); - const newDoc = newPdf.doc ?? await PDFDocument.load(newPdf.arrayBuffer); + const baseDoc = basePdf.doc ?? (await PDFDocument.load(basePdf.arrayBuffer)); + const newDoc = newPdf.doc ?? (await PDFDocument.load(newPdf.arrayBuffer)); const mergedDoc = await PDFDocument.create(); @@ -35,11 +45,11 @@ export async function mergePdfFiles( const basePages = await mergedDoc.copyPages( baseDoc, - Array.from({ length: basePageCount }, (_, i) => i) + Array.from({ length: basePageCount }, (_, i) => i), ); const newPages = await mergedDoc.copyPages( newDoc, - Array.from({ length: newPageCount }, (_, i) => i) + Array.from({ length: newPageCount }, (_, i) => i), ); for (let i = 0; i < clampedInsertAt; i += 1) { @@ -53,11 +63,10 @@ export async function mergePdfFiles( } const bytes = await mergedDoc.save(); - const buffer = new ArrayBuffer(bytes.byteLength); - new Uint8Array(buffer).set(bytes); + const buffer = pdfBytesToArrayBuffer(bytes); - const baseName = basePdf.name.replace(/\.pdf$/i, ''); - const newName = newPdf.name.replace(/\.pdf$/i, ''); + const baseName = basePdf.name.replace(/\.pdf$/i, ""); + const newName = newPdf.name.replace(/\.pdf$/i, ""); return { id: createId(), @@ -69,7 +78,7 @@ export async function mergePdfFiles( } export async function splitIntoSinglePages( - pdf: PdfFile + pdf: PdfFile, ): Promise { const { doc, name } = pdf; @@ -99,10 +108,10 @@ export async function splitIntoSinglePages( if (modificationDate) newDoc.setModificationDate(modificationDate); const bytes = await newDoc.save(); - const blob = new Blob([bytes], { type: 'application/pdf' }); + const blob = pdfBytesToBlob(bytes); - const base = name.replace(/\.pdf$/i, ''); - const filename = `${base}_page_${String(i + 1).padStart(3, '0')}.pdf`; + const base = name.replace(/\.pdf$/i, ""); + const filename = `${base}_page_${String(i + 1).padStart(3, "0")}.pdf`; results.push({ pageIndex: i, @@ -114,10 +123,7 @@ export async function splitIntoSinglePages( return results; } -export async function extractRange( - pdf: PdfFile, - range: Range -): Promise { +export async function extractRange(pdf: PdfFile, range: Range): Promise { const { doc } = pdf; const pageCount = doc.getPageCount(); @@ -125,7 +131,7 @@ export async function extractRange( const toIndex = Math.min(pageCount - 1, range.to - 1); if (fromIndex > toIndex) { - throw new Error('Invalid range: from > to'); + throw new Error("Invalid range: from > to"); } const newDoc = await PDFDocument.create(); @@ -136,7 +142,7 @@ export async function extractRange( copiedPages.forEach((p) => newDoc.addPage(p)); const bytes = await newDoc.save(); - return new Blob([bytes], { type: 'application/pdf' }); + return pdfBytesToBlob(bytes); } export async function mergePdfs(pdfs: PdfFile[]): Promise { @@ -150,26 +156,26 @@ export async function mergePdfs(pdfs: PdfFile[]): Promise { } const bytes = await newDoc.save(); - return new Blob([bytes], { type: 'application/pdf' }); + return pdfBytesToBlob(bytes); } export async function exportPages( pdf: PdfFile, - pages: PageRef[] + pages: PageRef[], ): Promise { const { doc } = pdf; const pageCount = doc.getPageCount(); if (pages.length === 0) { - throw new Error('Pages must contain at least one page'); + throw new Error("Pages must contain at least one page"); } if ( pages.some( - (page) => page.sourcePageIndex < 0 || page.sourcePageIndex >= pageCount + (page) => page.sourcePageIndex < 0 || page.sourcePageIndex >= pageCount, ) ) { - throw new Error('Pages contain invalid source page indices'); + throw new Error("Pages contain invalid source page indices"); } const newDoc = await PDFDocument.create(); @@ -180,7 +186,7 @@ export async function exportPages( copiedPages.forEach((page, idx) => { const angle = pages[idx].rotation; - if (typeof angle === 'number' && angle % 360 !== 0) { + if (typeof angle === "number" && angle % 360 !== 0) { page.setRotation(degrees(angle)); } @@ -188,13 +194,13 @@ export async function exportPages( }); const bytes = await newDoc.save(); - return new Blob([bytes], { type: 'application/pdf' }); + return pdfBytesToBlob(bytes); } export async function exportReordered( pdf: PdfFile, order: number[], - rotations?: Record + rotations?: Record, ): Promise { return exportPages( pdf, @@ -202,6 +208,6 @@ export async function exportReordered( id: String(sourcePageIndex), sourcePageIndex, rotation: rotations?.[sourcePageIndex] ?? 0, - })) + })), ); -} \ No newline at end of file +} diff --git a/src/pdf/pdfThumbnailService.ts b/src/pdf/pdfThumbnailService.ts index 4bfdff1..292afe0 100644 --- a/src/pdf/pdfThumbnailService.ts +++ b/src/pdf/pdfThumbnailService.ts @@ -1,5 +1,5 @@ -import * as pdfjsLib from 'pdfjs-dist'; -import pdfjsWorker from 'pdfjs-dist/build/pdf.worker?worker&url'; +import * as pdfjsLib from "pdfjs-dist"; +import pdfjsWorker from "pdfjs-dist/build/pdf.worker?worker&url"; // pdf.js worker setup for Vite // eslint-disable-next-line @typescript-eslint/no-explicit-any @@ -37,13 +37,12 @@ interface ThumbnailGenerationOptions { onThumbnail?: (update: ThumbnailUpdate) => void; } - /** * Unrotated thumbnails – used e.g. in the Split/Extract view. */ export async function generateThumbnailsProgressive( arrayBuffer: ArrayBuffer, - options: ThumbnailGenerationOptions = {} + options: ThumbnailGenerationOptions = {}, ): Promise { return generateThumbnailsInternal(arrayBuffer, {}, options); } @@ -54,7 +53,7 @@ export async function generateThumbnailsProgressive( export async function generateThumbnailsWithRotationsProgressive( arrayBuffer: ArrayBuffer, rotations: RotationsMap, - options: ThumbnailGenerationOptions = {} + options: ThumbnailGenerationOptions = {}, ): Promise { return generateThumbnailsInternal(arrayBuffer, rotations, options); } @@ -62,7 +61,7 @@ export async function generateThumbnailsWithRotationsProgressive( async function generateThumbnailsInternal( arrayBuffer: ArrayBuffer, rotations: RotationsMap, - options: ThumbnailGenerationOptions = {} + options: ThumbnailGenerationOptions = {}, ): Promise { const maxHeight = options.maxHeight ?? 150; const maxWidth = options.maxWidth ?? 140; @@ -73,15 +72,15 @@ async function generateThumbnailsInternal( const loadingTask = pdfjsLib.getDocument({ data: dataCopy }); const pdf = await loadingTask.promise; - const thumbs = Array(pdf.numPages).fill(''); + const thumbs = Array(pdf.numPages).fill(""); const pageNums = options.pageIndices ? Array.from( new Set( options.pageIndices .filter((pageIndex) => pageIndex >= 0 && pageIndex < pdf.numPages) - .map((pageIndex) => pageIndex + 1) - ) + .map((pageIndex) => pageIndex + 1), + ), ) : Array.from({ length: pdf.numPages }, (_, index) => index + 1); @@ -99,7 +98,7 @@ async function generateThumbnailsInternal( pageIndex, rotations, maxHeight, - maxWidth + maxWidth, ); if (signal?.aborted) return; @@ -112,7 +111,7 @@ async function generateThumbnailsInternal( while (!signal?.aborted) { const pageNum = pageNums[nextPageIndex]; nextPageIndex += 1; - + if (pageNum == null) return; await renderOne(pageNum); @@ -133,11 +132,15 @@ async function generateThumbnailsInternal( } async function renderPageThumbnail( - page: Awaited['promise']>['getPage']>>, + page: Awaited< + ReturnType< + Awaited["promise"]>["getPage"] + > + >, originalIndex: number, rotations: RotationsMap, maxHeight: number, - maxWidth: number + maxWidth: number, ): Promise { const viewport = page.getViewport({ scale: 1 }); const scaleH = maxHeight / viewport.height; @@ -145,10 +148,10 @@ async function renderPageThumbnail( const scale = Math.min(scaleH, scaleW); const scaledViewport = page.getViewport({ scale }); - const baseCanvas = document.createElement('canvas'); - const baseCtx = baseCanvas.getContext('2d'); + const baseCanvas = document.createElement("canvas"); + const baseCtx = baseCanvas.getContext("2d"); - if (!baseCtx) return ''; + if (!baseCtx) return ""; baseCanvas.width = scaledViewport.width; baseCanvas.height = scaledViewport.height; @@ -164,14 +167,14 @@ async function renderPageThumbnail( const rotationDeg = ((rotationDegRaw % 360) + 360) % 360; if (rotationDeg === 0) { - return baseCanvas.toDataURL('image/png'); + return baseCanvas.toDataURL("image/png"); } - const rotatedCanvas = document.createElement('canvas'); - const rotatedCtx = rotatedCanvas.getContext('2d'); + const rotatedCanvas = document.createElement("canvas"); + const rotatedCtx = rotatedCanvas.getContext("2d"); if (!rotatedCtx) { - return baseCanvas.toDataURL('image/png'); + return baseCanvas.toDataURL("image/png"); } const rad = (rotationDeg * Math.PI) / 180; @@ -204,5 +207,5 @@ async function renderPageThumbnail( rotatedCtx.drawImage(baseCanvas, 0, 0); rotatedCtx.restore(); - return rotatedCanvas.toDataURL('image/png'); -} \ No newline at end of file + return rotatedCanvas.toDataURL("image/png"); +} diff --git a/src/pdf/pdfTypes.ts b/src/pdf/pdfTypes.ts index e84e997..81a02e7 100644 --- a/src/pdf/pdfTypes.ts +++ b/src/pdf/pdfTypes.ts @@ -1,4 +1,4 @@ -import type { PDFDocument } from 'pdf-lib'; +import type { PDFDocument } from "pdf-lib"; export interface PdfFile { id: string; diff --git a/src/pdf/usePdfThumbnails.ts b/src/pdf/usePdfThumbnails.ts new file mode 100644 index 0000000..0a5eaea --- /dev/null +++ b/src/pdf/usePdfThumbnails.ts @@ -0,0 +1,212 @@ +import { useCallback, useEffect, useRef, useState } from "react"; +import type { PageRef, PdfFile } from "./pdfTypes"; +import { generateThumbnailsWithRotationsProgressive } from "./pdfThumbnailService"; +import { normalizeRotation } from "../workspace/useWorkspaceState"; + +const DEFAULT_MAX_HEIGHT = 150; +const DEFAULT_MAX_WIDTH = 140; +const DEFAULT_CONCURRENCY = 3; + +interface UsePdfThumbnailsOptions { + pdf: PdfFile | null; + pages: PageRef[]; + maxHeight?: number; + maxWidth?: number; + concurrency?: number; + onError?: (message: string, error: unknown) => void; +} + +interface UsePdfThumbnailsResult { + thumbnails: Record; + resetThumbnails: () => void; + clearThumbnailCache: () => void; +} + +function thumbnailCacheKey( + pdfId: string, + sourcePageIndex: number, + rotation: number, + maxWidth: number, + maxHeight: number, +): string { + return [ + pdfId, + sourcePageIndex, + normalizeRotation(rotation), + maxWidth, + maxHeight, + ].join(":"); +} + +function pruneAndMergeThumbnails( + previous: Record, + pages: PageRef[], + updates: Record, +): Record { + const pageIds = new Set(pages.map((page) => page.id)); + const next: Record = {}; + + for (const [pageId, thumbnail] of Object.entries(previous)) { + if (pageIds.has(pageId)) { + next[pageId] = thumbnail; + } + } + + return { + ...next, + ...updates, + }; +} + +export function usePdfThumbnails({ + pdf, + pages, + maxHeight = DEFAULT_MAX_HEIGHT, + maxWidth = DEFAULT_MAX_WIDTH, + concurrency = DEFAULT_CONCURRENCY, + onError, +}: UsePdfThumbnailsOptions): UsePdfThumbnailsResult { + const [thumbnails, setThumbnails] = useState>({}); + const thumbnailCacheRef = useRef>(new Map()); + const latestPagesRef = useRef(pages); + const previousPdfIdRef = useRef(null); + + useEffect(() => { + latestPagesRef.current = pages; + }, [pages]); + + const resetThumbnails = useCallback(() => { + setThumbnails({}); + }, []); + + const clearThumbnailCache = useCallback(() => { + thumbnailCacheRef.current.clear(); + setThumbnails({}); + }, []); + + useEffect(() => { + const currentPdfId = pdf?.id ?? null; + + if (!pdf) { + previousPdfIdRef.current = null; + clearThumbnailCache(); + return; + } + + if (previousPdfIdRef.current !== currentPdfId) { + previousPdfIdRef.current = currentPdfId; + clearThumbnailCache(); + } + }, [clearThumbnailCache, pdf]); + + useEffect(() => { + if (!pdf) return; + + const controller = new AbortController(); + const cachedUpdates: Record = {}; + const renderGroups = new Map>(); + + for (const page of pages) { + const rotation = normalizeRotation(page.rotation); + const cacheKey = thumbnailCacheKey( + pdf.id, + page.sourcePageIndex, + rotation, + maxWidth, + maxHeight, + ); + const cached = thumbnailCacheRef.current.get(cacheKey); + + if (cached) { + cachedUpdates[page.id] = cached; + continue; + } + + const pageIndices = renderGroups.get(rotation) ?? new Set(); + pageIndices.add(page.sourcePageIndex); + renderGroups.set(rotation, pageIndices); + } + + setThumbnails((previous) => + pruneAndMergeThumbnails(previous, pages, cachedUpdates), + ); + + if (renderGroups.size === 0) return; + + const renderMissingThumbnails = async () => { + for (const [rotation, pageIndexSet] of renderGroups) { + if (controller.signal.aborted) return; + + const pageIndices = Array.from(pageIndexSet); + const rotationsBySourcePage: Record = {}; + + for (const pageIndex of pageIndices) { + rotationsBySourcePage[pageIndex] = rotation; + } + + await generateThumbnailsWithRotationsProgressive( + pdf.arrayBuffer, + rotationsBySourcePage, + { + maxHeight, + maxWidth, + concurrency: Math.min(concurrency, pageIndices.length), + pageIndices, + signal: controller.signal, + onThumbnail: ({ pageIndex, dataUrl }) => { + if (controller.signal.aborted) return; + + thumbnailCacheRef.current.set( + thumbnailCacheKey( + pdf.id, + pageIndex, + rotation, + maxWidth, + maxHeight, + ), + dataUrl, + ); + + const updates: Record = {}; + + for (const page of latestPagesRef.current) { + if ( + page.sourcePageIndex === pageIndex && + normalizeRotation(page.rotation) === rotation + ) { + updates[page.id] = dataUrl; + } + } + + if (Object.keys(updates).length === 0) return; + + setThumbnails((previous) => + pruneAndMergeThumbnails( + previous, + latestPagesRef.current, + updates, + ), + ); + }, + }, + ); + } + }; + + void renderMissingThumbnails().catch((error) => { + if (!controller.signal.aborted) { + onError?.("Failed to generate thumbnails (see console).", error); + } + }); + + return () => { + controller.abort(); + }; + }, [concurrency, maxHeight, maxWidth, onError, pages, pdf]); + + return { + thumbnails, + resetThumbnails, + clearThumbnailCache, + }; +} diff --git a/src/styles.css b/src/styles.css index 2aabf50..3fa6d6f 100644 --- a/src/styles.css +++ b/src/styles.css @@ -6,7 +6,11 @@ body { margin: 0; - font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', + font-family: + system-ui, + -apple-system, + BlinkMacSystemFont, + "Segoe UI", sans-serif; background-color: #f3f4f6; color: #111827; diff --git a/src/version.ts b/src/version.ts index 80b5224..e59bb90 100644 --- a/src/version.ts +++ b/src/version.ts @@ -1 +1 @@ -export const APP_VERSION = '0.2.0'; \ No newline at end of file +export const APP_VERSION = "0.2.1"; diff --git a/src/vite-env.d.ts b/src/vite-env.d.ts new file mode 100644 index 0000000..11f02fe --- /dev/null +++ b/src/vite-env.d.ts @@ -0,0 +1 @@ +/// diff --git a/src/workspace/useWorkspaceState.test.tsx b/src/workspace/useWorkspaceState.test.tsx new file mode 100644 index 0000000..4905213 --- /dev/null +++ b/src/workspace/useWorkspaceState.test.tsx @@ -0,0 +1,208 @@ +import React, { forwardRef, useImperativeHandle } from "react"; +import { act, render } from "@testing-library/react"; +import { describe, expect, it, vi } from "vitest"; +import type { PageRef } from "../pdf/pdfTypes"; +import type { + WorkspaceCommandRecord, + WorkspaceCommandState, +} from "./workspaceCommands"; +import { useWorkspaceState } from "./useWorkspaceState"; + +function page(id: string, sourcePageIndex: number, rotation = 0): PageRef { + return { id, sourcePageIndex, rotation }; +} + +function state( + pages: PageRef[], + selectedPageIds: string[] = [], + lastSelectedVisualIndex: number | null = null, +): WorkspaceCommandState { + return { pages, selectedPageIds, lastSelectedVisualIndex }; +} + +interface HarnessRef { + snapshot: () => { + pages: PageRef[]; + selectedPageIds: string[]; + lastSelectedVisualIndex: number | null; + workspaceDirty: boolean; + workspaceMessage: string | null; + workspaceHistory: WorkspaceCommandRecord[]; + redoHistory: WorkspaceCommandRecord[]; + }; + replaceWorkspaceState: ReturnType< + typeof useWorkspaceState + >["replaceWorkspaceState"]; + getCurrentCommandState: ReturnType< + typeof useWorkspaceState + >["getCurrentCommandState"]; + createWorkspaceCommand: ReturnType< + typeof useWorkspaceState + >["createWorkspaceCommand"]; + executeWorkspaceCommand: ReturnType< + typeof useWorkspaceState + >["executeWorkspaceCommand"]; + handleUndo: ReturnType["handleUndo"]; + handleRedo: ReturnType["handleRedo"]; +} + +const Harness = forwardRef void }>( + ({ onContentChanged }, ref) => { + const workspace = useWorkspaceState({ onContentChanged }); + + useImperativeHandle(ref, () => ({ + snapshot: () => ({ + pages: workspace.pages, + selectedPageIds: workspace.selectedPageIds, + lastSelectedVisualIndex: workspace.lastSelectedVisualIndex, + workspaceDirty: workspace.workspaceDirty, + workspaceMessage: workspace.workspaceMessage, + workspaceHistory: workspace.workspaceHistory, + redoHistory: workspace.redoHistory, + }), + replaceWorkspaceState: workspace.replaceWorkspaceState, + getCurrentCommandState: workspace.getCurrentCommandState, + createWorkspaceCommand: workspace.createWorkspaceCommand, + executeWorkspaceCommand: workspace.executeWorkspaceCommand, + handleUndo: workspace.handleUndo, + handleRedo: workspace.handleRedo, + })); + + return null; + }, +); + +function renderHarness(onContentChanged = vi.fn()) { + const ref = React.createRef(); + render(); + + if (!ref.current) { + throw new Error("Harness ref was not initialized"); + } + + return { ref, onContentChanged }; +} + +describe("useWorkspaceState", () => { + it("replaces workspace state from loaded data without marking it dirty", () => { + const { ref } = renderHarness(); + const loadedPages = [page("p1", 0), page("p2", 1, 90)]; + + act(() => { + ref.current?.replaceWorkspaceState({ + pages: loadedPages, + selectedPageIds: ["p2"], + lastSelectedVisualIndex: 1, + history: [], + redoHistory: [], + dirty: false, + message: "Workspace loaded.", + }); + }); + + expect(ref.current?.snapshot()).toMatchObject({ + pages: loadedPages, + selectedPageIds: ["p2"], + lastSelectedVisualIndex: 1, + workspaceDirty: false, + workspaceMessage: "Workspace loaded.", + workspaceHistory: [], + redoHistory: [], + }); + }); + + it("executes commands, stores history, clears redo, and marks content changed", () => { + const { ref, onContentChanged } = renderHarness(); + const before = state([page("p1", 0), page("p2", 1)], ["p1"], 0); + const after = state([page("p2", 1), page("p1", 0)], ["p2"], 0); + + act(() => { + ref.current?.replaceWorkspaceState({ + ...before, + redoHistory: [ + { + id: "redo-record", + type: "old-redo", + label: "Old redo", + timestamp: "2026-05-17T10:00:00.000Z", + payload: { before, after }, + }, + ], + }); + }); + + act(() => { + const command = ref.current?.createWorkspaceCommand({ + type: "reorder-pages", + label: "Move page 2 before page 1", + before, + after, + }); + + if (!command) throw new Error("Command was not created"); + ref.current?.executeWorkspaceCommand(command); + }); + + const snapshot = ref.current?.snapshot(); + expect(snapshot?.pages).toEqual(after.pages); + expect(snapshot?.selectedPageIds).toEqual(["p2"]); + expect(snapshot?.workspaceDirty).toBe(true); + expect(snapshot?.workspaceMessage).toBeNull(); + expect(snapshot?.workspaceHistory).toHaveLength(1); + expect(snapshot?.workspaceHistory[0]).toMatchObject({ + type: "reorder-pages", + label: "Move page 2 before page 1", + }); + expect(snapshot?.redoHistory).toHaveLength(0); + expect(onContentChanged).toHaveBeenCalledTimes(1); + }); + + it("undoes and redoes command records in stack order", () => { + const { ref, onContentChanged } = renderHarness(); + const initial = state([page("p1", 0), page("p2", 1)], ["p1"], 0); + const reordered = state([page("p2", 1), page("p1", 0)], ["p2"], 0); + + act(() => { + ref.current?.replaceWorkspaceState(initial); + }); + + act(() => { + const command = ref.current?.createWorkspaceCommand({ + type: "reorder-pages", + label: "Move page", + before: initial, + after: reordered, + }); + + if (!command) throw new Error("Command was not created"); + ref.current?.executeWorkspaceCommand(command); + }); + + act(() => { + ref.current?.handleUndo(); + }); + + expect(ref.current?.snapshot()).toMatchObject({ + pages: initial.pages, + selectedPageIds: initial.selectedPageIds, + lastSelectedVisualIndex: initial.lastSelectedVisualIndex, + workspaceDirty: true, + }); + expect(ref.current?.snapshot().workspaceHistory).toHaveLength(0); + expect(ref.current?.snapshot().redoHistory).toHaveLength(1); + + act(() => { + ref.current?.handleRedo(); + }); + + expect(ref.current?.snapshot()).toMatchObject({ + pages: reordered.pages, + selectedPageIds: reordered.selectedPageIds, + lastSelectedVisualIndex: reordered.lastSelectedVisualIndex, + workspaceDirty: true, + }); + expect(ref.current?.snapshot().workspaceHistory).toHaveLength(1); + expect(ref.current?.snapshot().redoHistory).toHaveLength(0); + expect(onContentChanged).toHaveBeenCalledTimes(3); + }); +}); diff --git a/src/workspace/useWorkspaceState.ts b/src/workspace/useWorkspaceState.ts new file mode 100644 index 0000000..fd70759 --- /dev/null +++ b/src/workspace/useWorkspaceState.ts @@ -0,0 +1,246 @@ +import { useCallback, useRef, useState } from "react"; +import type { PageRef } from "../pdf/pdfTypes"; +import type { + WorkspaceCommand, + WorkspaceCommandRecord, + WorkspaceCommandState, +} from "./workspaceCommands"; +import { + createSnapshotCommand, + reviveWorkspaceCommand, + toWorkspaceCommandRecord, +} from "./workspaceCommands"; + +function createId(prefix: string): string { + if (typeof crypto !== "undefined" && crypto.randomUUID) { + return crypto.randomUUID(); + } + + return `${prefix}_${Date.now()}_${Math.random().toString(36).slice(2)}`; +} + +export function createWorkspaceId(): string { + return createId("workspace"); +} + +export function defaultWorkspaceNameFromPdfName(pdfName: string): string { + return pdfName.replace(/\.pdf$/i, "") || "Untitled workspace"; +} + +export function createPageRefId(): string { + return createId("page"); +} + +export function createInitialPageRefs(pageCount: number): PageRef[] { + return Array.from({ length: pageCount }, (_, sourcePageIndex) => ({ + id: createPageRefId(), + sourcePageIndex, + rotation: 0, + })); +} + +export function normalizeRotation(rotation: number | undefined): number { + return (((rotation ?? 0) % 360) + 360) % 360; +} + +type SetStateAction = T | ((previous: T) => T); + +interface UseWorkspaceStateOptions { + onContentChanged?: () => void; +} + +interface ReplaceWorkspaceStateOptions { + pages?: PageRef[]; + selectedPageIds?: string[]; + lastSelectedVisualIndex?: number | null; + history?: WorkspaceCommandRecord[]; + redoHistory?: WorkspaceCommandRecord[]; + dirty?: boolean; + message?: string | null; +} + +export function useWorkspaceState({ + onContentChanged, +}: UseWorkspaceStateOptions = {}) { + const [pages, setPagesState] = useState([]); + const [selectedPageIds, setSelectedPageIdsState] = useState([]); + const [lastSelectedVisualIndex, setLastSelectedVisualIndexState] = useState< + number | null + >(null); + const [workspaceDirty, setWorkspaceDirty] = useState(false); + const [workspaceMessage, setWorkspaceMessage] = useState(null); + const [workspaceHistory, setWorkspaceHistory] = useState< + WorkspaceCommandRecord[] + >([]); + const [redoHistory, setRedoHistory] = useState([]); + + const latestPagesRef = useRef([]); + const selectedPageIdsRef = useRef([]); + const lastSelectedVisualIndexRef = useRef(null); + + const setPages = useCallback((action: SetStateAction) => { + setPagesState((previous) => { + const next = typeof action === "function" ? action(previous) : action; + latestPagesRef.current = next; + return next; + }); + }, []); + + const setSelectedPageIds = useCallback((action: SetStateAction) => { + setSelectedPageIdsState((previous) => { + const next = typeof action === "function" ? action(previous) : action; + selectedPageIdsRef.current = next; + return next; + }); + }, []); + + const setLastSelectedVisualIndex = useCallback( + (action: SetStateAction) => { + setLastSelectedVisualIndexState((previous) => { + const next = typeof action === "function" ? action(previous) : action; + lastSelectedVisualIndexRef.current = next; + return next; + }); + }, + [], + ); + + const getCurrentCommandState = useCallback( + (): WorkspaceCommandState => ({ + pages: latestPagesRef.current, + selectedPageIds: selectedPageIdsRef.current, + lastSelectedVisualIndex: lastSelectedVisualIndexRef.current, + }), + [], + ); + + const applyCommandState = useCallback( + (state: WorkspaceCommandState) => { + setPages(state.pages); + setSelectedPageIds(state.selectedPageIds); + setLastSelectedVisualIndex(state.lastSelectedVisualIndex); + }, + [setLastSelectedVisualIndex, setPages, setSelectedPageIds], + ); + + const markWorkspaceChanged = useCallback(() => { + setWorkspaceDirty(true); + setWorkspaceMessage(null); + onContentChanged?.(); + }, [onContentChanged]); + + const createWorkspaceCommand = useCallback( + (params: { + type: string; + label: string; + before: WorkspaceCommandState; + after: WorkspaceCommandState; + details?: Record; + }): WorkspaceCommand => + createSnapshotCommand({ + id: createId("command"), + type: params.type, + label: params.label, + before: params.before, + after: params.after, + details: params.details, + }), + [], + ); + + const executeWorkspaceCommand = useCallback( + (command: WorkspaceCommand) => { + const nextState = command.do(getCurrentCommandState()); + + applyCommandState(nextState); + setWorkspaceHistory((previous) => [ + ...previous, + toWorkspaceCommandRecord(command), + ]); + setRedoHistory([]); + markWorkspaceChanged(); + }, + [applyCommandState, getCurrentCommandState, markWorkspaceChanged], + ); + + const handleUndo = useCallback(() => { + const record = workspaceHistory[workspaceHistory.length - 1]; + if (!record) return; + + const command = reviveWorkspaceCommand(record); + const previousState = command.undo(getCurrentCommandState()); + + applyCommandState(previousState); + setWorkspaceHistory((previous) => previous.slice(0, -1)); + setRedoHistory((previous) => [...previous, record]); + markWorkspaceChanged(); + }, [ + applyCommandState, + getCurrentCommandState, + markWorkspaceChanged, + workspaceHistory, + ]); + + const handleRedo = useCallback(() => { + const record = redoHistory[redoHistory.length - 1]; + if (!record) return; + + const command = reviveWorkspaceCommand(record); + const nextState = command.do(getCurrentCommandState()); + + applyCommandState(nextState); + setRedoHistory((previous) => previous.slice(0, -1)); + setWorkspaceHistory((previous) => [...previous, record]); + markWorkspaceChanged(); + }, [ + applyCommandState, + getCurrentCommandState, + markWorkspaceChanged, + redoHistory, + ]); + + const replaceWorkspaceState = useCallback( + (state: ReplaceWorkspaceStateOptions = {}) => { + setPages(state.pages ?? []); + setSelectedPageIds(state.selectedPageIds ?? []); + setLastSelectedVisualIndex(state.lastSelectedVisualIndex ?? null); + setWorkspaceHistory(state.history ?? []); + setRedoHistory(state.redoHistory ?? []); + setWorkspaceDirty(state.dirty ?? false); + setWorkspaceMessage(state.message ?? null); + }, + [setLastSelectedVisualIndex, setPages, setSelectedPageIds], + ); + + const resetWorkspaceState = useCallback(() => { + replaceWorkspaceState(); + }, [replaceWorkspaceState]); + + return { + pages, + setPages, + selectedPageIds, + setSelectedPageIds, + lastSelectedVisualIndex, + setLastSelectedVisualIndex, + latestPagesRef, + + workspaceDirty, + setWorkspaceDirty, + workspaceMessage, + setWorkspaceMessage, + workspaceHistory, + setWorkspaceHistory, + redoHistory, + setRedoHistory, + + getCurrentCommandState, + applyCommandState, + createWorkspaceCommand, + executeWorkspaceCommand, + handleUndo, + handleRedo, + replaceWorkspaceState, + resetWorkspaceState, + }; +} diff --git a/src/workspace/workspaceCommands.test.ts b/src/workspace/workspaceCommands.test.ts new file mode 100644 index 0000000..f932ba4 --- /dev/null +++ b/src/workspace/workspaceCommands.test.ts @@ -0,0 +1,105 @@ +import { describe, expect, it } from "vitest"; +import type { WorkspaceCommandState } from "./workspaceCommands"; +import { + cloneCommandState, + createSnapshotCommand, + reviveWorkspaceCommand, + toWorkspaceCommandRecord, +} from "./workspaceCommands"; + +function makeState(pageIds: string[]): WorkspaceCommandState { + return { + pages: pageIds.map((id, index) => ({ + id, + sourcePageIndex: index, + rotation: index * 90, + })), + selectedPageIds: pageIds.slice(0, 1), + lastSelectedVisualIndex: pageIds.length > 0 ? 0 : null, + }; +} + +describe("workspaceCommands", () => { + it("clones command state deeply enough for page and selection changes", () => { + const original = makeState(["a", "b"]); + const cloned = cloneCommandState(original); + + original.pages[0].rotation = 270; + original.selectedPageIds.push("b"); + original.lastSelectedVisualIndex = 1; + + expect(cloned).toEqual({ + pages: [ + { id: "a", sourcePageIndex: 0, rotation: 0 }, + { id: "b", sourcePageIndex: 1, rotation: 90 }, + ], + selectedPageIds: ["a"], + lastSelectedVisualIndex: 0, + }); + }); + + it("creates snapshot commands that are stable after source states mutate", () => { + const before = makeState(["a", "b"]); + const after = makeState(["b", "a"]); + after.selectedPageIds = ["b"]; + after.lastSelectedVisualIndex = 0; + + const command = createSnapshotCommand({ + id: "cmd-1", + type: "reorder-pages", + label: "Move page", + timestamp: "2026-05-17T10:00:00.000Z", + before, + after, + details: { moved: 1 }, + }); + + before.pages.length = 0; + after.pages[0].rotation = 180; + after.selectedPageIds.push("a"); + + expect(command.undo(makeState(["ignored"]))).toEqual({ + pages: [ + { id: "a", sourcePageIndex: 0, rotation: 0 }, + { id: "b", sourcePageIndex: 1, rotation: 90 }, + ], + selectedPageIds: ["a"], + lastSelectedVisualIndex: 0, + }); + + expect(command.do(makeState(["ignored"]))).toEqual({ + pages: [ + { id: "b", sourcePageIndex: 0, rotation: 0 }, + { id: "a", sourcePageIndex: 1, rotation: 90 }, + ], + selectedPageIds: ["b"], + lastSelectedVisualIndex: 0, + }); + }); + + it("round-trips commands through serializable records", () => { + const before = makeState(["a", "b", "c"]); + const after: WorkspaceCommandState = { + pages: [before.pages[2], before.pages[0], before.pages[1]], + selectedPageIds: ["c"], + lastSelectedVisualIndex: 0, + }; + + const command = createSnapshotCommand({ + id: "cmd-2", + type: "copy-pages", + label: "Copy pages", + timestamp: "2026-05-17T10:05:00.000Z", + before, + after, + }); + + const record = toWorkspaceCommandRecord(command); + const revived = reviveWorkspaceCommand(record); + + expect(record).not.toHaveProperty("do"); + expect(record).not.toHaveProperty("undo"); + expect(revived.do(before)).toEqual(after); + expect(revived.undo(after)).toEqual(before); + }); +}); diff --git a/src/workspace/workspaceCommands.ts b/src/workspace/workspaceCommands.ts index be024a5..7a66823 100644 --- a/src/workspace/workspaceCommands.ts +++ b/src/workspace/workspaceCommands.ts @@ -1,4 +1,4 @@ -import type { PageRef } from '../pdf/pdfTypes'; +import type { PageRef } from "../pdf/pdfTypes"; export interface WorkspaceCommandState { pages: PageRef[]; @@ -26,7 +26,7 @@ export interface WorkspaceCommand extends WorkspaceCommandRecord { } export function cloneCommandState( - state: WorkspaceCommandState + state: WorkspaceCommandState, ): WorkspaceCommandState { return { pages: state.pages.map((page) => ({ ...page })), @@ -58,7 +58,7 @@ export function createSnapshotCommand(params: { } export function reviveWorkspaceCommand( - record: WorkspaceCommandRecord + record: WorkspaceCommandRecord, ): WorkspaceCommand { return { ...record, @@ -68,7 +68,7 @@ export function reviveWorkspaceCommand( } export function toWorkspaceCommandRecord( - command: WorkspaceCommand + command: WorkspaceCommand, ): WorkspaceCommandRecord { return { id: command.id, @@ -81,4 +81,4 @@ export function toWorkspaceCommandRecord( details: command.payload.details, }, }; -} \ No newline at end of file +} diff --git a/src/workspace/workspaceDb.ts b/src/workspace/workspaceDb.ts index 34e122a..dc920c0 100644 --- a/src/workspace/workspaceDb.ts +++ b/src/workspace/workspaceDb.ts @@ -2,13 +2,13 @@ import type { LoadedWorkspace, StoredWorkspace, WorkspaceSummary, -} from './workspaceTypes'; +} from "./workspaceTypes"; -const DB_NAME = 'pdf-tools-workspaces'; +const DB_NAME = "pdf-tools-workspaces"; const DB_VERSION = 1; -const WORKSPACE_STORE = 'workspaces'; -const PDF_STORE = 'pdfBinaries'; +const WORKSPACE_STORE = "workspaces"; +const PDF_STORE = "pdfBinaries"; interface PdfBinaryRecord { pdfId: string; @@ -48,21 +48,21 @@ function openWorkspaceDb(): Promise { if (!db.objectStoreNames.contains(WORKSPACE_STORE)) { const workspaceStore = db.createObjectStore(WORKSPACE_STORE, { - keyPath: 'id', + keyPath: "id", }); - workspaceStore.createIndex('updatedAt', 'updatedAt', { + workspaceStore.createIndex("updatedAt", "updatedAt", { unique: false, }); - workspaceStore.createIndex('pdfId', 'pdfId', { + workspaceStore.createIndex("pdfId", "pdfId", { unique: false, }); } if (!db.objectStoreNames.contains(PDF_STORE)) { db.createObjectStore(PDF_STORE, { - keyPath: 'pdfId', + keyPath: "pdfId", }); } }; @@ -76,7 +76,7 @@ export async function listWorkspaces(): Promise { const db = await openWorkspaceDb(); try { - const tx = db.transaction(WORKSPACE_STORE, 'readonly'); + const tx = db.transaction(WORKSPACE_STORE, "readonly"); const store = tx.objectStore(WORKSPACE_STORE); const records = await requestToPromise(store.getAll()); @@ -113,13 +113,13 @@ export async function saveWorkspaceToIndexedDb({ const pdfRecord: PdfBinaryRecord = { pdfId: workspace.pdfId, name: workspace.pdfName, - blob: new Blob([pdfArrayBuffer], { type: 'application/pdf' }), + blob: new Blob([pdfArrayBuffer], { type: "application/pdf" }), size: pdfArrayBuffer.byteLength, createdAt: workspace.createdAt, updatedAt: now, }; - const tx = db.transaction([WORKSPACE_STORE, PDF_STORE], 'readwrite'); + const tx = db.transaction([WORKSPACE_STORE, PDF_STORE], "readwrite"); tx.objectStore(PDF_STORE).put(pdfRecord); tx.objectStore(WORKSPACE_STORE).put(workspace); @@ -131,15 +131,15 @@ export async function saveWorkspaceToIndexedDb({ } export async function loadWorkspaceFromIndexedDb( - workspaceId: string + workspaceId: string, ): Promise { const db = await openWorkspaceDb(); try { - const tx = db.transaction([WORKSPACE_STORE, PDF_STORE], 'readonly'); + const tx = db.transaction([WORKSPACE_STORE, PDF_STORE], "readonly"); const workspace = await requestToPromise( - tx.objectStore(WORKSPACE_STORE).get(workspaceId) + tx.objectStore(WORKSPACE_STORE).get(workspaceId), ); if (!workspace) { @@ -148,7 +148,7 @@ export async function loadWorkspaceFromIndexedDb( } const pdfRecord = await requestToPromise( - tx.objectStore(PDF_STORE).get(workspace.pdfId) + tx.objectStore(PDF_STORE).get(workspace.pdfId), ); await transactionDone(tx); @@ -169,20 +169,20 @@ export async function loadWorkspaceFromIndexedDb( } export async function deleteWorkspaceFromIndexedDb( - workspaceId: string + workspaceId: string, ): Promise { const db = await openWorkspaceDb(); try { - const lookupTx = db.transaction(WORKSPACE_STORE, 'readonly'); + const lookupTx = db.transaction(WORKSPACE_STORE, "readonly"); const workspace = await requestToPromise( - lookupTx.objectStore(WORKSPACE_STORE).get(workspaceId) + lookupTx.objectStore(WORKSPACE_STORE).get(workspaceId), ); await transactionDone(lookupTx); if (!workspace) return; - const deleteTx = db.transaction([WORKSPACE_STORE, PDF_STORE], 'readwrite'); + const deleteTx = db.transaction([WORKSPACE_STORE, PDF_STORE], "readwrite"); deleteTx.objectStore(WORKSPACE_STORE).delete(workspaceId); await transactionDone(deleteTx); @@ -190,14 +190,14 @@ export async function deleteWorkspaceFromIndexedDb( const remainingWorkspaces = await listWorkspaces(); const pdfStillUsed = remainingWorkspaces.some( - (summary) => summary.pdfId === workspace.pdfId + (summary) => summary.pdfId === workspace.pdfId, ); if (!pdfStillUsed) { const cleanupDb = await openWorkspaceDb(); try { - const cleanupTx = cleanupDb.transaction(PDF_STORE, 'readwrite'); + const cleanupTx = cleanupDb.transaction(PDF_STORE, "readwrite"); cleanupTx.objectStore(PDF_STORE).delete(workspace.pdfId); await transactionDone(cleanupTx); } finally { @@ -207,4 +207,4 @@ export async function deleteWorkspaceFromIndexedDb( } finally { db.close(); } -} \ No newline at end of file +} diff --git a/src/workspace/workspaceTypes.ts b/src/workspace/workspaceTypes.ts index 2c08077..924802a 100644 --- a/src/workspace/workspaceTypes.ts +++ b/src/workspace/workspaceTypes.ts @@ -1,5 +1,5 @@ -import type { PageRef } from '../pdf/pdfTypes'; -import type { WorkspaceCommandRecord } from './workspaceCommands'; +import type { PageRef } from "../pdf/pdfTypes"; +import type { WorkspaceCommandRecord } from "./workspaceCommands"; export interface StoredWorkspace { schemaVersion: 1; @@ -37,4 +37,4 @@ export interface WorkspaceSummary { export interface LoadedWorkspace { workspace: StoredWorkspace; pdfArrayBuffer: ArrayBuffer; -} \ No newline at end of file +} diff --git a/vite.config.ts b/vite.config.ts index 3cb33f8..6a2e0b8 100644 --- a/vite.config.ts +++ b/vite.config.ts @@ -1,11 +1,10 @@ -import { defineConfig } from 'vite'; -import react from '@vitejs/plugin-react-swc'; +import { defineConfig } from "vite"; +import react from "@vitejs/plugin-react"; export default defineConfig({ plugins: [react()], server: { host: true, - allowedHosts: ['pdftools.add-ideas.de'], // ← ADD THIS + allowedHosts: ["pdftools.add-ideas.de"], // ← ADD THIS }, }); -