apoorvrajdev commited on
Commit
a8f12e6
·
1 Parent(s): 4b41a19

feat(frontend): complete Phase 2B frontend-backend integration

Browse files
configs/base.yaml CHANGED
@@ -41,3 +41,7 @@ serve:
41
  beam_width: 3
42
  cors_allowed_origins:
43
  - http://localhost:3000
 
 
 
 
 
41
  beam_width: 3
42
  cors_allowed_origins:
43
  - http://localhost:3000
44
+ - http://localhost:5173
45
+ - http://localhost:5174
46
+ - http://127.0.0.1:5173
47
+ - http://127.0.0.1:5174
frontend/.env.example ADDED
@@ -0,0 +1,5 @@
 
 
 
 
 
 
1
+ # Base URL for the FastAPI inference backend.
2
+ # Leave unset to use the default (http://127.0.0.1:8000).
3
+ # In production, set this to the deployed API origin, e.g.:
4
+ # VITE_API_BASE=https://api.example.com
5
+ VITE_API_BASE=http://127.0.0.1:8000
frontend/.gitignore ADDED
@@ -0,0 +1,24 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # Logs
2
+ logs
3
+ *.log
4
+ npm-debug.log*
5
+ yarn-debug.log*
6
+ yarn-error.log*
7
+ pnpm-debug.log*
8
+ lerna-debug.log*
9
+
10
+ node_modules
11
+ dist
12
+ dist-ssr
13
+ *.local
14
+
15
+ # Editor directories and files
16
+ .vscode/*
17
+ !.vscode/extensions.json
18
+ .idea
19
+ .DS_Store
20
+ *.suo
21
+ *.ntvs*
22
+ *.njsproj
23
+ *.sln
24
+ *.sw?
frontend/README.md ADDED
@@ -0,0 +1,16 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # React + Vite
2
+
3
+ This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
4
+
5
+ Currently, two official plugins are available:
6
+
7
+ - [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Oxc](https://oxc.rs)
8
+ - [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/)
9
+
10
+ ## React Compiler
11
+
12
+ The React Compiler is not enabled on this template because of its impact on dev & build performances. To add it, see [this documentation](https://react.dev/learn/react-compiler/installation).
13
+
14
+ ## Expanding the ESLint configuration
15
+
16
+ If you are developing a production application, we recommend using TypeScript with type-aware lint rules enabled. Check out the [TS template](https://github.com/vitejs/vite/tree/main/packages/create-vite/template-react-ts) for information on how to integrate TypeScript and [`typescript-eslint`](https://typescript-eslint.io) in your project.
frontend/eslint.config.js ADDED
@@ -0,0 +1,21 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import js from "@eslint/js";
2
+ import globals from "globals";
3
+ import reactHooks from "eslint-plugin-react-hooks";
4
+ import reactRefresh from "eslint-plugin-react-refresh";
5
+ import { defineConfig, globalIgnores } from "eslint/config";
6
+
7
+ export default defineConfig([
8
+ globalIgnores(["dist"]),
9
+ {
10
+ files: ["**/*.{js,jsx}"],
11
+ extends: [
12
+ js.configs.recommended,
13
+ reactHooks.configs.flat.recommended,
14
+ reactRefresh.configs.vite,
15
+ ],
16
+ languageOptions: {
17
+ globals: globals.browser,
18
+ parserOptions: { ecmaFeatures: { jsx: true } },
19
+ },
20
+ },
21
+ ]);
frontend/index.html ADDED
@@ -0,0 +1,13 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
+ <title>frontend</title>
8
+ </head>
9
+ <body>
10
+ <div id="root"></div>
11
+ <script type="module" src="/src/main.jsx"></script>
12
+ </body>
13
+ </html>
frontend/package-lock.json ADDED
@@ -0,0 +1,2751 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "frontend",
3
+ "version": "0.0.0",
4
+ "lockfileVersion": 3,
5
+ "requires": true,
6
+ "packages": {
7
+ "": {
8
+ "name": "frontend",
9
+ "version": "0.0.0",
10
+ "dependencies": {
11
+ "react": "^19.2.6",
12
+ "react-dom": "^19.2.6"
13
+ },
14
+ "devDependencies": {
15
+ "@eslint/js": "^10.0.1",
16
+ "@tailwindcss/vite": "^4.3.0",
17
+ "@types/react": "^19.2.14",
18
+ "@types/react-dom": "^19.2.3",
19
+ "@vitejs/plugin-react": "^6.0.1",
20
+ "eslint": "^10.3.0",
21
+ "eslint-plugin-react-hooks": "^7.1.1",
22
+ "eslint-plugin-react-refresh": "^0.5.2",
23
+ "globals": "^17.6.0",
24
+ "tailwindcss": "^4.3.0",
25
+ "vite": "^8.0.12"
26
+ }
27
+ },
28
+ "node_modules/@babel/code-frame": {
29
+ "version": "7.29.0",
30
+ "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz",
31
+ "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==",
32
+ "dev": true,
33
+ "license": "MIT",
34
+ "dependencies": {
35
+ "@babel/helper-validator-identifier": "^7.28.5",
36
+ "js-tokens": "^4.0.0",
37
+ "picocolors": "^1.1.1"
38
+ },
39
+ "engines": {
40
+ "node": ">=6.9.0"
41
+ }
42
+ },
43
+ "node_modules/@babel/compat-data": {
44
+ "version": "7.29.3",
45
+ "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.3.tgz",
46
+ "integrity": "sha512-LIVqM46zQWZhj17qA8wb4nW/ixr2y1Nw+r1etiAWgRM6U1IqP+LNhL1yg440jYZR72jCWcWbLWzIosH+uP1fqg==",
47
+ "dev": true,
48
+ "license": "MIT",
49
+ "engines": {
50
+ "node": ">=6.9.0"
51
+ }
52
+ },
53
+ "node_modules/@babel/core": {
54
+ "version": "7.29.0",
55
+ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz",
56
+ "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==",
57
+ "dev": true,
58
+ "license": "MIT",
59
+ "peer": true,
60
+ "dependencies": {
61
+ "@babel/code-frame": "^7.29.0",
62
+ "@babel/generator": "^7.29.0",
63
+ "@babel/helper-compilation-targets": "^7.28.6",
64
+ "@babel/helper-module-transforms": "^7.28.6",
65
+ "@babel/helpers": "^7.28.6",
66
+ "@babel/parser": "^7.29.0",
67
+ "@babel/template": "^7.28.6",
68
+ "@babel/traverse": "^7.29.0",
69
+ "@babel/types": "^7.29.0",
70
+ "@jridgewell/remapping": "^2.3.5",
71
+ "convert-source-map": "^2.0.0",
72
+ "debug": "^4.1.0",
73
+ "gensync": "^1.0.0-beta.2",
74
+ "json5": "^2.2.3",
75
+ "semver": "^6.3.1"
76
+ },
77
+ "engines": {
78
+ "node": ">=6.9.0"
79
+ },
80
+ "funding": {
81
+ "type": "opencollective",
82
+ "url": "https://opencollective.com/babel"
83
+ }
84
+ },
85
+ "node_modules/@babel/generator": {
86
+ "version": "7.29.1",
87
+ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz",
88
+ "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==",
89
+ "dev": true,
90
+ "license": "MIT",
91
+ "dependencies": {
92
+ "@babel/parser": "^7.29.0",
93
+ "@babel/types": "^7.29.0",
94
+ "@jridgewell/gen-mapping": "^0.3.12",
95
+ "@jridgewell/trace-mapping": "^0.3.28",
96
+ "jsesc": "^3.0.2"
97
+ },
98
+ "engines": {
99
+ "node": ">=6.9.0"
100
+ }
101
+ },
102
+ "node_modules/@babel/helper-compilation-targets": {
103
+ "version": "7.28.6",
104
+ "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz",
105
+ "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==",
106
+ "dev": true,
107
+ "license": "MIT",
108
+ "dependencies": {
109
+ "@babel/compat-data": "^7.28.6",
110
+ "@babel/helper-validator-option": "^7.27.1",
111
+ "browserslist": "^4.24.0",
112
+ "lru-cache": "^5.1.1",
113
+ "semver": "^6.3.1"
114
+ },
115
+ "engines": {
116
+ "node": ">=6.9.0"
117
+ }
118
+ },
119
+ "node_modules/@babel/helper-globals": {
120
+ "version": "7.28.0",
121
+ "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz",
122
+ "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==",
123
+ "dev": true,
124
+ "license": "MIT",
125
+ "engines": {
126
+ "node": ">=6.9.0"
127
+ }
128
+ },
129
+ "node_modules/@babel/helper-module-imports": {
130
+ "version": "7.28.6",
131
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz",
132
+ "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==",
133
+ "dev": true,
134
+ "license": "MIT",
135
+ "dependencies": {
136
+ "@babel/traverse": "^7.28.6",
137
+ "@babel/types": "^7.28.6"
138
+ },
139
+ "engines": {
140
+ "node": ">=6.9.0"
141
+ }
142
+ },
143
+ "node_modules/@babel/helper-module-transforms": {
144
+ "version": "7.28.6",
145
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz",
146
+ "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==",
147
+ "dev": true,
148
+ "license": "MIT",
149
+ "dependencies": {
150
+ "@babel/helper-module-imports": "^7.28.6",
151
+ "@babel/helper-validator-identifier": "^7.28.5",
152
+ "@babel/traverse": "^7.28.6"
153
+ },
154
+ "engines": {
155
+ "node": ">=6.9.0"
156
+ },
157
+ "peerDependencies": {
158
+ "@babel/core": "^7.0.0"
159
+ }
160
+ },
161
+ "node_modules/@babel/helper-string-parser": {
162
+ "version": "7.27.1",
163
+ "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz",
164
+ "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==",
165
+ "dev": true,
166
+ "license": "MIT",
167
+ "engines": {
168
+ "node": ">=6.9.0"
169
+ }
170
+ },
171
+ "node_modules/@babel/helper-validator-identifier": {
172
+ "version": "7.28.5",
173
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
174
+ "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
175
+ "dev": true,
176
+ "license": "MIT",
177
+ "engines": {
178
+ "node": ">=6.9.0"
179
+ }
180
+ },
181
+ "node_modules/@babel/helper-validator-option": {
182
+ "version": "7.27.1",
183
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz",
184
+ "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==",
185
+ "dev": true,
186
+ "license": "MIT",
187
+ "engines": {
188
+ "node": ">=6.9.0"
189
+ }
190
+ },
191
+ "node_modules/@babel/helpers": {
192
+ "version": "7.29.2",
193
+ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.2.tgz",
194
+ "integrity": "sha512-HoGuUs4sCZNezVEKdVcwqmZN8GoHirLUcLaYVNBK2J0DadGtdcqgr3BCbvH8+XUo4NGjNl3VOtSjEKNzqfFgKw==",
195
+ "dev": true,
196
+ "license": "MIT",
197
+ "dependencies": {
198
+ "@babel/template": "^7.28.6",
199
+ "@babel/types": "^7.29.0"
200
+ },
201
+ "engines": {
202
+ "node": ">=6.9.0"
203
+ }
204
+ },
205
+ "node_modules/@babel/parser": {
206
+ "version": "7.29.3",
207
+ "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.3.tgz",
208
+ "integrity": "sha512-b3ctpQwp+PROvU/cttc4OYl4MzfJUWy6FZg+PMXfzmt/+39iHVF0sDfqay8TQM3JA2EUOyKcFZt75jWriQijsA==",
209
+ "dev": true,
210
+ "license": "MIT",
211
+ "dependencies": {
212
+ "@babel/types": "^7.29.0"
213
+ },
214
+ "bin": {
215
+ "parser": "bin/babel-parser.js"
216
+ },
217
+ "engines": {
218
+ "node": ">=6.0.0"
219
+ }
220
+ },
221
+ "node_modules/@babel/template": {
222
+ "version": "7.28.6",
223
+ "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz",
224
+ "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==",
225
+ "dev": true,
226
+ "license": "MIT",
227
+ "dependencies": {
228
+ "@babel/code-frame": "^7.28.6",
229
+ "@babel/parser": "^7.28.6",
230
+ "@babel/types": "^7.28.6"
231
+ },
232
+ "engines": {
233
+ "node": ">=6.9.0"
234
+ }
235
+ },
236
+ "node_modules/@babel/traverse": {
237
+ "version": "7.29.0",
238
+ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz",
239
+ "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==",
240
+ "dev": true,
241
+ "license": "MIT",
242
+ "dependencies": {
243
+ "@babel/code-frame": "^7.29.0",
244
+ "@babel/generator": "^7.29.0",
245
+ "@babel/helper-globals": "^7.28.0",
246
+ "@babel/parser": "^7.29.0",
247
+ "@babel/template": "^7.28.6",
248
+ "@babel/types": "^7.29.0",
249
+ "debug": "^4.3.1"
250
+ },
251
+ "engines": {
252
+ "node": ">=6.9.0"
253
+ }
254
+ },
255
+ "node_modules/@babel/types": {
256
+ "version": "7.29.0",
257
+ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz",
258
+ "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==",
259
+ "dev": true,
260
+ "license": "MIT",
261
+ "dependencies": {
262
+ "@babel/helper-string-parser": "^7.27.1",
263
+ "@babel/helper-validator-identifier": "^7.28.5"
264
+ },
265
+ "engines": {
266
+ "node": ">=6.9.0"
267
+ }
268
+ },
269
+ "node_modules/@emnapi/wasi-threads": {
270
+ "version": "1.2.1",
271
+ "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz",
272
+ "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==",
273
+ "dev": true,
274
+ "license": "MIT",
275
+ "optional": true,
276
+ "dependencies": {
277
+ "tslib": "^2.4.0"
278
+ }
279
+ },
280
+ "node_modules/@eslint-community/eslint-utils": {
281
+ "version": "4.9.1",
282
+ "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz",
283
+ "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==",
284
+ "dev": true,
285
+ "license": "MIT",
286
+ "dependencies": {
287
+ "eslint-visitor-keys": "^3.4.3"
288
+ },
289
+ "engines": {
290
+ "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
291
+ },
292
+ "funding": {
293
+ "url": "https://opencollective.com/eslint"
294
+ },
295
+ "peerDependencies": {
296
+ "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0"
297
+ }
298
+ },
299
+ "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": {
300
+ "version": "3.4.3",
301
+ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz",
302
+ "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==",
303
+ "dev": true,
304
+ "license": "Apache-2.0",
305
+ "engines": {
306
+ "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
307
+ },
308
+ "funding": {
309
+ "url": "https://opencollective.com/eslint"
310
+ }
311
+ },
312
+ "node_modules/@eslint-community/regexpp": {
313
+ "version": "4.12.2",
314
+ "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz",
315
+ "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==",
316
+ "dev": true,
317
+ "license": "MIT",
318
+ "engines": {
319
+ "node": "^12.0.0 || ^14.0.0 || >=16.0.0"
320
+ }
321
+ },
322
+ "node_modules/@eslint/config-array": {
323
+ "version": "0.23.5",
324
+ "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.23.5.tgz",
325
+ "integrity": "sha512-Y3kKLvC1dvTOT+oGlqNQ1XLqK6D1HU2YXPc52NmAlJZbMMWDzGYXMiPRJ8TYD39muD/OTjlZmNJ4ib7dvSrMBA==",
326
+ "dev": true,
327
+ "license": "Apache-2.0",
328
+ "dependencies": {
329
+ "@eslint/object-schema": "^3.0.5",
330
+ "debug": "^4.3.1",
331
+ "minimatch": "^10.2.4"
332
+ },
333
+ "engines": {
334
+ "node": "^20.19.0 || ^22.13.0 || >=24"
335
+ }
336
+ },
337
+ "node_modules/@eslint/config-helpers": {
338
+ "version": "0.5.5",
339
+ "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.5.5.tgz",
340
+ "integrity": "sha512-eIJYKTCECbP/nsKaaruF6LW967mtbQbsw4JTtSVkUQc9MneSkbrgPJAbKl9nWr0ZeowV8BfsarBmPpBzGelA2w==",
341
+ "dev": true,
342
+ "license": "Apache-2.0",
343
+ "dependencies": {
344
+ "@eslint/core": "^1.2.1"
345
+ },
346
+ "engines": {
347
+ "node": "^20.19.0 || ^22.13.0 || >=24"
348
+ }
349
+ },
350
+ "node_modules/@eslint/core": {
351
+ "version": "1.2.1",
352
+ "resolved": "https://registry.npmjs.org/@eslint/core/-/core-1.2.1.tgz",
353
+ "integrity": "sha512-MwcE1P+AZ4C6DWlpin/OmOA54mmIZ/+xZuJiQd4SyB29oAJjN30UW9wkKNptW2ctp4cEsvhlLY/CsQ1uoHDloQ==",
354
+ "dev": true,
355
+ "license": "Apache-2.0",
356
+ "dependencies": {
357
+ "@types/json-schema": "^7.0.15"
358
+ },
359
+ "engines": {
360
+ "node": "^20.19.0 || ^22.13.0 || >=24"
361
+ }
362
+ },
363
+ "node_modules/@eslint/js": {
364
+ "version": "10.0.1",
365
+ "resolved": "https://registry.npmjs.org/@eslint/js/-/js-10.0.1.tgz",
366
+ "integrity": "sha512-zeR9k5pd4gxjZ0abRoIaxdc7I3nDktoXZk2qOv9gCNWx3mVwEn32VRhyLaRsDiJjTs0xq/T8mfPtyuXu7GWBcA==",
367
+ "dev": true,
368
+ "license": "MIT",
369
+ "engines": {
370
+ "node": "^20.19.0 || ^22.13.0 || >=24"
371
+ },
372
+ "funding": {
373
+ "url": "https://eslint.org/donate"
374
+ },
375
+ "peerDependencies": {
376
+ "eslint": "^10.0.0"
377
+ },
378
+ "peerDependenciesMeta": {
379
+ "eslint": {
380
+ "optional": true
381
+ }
382
+ }
383
+ },
384
+ "node_modules/@eslint/object-schema": {
385
+ "version": "3.0.5",
386
+ "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-3.0.5.tgz",
387
+ "integrity": "sha512-vqTaUEgxzm+YDSdElad6PiRoX4t8VGDjCtt05zn4nU810UIx/uNEV7/lZJ6KwFThKZOzOxzXy48da+No7HZaMw==",
388
+ "dev": true,
389
+ "license": "Apache-2.0",
390
+ "engines": {
391
+ "node": "^20.19.0 || ^22.13.0 || >=24"
392
+ }
393
+ },
394
+ "node_modules/@eslint/plugin-kit": {
395
+ "version": "0.7.1",
396
+ "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.7.1.tgz",
397
+ "integrity": "sha512-rZAP3aVgB9ds9KOeUSL+zZ21hPmo8dh6fnIFwRQj5EAZl9gzR7wxYbYXYysAM8CTqGmUGyp2S4kUdV17MnGuWQ==",
398
+ "dev": true,
399
+ "license": "Apache-2.0",
400
+ "dependencies": {
401
+ "@eslint/core": "^1.2.1",
402
+ "levn": "^0.4.1"
403
+ },
404
+ "engines": {
405
+ "node": "^20.19.0 || ^22.13.0 || >=24"
406
+ }
407
+ },
408
+ "node_modules/@humanfs/core": {
409
+ "version": "0.19.2",
410
+ "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.2.tgz",
411
+ "integrity": "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA==",
412
+ "dev": true,
413
+ "license": "Apache-2.0",
414
+ "dependencies": {
415
+ "@humanfs/types": "^0.15.0"
416
+ },
417
+ "engines": {
418
+ "node": ">=18.18.0"
419
+ }
420
+ },
421
+ "node_modules/@humanfs/node": {
422
+ "version": "0.16.8",
423
+ "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.8.tgz",
424
+ "integrity": "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ==",
425
+ "dev": true,
426
+ "license": "Apache-2.0",
427
+ "dependencies": {
428
+ "@humanfs/core": "^0.19.2",
429
+ "@humanfs/types": "^0.15.0",
430
+ "@humanwhocodes/retry": "^0.4.0"
431
+ },
432
+ "engines": {
433
+ "node": ">=18.18.0"
434
+ }
435
+ },
436
+ "node_modules/@humanfs/types": {
437
+ "version": "0.15.0",
438
+ "resolved": "https://registry.npmjs.org/@humanfs/types/-/types-0.15.0.tgz",
439
+ "integrity": "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q==",
440
+ "dev": true,
441
+ "license": "Apache-2.0",
442
+ "engines": {
443
+ "node": ">=18.18.0"
444
+ }
445
+ },
446
+ "node_modules/@humanwhocodes/module-importer": {
447
+ "version": "1.0.1",
448
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
449
+ "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
450
+ "dev": true,
451
+ "license": "Apache-2.0",
452
+ "engines": {
453
+ "node": ">=12.22"
454
+ },
455
+ "funding": {
456
+ "type": "github",
457
+ "url": "https://github.com/sponsors/nzakas"
458
+ }
459
+ },
460
+ "node_modules/@humanwhocodes/retry": {
461
+ "version": "0.4.3",
462
+ "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz",
463
+ "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==",
464
+ "dev": true,
465
+ "license": "Apache-2.0",
466
+ "engines": {
467
+ "node": ">=18.18"
468
+ },
469
+ "funding": {
470
+ "type": "github",
471
+ "url": "https://github.com/sponsors/nzakas"
472
+ }
473
+ },
474
+ "node_modules/@jridgewell/gen-mapping": {
475
+ "version": "0.3.13",
476
+ "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
477
+ "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
478
+ "dev": true,
479
+ "license": "MIT",
480
+ "dependencies": {
481
+ "@jridgewell/sourcemap-codec": "^1.5.0",
482
+ "@jridgewell/trace-mapping": "^0.3.24"
483
+ }
484
+ },
485
+ "node_modules/@jridgewell/remapping": {
486
+ "version": "2.3.5",
487
+ "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
488
+ "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
489
+ "dev": true,
490
+ "license": "MIT",
491
+ "dependencies": {
492
+ "@jridgewell/gen-mapping": "^0.3.5",
493
+ "@jridgewell/trace-mapping": "^0.3.24"
494
+ }
495
+ },
496
+ "node_modules/@jridgewell/resolve-uri": {
497
+ "version": "3.1.2",
498
+ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
499
+ "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
500
+ "dev": true,
501
+ "license": "MIT",
502
+ "engines": {
503
+ "node": ">=6.0.0"
504
+ }
505
+ },
506
+ "node_modules/@jridgewell/sourcemap-codec": {
507
+ "version": "1.5.5",
508
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
509
+ "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
510
+ "dev": true,
511
+ "license": "MIT"
512
+ },
513
+ "node_modules/@jridgewell/trace-mapping": {
514
+ "version": "0.3.31",
515
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
516
+ "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
517
+ "dev": true,
518
+ "license": "MIT",
519
+ "dependencies": {
520
+ "@jridgewell/resolve-uri": "^3.1.0",
521
+ "@jridgewell/sourcemap-codec": "^1.4.14"
522
+ }
523
+ },
524
+ "node_modules/@napi-rs/wasm-runtime": {
525
+ "version": "1.1.4",
526
+ "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-1.1.4.tgz",
527
+ "integrity": "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow==",
528
+ "dev": true,
529
+ "license": "MIT",
530
+ "optional": true,
531
+ "dependencies": {
532
+ "@tybys/wasm-util": "^0.10.1"
533
+ },
534
+ "funding": {
535
+ "type": "github",
536
+ "url": "https://github.com/sponsors/Brooooooklyn"
537
+ },
538
+ "peerDependencies": {
539
+ "@emnapi/core": "^1.7.1",
540
+ "@emnapi/runtime": "^1.7.1"
541
+ }
542
+ },
543
+ "node_modules/@oxc-project/types": {
544
+ "version": "0.129.0",
545
+ "resolved": "https://registry.npmjs.org/@oxc-project/types/-/types-0.129.0.tgz",
546
+ "integrity": "sha512-3oz8m3FGdr2nDXVqmFUw7jolKliC4MoyXYIG2c7gpjBnzUWQpUGIYcXYKxTdTi+N2jusvt610ckTMkxdwHkYEg==",
547
+ "dev": true,
548
+ "license": "MIT",
549
+ "funding": {
550
+ "url": "https://github.com/sponsors/Boshen"
551
+ }
552
+ },
553
+ "node_modules/@rolldown/binding-android-arm64": {
554
+ "version": "1.0.0",
555
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.0.tgz",
556
+ "integrity": "sha512-TWMZnRLMe63C2Lhyicviu7ZHaU4kxa6PS3rofvc9GmcvptzNN11BcfQ4Sl7MwTOsisQoa2keB/EBdNCAnUo8vA==",
557
+ "cpu": [
558
+ "arm64"
559
+ ],
560
+ "dev": true,
561
+ "license": "MIT",
562
+ "optional": true,
563
+ "os": [
564
+ "android"
565
+ ],
566
+ "engines": {
567
+ "node": "^20.19.0 || >=22.12.0"
568
+ }
569
+ },
570
+ "node_modules/@rolldown/binding-darwin-arm64": {
571
+ "version": "1.0.0",
572
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-arm64/-/binding-darwin-arm64-1.0.0.tgz",
573
+ "integrity": "sha512-6XcD+8k0gPVItNagEw78/qqcBDwKcwDYS8V2hRmVsfUSIrd8cWe/CBvRDI5toqFyPfj+FJr6t8U6Xj2P2prEew==",
574
+ "cpu": [
575
+ "arm64"
576
+ ],
577
+ "dev": true,
578
+ "license": "MIT",
579
+ "optional": true,
580
+ "os": [
581
+ "darwin"
582
+ ],
583
+ "engines": {
584
+ "node": "^20.19.0 || >=22.12.0"
585
+ }
586
+ },
587
+ "node_modules/@rolldown/binding-darwin-x64": {
588
+ "version": "1.0.0",
589
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-x64/-/binding-darwin-x64-1.0.0.tgz",
590
+ "integrity": "sha512-iN/tWVXRQDWvmZlKdceP1Dwug9GDpEymhb9p4xnEe6zvCg5lFmzVljl+1qR1NVx3yfGpr2Na+CuLmv5IU8uzfQ==",
591
+ "cpu": [
592
+ "x64"
593
+ ],
594
+ "dev": true,
595
+ "license": "MIT",
596
+ "optional": true,
597
+ "os": [
598
+ "darwin"
599
+ ],
600
+ "engines": {
601
+ "node": "^20.19.0 || >=22.12.0"
602
+ }
603
+ },
604
+ "node_modules/@rolldown/binding-freebsd-x64": {
605
+ "version": "1.0.0",
606
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-freebsd-x64/-/binding-freebsd-x64-1.0.0.tgz",
607
+ "integrity": "sha512-jjQMDvvwSOuhOwMszD/klSOjyWMM3zI64hWTj9KT5x4MxRbZAf+7vLQ6qouRhtsLVFHr3f0ILaJAfgENPiQdAQ==",
608
+ "cpu": [
609
+ "x64"
610
+ ],
611
+ "dev": true,
612
+ "license": "MIT",
613
+ "optional": true,
614
+ "os": [
615
+ "freebsd"
616
+ ],
617
+ "engines": {
618
+ "node": "^20.19.0 || >=22.12.0"
619
+ }
620
+ },
621
+ "node_modules/@rolldown/binding-linux-arm-gnueabihf": {
622
+ "version": "1.0.0",
623
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm-gnueabihf/-/binding-linux-arm-gnueabihf-1.0.0.tgz",
624
+ "integrity": "sha512-d//Dtg2x6/m3mbV64yUGNnDGNZaDGRpDLLNGerHQUVObuNaIQaaDp25yUiqGXtHEXX+NP2d0wAlmKgpYgIAJ2A==",
625
+ "cpu": [
626
+ "arm"
627
+ ],
628
+ "dev": true,
629
+ "license": "MIT",
630
+ "optional": true,
631
+ "os": [
632
+ "linux"
633
+ ],
634
+ "engines": {
635
+ "node": "^20.19.0 || >=22.12.0"
636
+ }
637
+ },
638
+ "node_modules/@rolldown/binding-linux-arm64-gnu": {
639
+ "version": "1.0.0",
640
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-gnu/-/binding-linux-arm64-gnu-1.0.0.tgz",
641
+ "integrity": "sha512-n7Ofp0mx+aB2cC+Sdy5YtMnXtY9lchnHbY+3Yt0uq9JsWQExf4f5Whu0tK0R8Jdc9S6RchTHjIFY7uc92puOVQ==",
642
+ "cpu": [
643
+ "arm64"
644
+ ],
645
+ "dev": true,
646
+ "license": "MIT",
647
+ "optional": true,
648
+ "os": [
649
+ "linux"
650
+ ],
651
+ "engines": {
652
+ "node": "^20.19.0 || >=22.12.0"
653
+ }
654
+ },
655
+ "node_modules/@rolldown/binding-linux-arm64-musl": {
656
+ "version": "1.0.0",
657
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-musl/-/binding-linux-arm64-musl-1.0.0.tgz",
658
+ "integrity": "sha512-EIVjy2cgd7uuMMo94FVkBp7F6DhcZAUwNURkSG3RwUmvAXR6s0ISxM81U+IydcZByPG0pZIHsf1b6kTxoFDgJA==",
659
+ "cpu": [
660
+ "arm64"
661
+ ],
662
+ "dev": true,
663
+ "license": "MIT",
664
+ "optional": true,
665
+ "os": [
666
+ "linux"
667
+ ],
668
+ "engines": {
669
+ "node": "^20.19.0 || >=22.12.0"
670
+ }
671
+ },
672
+ "node_modules/@rolldown/binding-linux-ppc64-gnu": {
673
+ "version": "1.0.0",
674
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-ppc64-gnu/-/binding-linux-ppc64-gnu-1.0.0.tgz",
675
+ "integrity": "sha512-JEwwOPcwTLAcpDQlqSmjEmfs63xJnSiUNIGvLcDLUHCWK4XowpS/7c7tUsUH6uT/ct6bMUTdXKfI8967FYj6mg==",
676
+ "cpu": [
677
+ "ppc64"
678
+ ],
679
+ "dev": true,
680
+ "license": "MIT",
681
+ "optional": true,
682
+ "os": [
683
+ "linux"
684
+ ],
685
+ "engines": {
686
+ "node": "^20.19.0 || >=22.12.0"
687
+ }
688
+ },
689
+ "node_modules/@rolldown/binding-linux-s390x-gnu": {
690
+ "version": "1.0.0",
691
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-s390x-gnu/-/binding-linux-s390x-gnu-1.0.0.tgz",
692
+ "integrity": "sha512-0wjCFhLrihtAubnT9iA0N++0pSV0z5Hg7tNGdNJ4RFaINceHadoF+kiFGyY1qSSNVIAZtLotG8Ju1bgDPkjnFA==",
693
+ "cpu": [
694
+ "s390x"
695
+ ],
696
+ "dev": true,
697
+ "license": "MIT",
698
+ "optional": true,
699
+ "os": [
700
+ "linux"
701
+ ],
702
+ "engines": {
703
+ "node": "^20.19.0 || >=22.12.0"
704
+ }
705
+ },
706
+ "node_modules/@rolldown/binding-linux-x64-gnu": {
707
+ "version": "1.0.0",
708
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.0.tgz",
709
+ "integrity": "sha512-Dfn7iak9BcMMePxcoJfpSbWqnEyrp/dRF63/8qW/eHBdOZov6x5aShLLEYGYdIeSJ6vMLK/XCVB+lGIxm41bQA==",
710
+ "cpu": [
711
+ "x64"
712
+ ],
713
+ "dev": true,
714
+ "license": "MIT",
715
+ "optional": true,
716
+ "os": [
717
+ "linux"
718
+ ],
719
+ "engines": {
720
+ "node": "^20.19.0 || >=22.12.0"
721
+ }
722
+ },
723
+ "node_modules/@rolldown/binding-linux-x64-musl": {
724
+ "version": "1.0.0",
725
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.0.tgz",
726
+ "integrity": "sha512-5/utzzDmD/pD/bmuaUcbTf/sZYy0aztwIVlfpoW1fTjCZ0BaPOMVWGZL1zvgxyi7ZIVYWlxKONHmSbHuiOh8Jw==",
727
+ "cpu": [
728
+ "x64"
729
+ ],
730
+ "dev": true,
731
+ "license": "MIT",
732
+ "optional": true,
733
+ "os": [
734
+ "linux"
735
+ ],
736
+ "engines": {
737
+ "node": "^20.19.0 || >=22.12.0"
738
+ }
739
+ },
740
+ "node_modules/@rolldown/binding-openharmony-arm64": {
741
+ "version": "1.0.0",
742
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-openharmony-arm64/-/binding-openharmony-arm64-1.0.0.tgz",
743
+ "integrity": "sha512-ouJs8VcUomfLfpbUECqFMRqdV4x6aeAK3MA4m6vTrJJjKyWTV5KnxZx7Jd9G+GlDaQQxubcba00x16OyJ1meig==",
744
+ "cpu": [
745
+ "arm64"
746
+ ],
747
+ "dev": true,
748
+ "license": "MIT",
749
+ "optional": true,
750
+ "os": [
751
+ "openharmony"
752
+ ],
753
+ "engines": {
754
+ "node": "^20.19.0 || >=22.12.0"
755
+ }
756
+ },
757
+ "node_modules/@rolldown/binding-wasm32-wasi": {
758
+ "version": "1.0.0",
759
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-wasm32-wasi/-/binding-wasm32-wasi-1.0.0.tgz",
760
+ "integrity": "sha512-E+oHKGiDA+lsKMmFtffDDw91EryDT7uJocrIuCHqhm6bCTM6xFK+3gaCkYOHfPwQr0cCNarSM2xaELoQDz9jJg==",
761
+ "cpu": [
762
+ "wasm32"
763
+ ],
764
+ "dev": true,
765
+ "license": "MIT",
766
+ "optional": true,
767
+ "dependencies": {
768
+ "@emnapi/core": "1.10.0",
769
+ "@emnapi/runtime": "1.10.0",
770
+ "@napi-rs/wasm-runtime": "^1.1.4"
771
+ },
772
+ "engines": {
773
+ "node": "^20.19.0 || >=22.12.0"
774
+ }
775
+ },
776
+ "node_modules/@rolldown/binding-win32-arm64-msvc": {
777
+ "version": "1.0.0",
778
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-arm64-msvc/-/binding-win32-arm64-msvc-1.0.0.tgz",
779
+ "integrity": "sha512-yYK02n8Rngo+gbm1y6G0+7jk1sJ/2Wt7K0me0Y7k/ErBpyf+LJ2gFpqWVTcRV1rUepBlQRmpgWkTQCiiwrK0Ow==",
780
+ "cpu": [
781
+ "arm64"
782
+ ],
783
+ "dev": true,
784
+ "license": "MIT",
785
+ "optional": true,
786
+ "os": [
787
+ "win32"
788
+ ],
789
+ "engines": {
790
+ "node": "^20.19.0 || >=22.12.0"
791
+ }
792
+ },
793
+ "node_modules/@rolldown/binding-win32-x64-msvc": {
794
+ "version": "1.0.0",
795
+ "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-x64-msvc/-/binding-win32-x64-msvc-1.0.0.tgz",
796
+ "integrity": "sha512-14bpChMahXRRXiTwahSl+zzHPW6qQTXtkMuJBFlbo+pqSAews2d4BdCSHfrJ/MBsCZtpmTafsY+1QhBzitcmdg==",
797
+ "cpu": [
798
+ "x64"
799
+ ],
800
+ "dev": true,
801
+ "license": "MIT",
802
+ "optional": true,
803
+ "os": [
804
+ "win32"
805
+ ],
806
+ "engines": {
807
+ "node": "^20.19.0 || >=22.12.0"
808
+ }
809
+ },
810
+ "node_modules/@rolldown/pluginutils": {
811
+ "version": "1.0.0-rc.7",
812
+ "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-rc.7.tgz",
813
+ "integrity": "sha512-qujRfC8sFVInYSPPMLQByRh7zhwkGFS4+tyMQ83srV1qrxL4g8E2tyxVVyxd0+8QeBM1mIk9KbWxkegRr76XzA==",
814
+ "dev": true,
815
+ "license": "MIT"
816
+ },
817
+ "node_modules/@tailwindcss/node": {
818
+ "version": "4.3.0",
819
+ "resolved": "https://registry.npmjs.org/@tailwindcss/node/-/node-4.3.0.tgz",
820
+ "integrity": "sha512-aFb4gUhFOgdh9AXo4IzBEOzBkkAxm9VigwDJnMIYv3lcfXCJVesNfbEaBl4BNgVRyid92AmdviqwBUBRKSeY3g==",
821
+ "dev": true,
822
+ "license": "MIT",
823
+ "dependencies": {
824
+ "@jridgewell/remapping": "^2.3.5",
825
+ "enhanced-resolve": "^5.21.0",
826
+ "jiti": "^2.6.1",
827
+ "lightningcss": "1.32.0",
828
+ "magic-string": "^0.30.21",
829
+ "source-map-js": "^1.2.1",
830
+ "tailwindcss": "4.3.0"
831
+ }
832
+ },
833
+ "node_modules/@tailwindcss/oxide": {
834
+ "version": "4.3.0",
835
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide/-/oxide-4.3.0.tgz",
836
+ "integrity": "sha512-F7HZGBeN9I0/AuuJS5PwcD8xayx5ri5GhjYUDBEVYUkexyA/giwbDNjRVrxSezE3T250OU2K/wp/ltWx3UOefg==",
837
+ "dev": true,
838
+ "license": "MIT",
839
+ "engines": {
840
+ "node": ">= 20"
841
+ },
842
+ "optionalDependencies": {
843
+ "@tailwindcss/oxide-android-arm64": "4.3.0",
844
+ "@tailwindcss/oxide-darwin-arm64": "4.3.0",
845
+ "@tailwindcss/oxide-darwin-x64": "4.3.0",
846
+ "@tailwindcss/oxide-freebsd-x64": "4.3.0",
847
+ "@tailwindcss/oxide-linux-arm-gnueabihf": "4.3.0",
848
+ "@tailwindcss/oxide-linux-arm64-gnu": "4.3.0",
849
+ "@tailwindcss/oxide-linux-arm64-musl": "4.3.0",
850
+ "@tailwindcss/oxide-linux-x64-gnu": "4.3.0",
851
+ "@tailwindcss/oxide-linux-x64-musl": "4.3.0",
852
+ "@tailwindcss/oxide-wasm32-wasi": "4.3.0",
853
+ "@tailwindcss/oxide-win32-arm64-msvc": "4.3.0",
854
+ "@tailwindcss/oxide-win32-x64-msvc": "4.3.0"
855
+ }
856
+ },
857
+ "node_modules/@tailwindcss/oxide-android-arm64": {
858
+ "version": "4.3.0",
859
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-android-arm64/-/oxide-android-arm64-4.3.0.tgz",
860
+ "integrity": "sha512-TJPiq67tKlLuObP6RkwvVGDoxCMBVtDgKkLfa/uyj7/FyxvQwHS+UOnVrXXgbEsfUaMgiVvC4KbJnRr26ho4Ng==",
861
+ "cpu": [
862
+ "arm64"
863
+ ],
864
+ "dev": true,
865
+ "license": "MIT",
866
+ "optional": true,
867
+ "os": [
868
+ "android"
869
+ ],
870
+ "engines": {
871
+ "node": ">= 20"
872
+ }
873
+ },
874
+ "node_modules/@tailwindcss/oxide-darwin-arm64": {
875
+ "version": "4.3.0",
876
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-arm64/-/oxide-darwin-arm64-4.3.0.tgz",
877
+ "integrity": "sha512-oMN/WZRb+SO37BmUElEgeEWuU8E/HXRkiODxJxLe1UTHVXLrdVSgfaJV7pSlhRGMSOiXLuxTIjfsF3wYvz8cgQ==",
878
+ "cpu": [
879
+ "arm64"
880
+ ],
881
+ "dev": true,
882
+ "license": "MIT",
883
+ "optional": true,
884
+ "os": [
885
+ "darwin"
886
+ ],
887
+ "engines": {
888
+ "node": ">= 20"
889
+ }
890
+ },
891
+ "node_modules/@tailwindcss/oxide-darwin-x64": {
892
+ "version": "4.3.0",
893
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-darwin-x64/-/oxide-darwin-x64-4.3.0.tgz",
894
+ "integrity": "sha512-N6CUmu4a6bKVADfw77p+iw6Yd9Q3OBhe0veaDX+QazfuVYlQsHfDgxBrsjQ/IW+zywL8mTrNd0SdJT/zgtvMdA==",
895
+ "cpu": [
896
+ "x64"
897
+ ],
898
+ "dev": true,
899
+ "license": "MIT",
900
+ "optional": true,
901
+ "os": [
902
+ "darwin"
903
+ ],
904
+ "engines": {
905
+ "node": ">= 20"
906
+ }
907
+ },
908
+ "node_modules/@tailwindcss/oxide-freebsd-x64": {
909
+ "version": "4.3.0",
910
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-freebsd-x64/-/oxide-freebsd-x64-4.3.0.tgz",
911
+ "integrity": "sha512-zDL5hBkQdH5C6MpqbK3gQAgP80tsMwSI26vjOzjJtNCMUo0lFgOItzHKBIupOZNQxt3ouPH7RPhvNhiTfCe5CQ==",
912
+ "cpu": [
913
+ "x64"
914
+ ],
915
+ "dev": true,
916
+ "license": "MIT",
917
+ "optional": true,
918
+ "os": [
919
+ "freebsd"
920
+ ],
921
+ "engines": {
922
+ "node": ">= 20"
923
+ }
924
+ },
925
+ "node_modules/@tailwindcss/oxide-linux-arm-gnueabihf": {
926
+ "version": "4.3.0",
927
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm-gnueabihf/-/oxide-linux-arm-gnueabihf-4.3.0.tgz",
928
+ "integrity": "sha512-R06HdNi7A7OEoMsf6d4tjZ71RCWnZQPHj2mnotSFURjNLdBC+cIgXQ7l81CqeoiQftjf6OOblxXMInMgN2VzMA==",
929
+ "cpu": [
930
+ "arm"
931
+ ],
932
+ "dev": true,
933
+ "license": "MIT",
934
+ "optional": true,
935
+ "os": [
936
+ "linux"
937
+ ],
938
+ "engines": {
939
+ "node": ">= 20"
940
+ }
941
+ },
942
+ "node_modules/@tailwindcss/oxide-linux-arm64-gnu": {
943
+ "version": "4.3.0",
944
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-gnu/-/oxide-linux-arm64-gnu-4.3.0.tgz",
945
+ "integrity": "sha512-qTJHELX8jetjhRQHCLilkVLmybpzNQAtaI/gaoVoidn/ufbNDbAo8KlK2J+yPoc8wQxvDxCmh/5lr8nC1+lTbg==",
946
+ "cpu": [
947
+ "arm64"
948
+ ],
949
+ "dev": true,
950
+ "license": "MIT",
951
+ "optional": true,
952
+ "os": [
953
+ "linux"
954
+ ],
955
+ "engines": {
956
+ "node": ">= 20"
957
+ }
958
+ },
959
+ "node_modules/@tailwindcss/oxide-linux-arm64-musl": {
960
+ "version": "4.3.0",
961
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-arm64-musl/-/oxide-linux-arm64-musl-4.3.0.tgz",
962
+ "integrity": "sha512-Z6sukiQsngnWO+l39X4pPbiWT81IC+PLKF+PHxIlyZbGNb9MODfYlXEVlFvej5BOZInWX01kVyzeLvHsXhfczQ==",
963
+ "cpu": [
964
+ "arm64"
965
+ ],
966
+ "dev": true,
967
+ "license": "MIT",
968
+ "optional": true,
969
+ "os": [
970
+ "linux"
971
+ ],
972
+ "engines": {
973
+ "node": ">= 20"
974
+ }
975
+ },
976
+ "node_modules/@tailwindcss/oxide-linux-x64-gnu": {
977
+ "version": "4.3.0",
978
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-gnu/-/oxide-linux-x64-gnu-4.3.0.tgz",
979
+ "integrity": "sha512-DRNdQRpSGzRGfARVuVkxvM8Q12nh19l4BF/G7zGA1oe+9wcC6saFBHTISrpIcKzhiXtSrlSrluCfvMuledoCTQ==",
980
+ "cpu": [
981
+ "x64"
982
+ ],
983
+ "dev": true,
984
+ "license": "MIT",
985
+ "optional": true,
986
+ "os": [
987
+ "linux"
988
+ ],
989
+ "engines": {
990
+ "node": ">= 20"
991
+ }
992
+ },
993
+ "node_modules/@tailwindcss/oxide-linux-x64-musl": {
994
+ "version": "4.3.0",
995
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-linux-x64-musl/-/oxide-linux-x64-musl-4.3.0.tgz",
996
+ "integrity": "sha512-Z0IADbDo8bh6I7h2IQMx601AdXBLfFpEdUotft86evd/8ZPflZe9COPO8Q1vw+pfLWIUo9zN/JGZvwuAJqduqg==",
997
+ "cpu": [
998
+ "x64"
999
+ ],
1000
+ "dev": true,
1001
+ "license": "MIT",
1002
+ "optional": true,
1003
+ "os": [
1004
+ "linux"
1005
+ ],
1006
+ "engines": {
1007
+ "node": ">= 20"
1008
+ }
1009
+ },
1010
+ "node_modules/@tailwindcss/oxide-wasm32-wasi": {
1011
+ "version": "4.3.0",
1012
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-wasm32-wasi/-/oxide-wasm32-wasi-4.3.0.tgz",
1013
+ "integrity": "sha512-HNZGOUxEmElksYR7S6sC5jTeNGpobAsy9u7Gu0AskJ8/20FR9GqebUyB+HBcU/ax6BHuiuJi+Oda4B+YX6H1yA==",
1014
+ "bundleDependencies": [
1015
+ "@napi-rs/wasm-runtime",
1016
+ "@emnapi/core",
1017
+ "@emnapi/runtime",
1018
+ "@tybys/wasm-util",
1019
+ "@emnapi/wasi-threads",
1020
+ "tslib"
1021
+ ],
1022
+ "cpu": [
1023
+ "wasm32"
1024
+ ],
1025
+ "dev": true,
1026
+ "license": "MIT",
1027
+ "optional": true,
1028
+ "dependencies": {
1029
+ "@emnapi/core": "^1.10.0",
1030
+ "@emnapi/runtime": "^1.10.0",
1031
+ "@emnapi/wasi-threads": "^1.2.1",
1032
+ "@napi-rs/wasm-runtime": "^1.1.4",
1033
+ "@tybys/wasm-util": "^0.10.1",
1034
+ "tslib": "^2.8.1"
1035
+ },
1036
+ "engines": {
1037
+ "node": ">=14.0.0"
1038
+ }
1039
+ },
1040
+ "node_modules/@tailwindcss/oxide-win32-arm64-msvc": {
1041
+ "version": "4.3.0",
1042
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-arm64-msvc/-/oxide-win32-arm64-msvc-4.3.0.tgz",
1043
+ "integrity": "sha512-Pe+RPVTi1T+qymuuRpcdvwSVZjnll/f7n8gBxMMh3xLTctMDKqpdfGimbMyioqtLhUYZxdJ9wGNhV7MKHvgZsQ==",
1044
+ "cpu": [
1045
+ "arm64"
1046
+ ],
1047
+ "dev": true,
1048
+ "license": "MIT",
1049
+ "optional": true,
1050
+ "os": [
1051
+ "win32"
1052
+ ],
1053
+ "engines": {
1054
+ "node": ">= 20"
1055
+ }
1056
+ },
1057
+ "node_modules/@tailwindcss/oxide-win32-x64-msvc": {
1058
+ "version": "4.3.0",
1059
+ "resolved": "https://registry.npmjs.org/@tailwindcss/oxide-win32-x64-msvc/-/oxide-win32-x64-msvc-4.3.0.tgz",
1060
+ "integrity": "sha512-Mvrf2kXW/yeW/OTezZlCGOirXRcUuLIBx/5Y12BaPM7wJoryG6dfS/NJL8aBPqtTEx/Vm4T4vKzFUcKDT+TKUA==",
1061
+ "cpu": [
1062
+ "x64"
1063
+ ],
1064
+ "dev": true,
1065
+ "license": "MIT",
1066
+ "optional": true,
1067
+ "os": [
1068
+ "win32"
1069
+ ],
1070
+ "engines": {
1071
+ "node": ">= 20"
1072
+ }
1073
+ },
1074
+ "node_modules/@tailwindcss/vite": {
1075
+ "version": "4.3.0",
1076
+ "resolved": "https://registry.npmjs.org/@tailwindcss/vite/-/vite-4.3.0.tgz",
1077
+ "integrity": "sha512-t6J3OrB5Fc0ExuhohouH0fWUGMYL6PTLhW+E7zIk/pdbnJARZDCwjBznFnkh5ynRnIRSI4YjtTH0t6USjJISrw==",
1078
+ "dev": true,
1079
+ "license": "MIT",
1080
+ "dependencies": {
1081
+ "@tailwindcss/node": "4.3.0",
1082
+ "@tailwindcss/oxide": "4.3.0",
1083
+ "tailwindcss": "4.3.0"
1084
+ },
1085
+ "peerDependencies": {
1086
+ "vite": "^5.2.0 || ^6 || ^7 || ^8"
1087
+ }
1088
+ },
1089
+ "node_modules/@tybys/wasm-util": {
1090
+ "version": "0.10.2",
1091
+ "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz",
1092
+ "integrity": "sha512-RoBvJ2X0wuKlWFIjrwffGw1IqZHKQqzIchKaadZZfnNpsAYp2mM0h36JtPCjNDAHGgYez/15uMBpfGwchhiMgg==",
1093
+ "dev": true,
1094
+ "license": "MIT",
1095
+ "optional": true,
1096
+ "dependencies": {
1097
+ "tslib": "^2.4.0"
1098
+ }
1099
+ },
1100
+ "node_modules/@types/esrecurse": {
1101
+ "version": "4.3.1",
1102
+ "resolved": "https://registry.npmjs.org/@types/esrecurse/-/esrecurse-4.3.1.tgz",
1103
+ "integrity": "sha512-xJBAbDifo5hpffDBuHl0Y8ywswbiAp/Wi7Y/GtAgSlZyIABppyurxVueOPE8LUQOxdlgi6Zqce7uoEpqNTeiUw==",
1104
+ "dev": true,
1105
+ "license": "MIT"
1106
+ },
1107
+ "node_modules/@types/estree": {
1108
+ "version": "1.0.9",
1109
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.9.tgz",
1110
+ "integrity": "sha512-GhdPgy1el4/ImP05X05Uw4cw2/M93BCUmnEvWZNStlCzEKME4Fkk+YpoA5OiHNQmoS7Cafb8Xa3Pya8m1Qrzeg==",
1111
+ "dev": true,
1112
+ "license": "MIT"
1113
+ },
1114
+ "node_modules/@types/json-schema": {
1115
+ "version": "7.0.15",
1116
+ "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
1117
+ "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==",
1118
+ "dev": true,
1119
+ "license": "MIT"
1120
+ },
1121
+ "node_modules/@types/react": {
1122
+ "version": "19.2.14",
1123
+ "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.14.tgz",
1124
+ "integrity": "sha512-ilcTH/UniCkMdtexkoCN0bI7pMcJDvmQFPvuPvmEaYA/NSfFTAgdUSLAoVjaRJm7+6PvcM+q1zYOwS4wTYMF9w==",
1125
+ "dev": true,
1126
+ "license": "MIT",
1127
+ "peer": true,
1128
+ "dependencies": {
1129
+ "csstype": "^3.2.2"
1130
+ }
1131
+ },
1132
+ "node_modules/@types/react-dom": {
1133
+ "version": "19.2.3",
1134
+ "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz",
1135
+ "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==",
1136
+ "dev": true,
1137
+ "license": "MIT",
1138
+ "peerDependencies": {
1139
+ "@types/react": "^19.2.0"
1140
+ }
1141
+ },
1142
+ "node_modules/@vitejs/plugin-react": {
1143
+ "version": "6.0.1",
1144
+ "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.1.tgz",
1145
+ "integrity": "sha512-l9X/E3cDb+xY3SWzlG1MOGt2usfEHGMNIaegaUGFsLkb3RCn/k8/TOXBcab+OndDI4TBtktT8/9BwwW8Vi9KUQ==",
1146
+ "dev": true,
1147
+ "license": "MIT",
1148
+ "dependencies": {
1149
+ "@rolldown/pluginutils": "1.0.0-rc.7"
1150
+ },
1151
+ "engines": {
1152
+ "node": "^20.19.0 || >=22.12.0"
1153
+ },
1154
+ "peerDependencies": {
1155
+ "@rolldown/plugin-babel": "^0.1.7 || ^0.2.0",
1156
+ "babel-plugin-react-compiler": "^1.0.0",
1157
+ "vite": "^8.0.0"
1158
+ },
1159
+ "peerDependenciesMeta": {
1160
+ "@rolldown/plugin-babel": {
1161
+ "optional": true
1162
+ },
1163
+ "babel-plugin-react-compiler": {
1164
+ "optional": true
1165
+ }
1166
+ }
1167
+ },
1168
+ "node_modules/acorn": {
1169
+ "version": "8.16.0",
1170
+ "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz",
1171
+ "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==",
1172
+ "dev": true,
1173
+ "license": "MIT",
1174
+ "peer": true,
1175
+ "bin": {
1176
+ "acorn": "bin/acorn"
1177
+ },
1178
+ "engines": {
1179
+ "node": ">=0.4.0"
1180
+ }
1181
+ },
1182
+ "node_modules/acorn-jsx": {
1183
+ "version": "5.3.2",
1184
+ "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz",
1185
+ "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==",
1186
+ "dev": true,
1187
+ "license": "MIT",
1188
+ "peerDependencies": {
1189
+ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
1190
+ }
1191
+ },
1192
+ "node_modules/ajv": {
1193
+ "version": "6.15.0",
1194
+ "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.15.0.tgz",
1195
+ "integrity": "sha512-fgFx7Hfoq60ytK2c7DhnF8jIvzYgOMxfugjLOSMHjLIPgenqa7S7oaagATUq99mV6IYvN2tRmC0wnTYX6iPbMw==",
1196
+ "dev": true,
1197
+ "license": "MIT",
1198
+ "dependencies": {
1199
+ "fast-deep-equal": "^3.1.1",
1200
+ "fast-json-stable-stringify": "^2.0.0",
1201
+ "json-schema-traverse": "^0.4.1",
1202
+ "uri-js": "^4.2.2"
1203
+ },
1204
+ "funding": {
1205
+ "type": "github",
1206
+ "url": "https://github.com/sponsors/epoberezkin"
1207
+ }
1208
+ },
1209
+ "node_modules/balanced-match": {
1210
+ "version": "4.0.4",
1211
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz",
1212
+ "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==",
1213
+ "dev": true,
1214
+ "license": "MIT",
1215
+ "engines": {
1216
+ "node": "18 || 20 || >=22"
1217
+ }
1218
+ },
1219
+ "node_modules/baseline-browser-mapping": {
1220
+ "version": "2.10.29",
1221
+ "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.29.tgz",
1222
+ "integrity": "sha512-Asa2krT+XTPZINCS+2QcyS8WTkObE77RwkydwF7h6DmnKqbvlalz93m/dnphUyCa6SWSP51VgtEUf2FN+gelFQ==",
1223
+ "dev": true,
1224
+ "license": "Apache-2.0",
1225
+ "bin": {
1226
+ "baseline-browser-mapping": "dist/cli.cjs"
1227
+ },
1228
+ "engines": {
1229
+ "node": ">=6.0.0"
1230
+ }
1231
+ },
1232
+ "node_modules/brace-expansion": {
1233
+ "version": "5.0.6",
1234
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.6.tgz",
1235
+ "integrity": "sha512-kLpxurY4Z4r9sgMsyG0Z9uzsBlgiU/EFKhj/h91/8yHu0edo7XuixOIH3VcJ8kkxs6/jPzoI6U9Vj3WqbMQ94g==",
1236
+ "dev": true,
1237
+ "license": "MIT",
1238
+ "dependencies": {
1239
+ "balanced-match": "^4.0.2"
1240
+ },
1241
+ "engines": {
1242
+ "node": "18 || 20 || >=22"
1243
+ }
1244
+ },
1245
+ "node_modules/browserslist": {
1246
+ "version": "4.28.2",
1247
+ "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz",
1248
+ "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==",
1249
+ "dev": true,
1250
+ "funding": [
1251
+ {
1252
+ "type": "opencollective",
1253
+ "url": "https://opencollective.com/browserslist"
1254
+ },
1255
+ {
1256
+ "type": "tidelift",
1257
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
1258
+ },
1259
+ {
1260
+ "type": "github",
1261
+ "url": "https://github.com/sponsors/ai"
1262
+ }
1263
+ ],
1264
+ "license": "MIT",
1265
+ "peer": true,
1266
+ "dependencies": {
1267
+ "baseline-browser-mapping": "^2.10.12",
1268
+ "caniuse-lite": "^1.0.30001782",
1269
+ "electron-to-chromium": "^1.5.328",
1270
+ "node-releases": "^2.0.36",
1271
+ "update-browserslist-db": "^1.2.3"
1272
+ },
1273
+ "bin": {
1274
+ "browserslist": "cli.js"
1275
+ },
1276
+ "engines": {
1277
+ "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
1278
+ }
1279
+ },
1280
+ "node_modules/caniuse-lite": {
1281
+ "version": "1.0.30001792",
1282
+ "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001792.tgz",
1283
+ "integrity": "sha512-hVLMUZFgR4JJ6ACt1uEESvQN1/dBVqPAKY0hgrV70eN3391K6juAfTjKZLKvOMsx8PxA7gsY1/tLMMTcfFLLpw==",
1284
+ "dev": true,
1285
+ "funding": [
1286
+ {
1287
+ "type": "opencollective",
1288
+ "url": "https://opencollective.com/browserslist"
1289
+ },
1290
+ {
1291
+ "type": "tidelift",
1292
+ "url": "https://tidelift.com/funding/github/npm/caniuse-lite"
1293
+ },
1294
+ {
1295
+ "type": "github",
1296
+ "url": "https://github.com/sponsors/ai"
1297
+ }
1298
+ ],
1299
+ "license": "CC-BY-4.0"
1300
+ },
1301
+ "node_modules/convert-source-map": {
1302
+ "version": "2.0.0",
1303
+ "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
1304
+ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==",
1305
+ "dev": true,
1306
+ "license": "MIT"
1307
+ },
1308
+ "node_modules/cross-spawn": {
1309
+ "version": "7.0.6",
1310
+ "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
1311
+ "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==",
1312
+ "dev": true,
1313
+ "license": "MIT",
1314
+ "dependencies": {
1315
+ "path-key": "^3.1.0",
1316
+ "shebang-command": "^2.0.0",
1317
+ "which": "^2.0.1"
1318
+ },
1319
+ "engines": {
1320
+ "node": ">= 8"
1321
+ }
1322
+ },
1323
+ "node_modules/csstype": {
1324
+ "version": "3.2.3",
1325
+ "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
1326
+ "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==",
1327
+ "dev": true,
1328
+ "license": "MIT"
1329
+ },
1330
+ "node_modules/debug": {
1331
+ "version": "4.4.3",
1332
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
1333
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
1334
+ "dev": true,
1335
+ "license": "MIT",
1336
+ "dependencies": {
1337
+ "ms": "^2.1.3"
1338
+ },
1339
+ "engines": {
1340
+ "node": ">=6.0"
1341
+ },
1342
+ "peerDependenciesMeta": {
1343
+ "supports-color": {
1344
+ "optional": true
1345
+ }
1346
+ }
1347
+ },
1348
+ "node_modules/deep-is": {
1349
+ "version": "0.1.4",
1350
+ "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
1351
+ "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==",
1352
+ "dev": true,
1353
+ "license": "MIT"
1354
+ },
1355
+ "node_modules/detect-libc": {
1356
+ "version": "2.1.2",
1357
+ "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
1358
+ "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
1359
+ "dev": true,
1360
+ "license": "Apache-2.0",
1361
+ "engines": {
1362
+ "node": ">=8"
1363
+ }
1364
+ },
1365
+ "node_modules/electron-to-chromium": {
1366
+ "version": "1.5.354",
1367
+ "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.354.tgz",
1368
+ "integrity": "sha512-JaBHwWcfIdmSAfWM5l3uwjGd431j8YEMikZ+K/2nXVuBqJKyZ0f+2h4n4JY5AyNiZmnY9qQr2RU3v9DxDmHMNg==",
1369
+ "dev": true,
1370
+ "license": "ISC"
1371
+ },
1372
+ "node_modules/enhanced-resolve": {
1373
+ "version": "5.21.3",
1374
+ "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.21.3.tgz",
1375
+ "integrity": "sha512-QyL119InA+XXEkNLNTPCXPugSvOfhwv0JOlGNzvxs0hZaiHLNvXSpudUWsOlsXGWJh8G6ckCScEkVHfX3kw/2Q==",
1376
+ "dev": true,
1377
+ "license": "MIT",
1378
+ "dependencies": {
1379
+ "graceful-fs": "^4.2.4",
1380
+ "tapable": "^2.3.3"
1381
+ },
1382
+ "engines": {
1383
+ "node": ">=10.13.0"
1384
+ }
1385
+ },
1386
+ "node_modules/escalade": {
1387
+ "version": "3.2.0",
1388
+ "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
1389
+ "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
1390
+ "dev": true,
1391
+ "license": "MIT",
1392
+ "engines": {
1393
+ "node": ">=6"
1394
+ }
1395
+ },
1396
+ "node_modules/escape-string-regexp": {
1397
+ "version": "4.0.0",
1398
+ "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
1399
+ "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
1400
+ "dev": true,
1401
+ "license": "MIT",
1402
+ "engines": {
1403
+ "node": ">=10"
1404
+ },
1405
+ "funding": {
1406
+ "url": "https://github.com/sponsors/sindresorhus"
1407
+ }
1408
+ },
1409
+ "node_modules/eslint": {
1410
+ "version": "10.3.0",
1411
+ "resolved": "https://registry.npmjs.org/eslint/-/eslint-10.3.0.tgz",
1412
+ "integrity": "sha512-XbEXaRva5cF0ZQB8w6MluHA0kZZfV2DuCMJ3ozyEOHLwDpZX2Lmm/7Pp0xdJmI0GL1W05VH5VwIFHEm1Vcw2gw==",
1413
+ "dev": true,
1414
+ "license": "MIT",
1415
+ "peer": true,
1416
+ "dependencies": {
1417
+ "@eslint-community/eslint-utils": "^4.8.0",
1418
+ "@eslint-community/regexpp": "^4.12.2",
1419
+ "@eslint/config-array": "^0.23.5",
1420
+ "@eslint/config-helpers": "^0.5.5",
1421
+ "@eslint/core": "^1.2.1",
1422
+ "@eslint/plugin-kit": "^0.7.1",
1423
+ "@humanfs/node": "^0.16.6",
1424
+ "@humanwhocodes/module-importer": "^1.0.1",
1425
+ "@humanwhocodes/retry": "^0.4.2",
1426
+ "@types/estree": "^1.0.6",
1427
+ "ajv": "^6.14.0",
1428
+ "cross-spawn": "^7.0.6",
1429
+ "debug": "^4.3.2",
1430
+ "escape-string-regexp": "^4.0.0",
1431
+ "eslint-scope": "^9.1.2",
1432
+ "eslint-visitor-keys": "^5.0.1",
1433
+ "espree": "^11.2.0",
1434
+ "esquery": "^1.7.0",
1435
+ "esutils": "^2.0.2",
1436
+ "fast-deep-equal": "^3.1.3",
1437
+ "file-entry-cache": "^8.0.0",
1438
+ "find-up": "^5.0.0",
1439
+ "glob-parent": "^6.0.2",
1440
+ "ignore": "^5.2.0",
1441
+ "imurmurhash": "^0.1.4",
1442
+ "is-glob": "^4.0.0",
1443
+ "json-stable-stringify-without-jsonify": "^1.0.1",
1444
+ "minimatch": "^10.2.4",
1445
+ "natural-compare": "^1.4.0",
1446
+ "optionator": "^0.9.3"
1447
+ },
1448
+ "bin": {
1449
+ "eslint": "bin/eslint.js"
1450
+ },
1451
+ "engines": {
1452
+ "node": "^20.19.0 || ^22.13.0 || >=24"
1453
+ },
1454
+ "funding": {
1455
+ "url": "https://eslint.org/donate"
1456
+ },
1457
+ "peerDependencies": {
1458
+ "jiti": "*"
1459
+ },
1460
+ "peerDependenciesMeta": {
1461
+ "jiti": {
1462
+ "optional": true
1463
+ }
1464
+ }
1465
+ },
1466
+ "node_modules/eslint-plugin-react-hooks": {
1467
+ "version": "7.1.1",
1468
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.1.1.tgz",
1469
+ "integrity": "sha512-f2I7Gw6JbvCexzIInuSbZpfdQ44D7iqdWX01FKLvrPgqxoE7oMj8clOfto8U6vYiz4yd5oKu39rRSVOe1zRu0g==",
1470
+ "dev": true,
1471
+ "license": "MIT",
1472
+ "dependencies": {
1473
+ "@babel/core": "^7.24.4",
1474
+ "@babel/parser": "^7.24.4",
1475
+ "hermes-parser": "^0.25.1",
1476
+ "zod": "^3.25.0 || ^4.0.0",
1477
+ "zod-validation-error": "^3.5.0 || ^4.0.0"
1478
+ },
1479
+ "engines": {
1480
+ "node": ">=18"
1481
+ },
1482
+ "peerDependencies": {
1483
+ "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"
1484
+ }
1485
+ },
1486
+ "node_modules/eslint-plugin-react-refresh": {
1487
+ "version": "0.5.2",
1488
+ "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.5.2.tgz",
1489
+ "integrity": "sha512-hmgTH57GfzoTFjVN0yBwTggnsVUF2tcqi7RJZHqi9lIezSs4eFyAMktA68YD4r5kNw1mxyY4dmkyoFDb3FIqrA==",
1490
+ "dev": true,
1491
+ "license": "MIT",
1492
+ "peerDependencies": {
1493
+ "eslint": "^9 || ^10"
1494
+ }
1495
+ },
1496
+ "node_modules/eslint-scope": {
1497
+ "version": "9.1.2",
1498
+ "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-9.1.2.tgz",
1499
+ "integrity": "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ==",
1500
+ "dev": true,
1501
+ "license": "BSD-2-Clause",
1502
+ "dependencies": {
1503
+ "@types/esrecurse": "^4.3.1",
1504
+ "@types/estree": "^1.0.8",
1505
+ "esrecurse": "^4.3.0",
1506
+ "estraverse": "^5.2.0"
1507
+ },
1508
+ "engines": {
1509
+ "node": "^20.19.0 || ^22.13.0 || >=24"
1510
+ },
1511
+ "funding": {
1512
+ "url": "https://opencollective.com/eslint"
1513
+ }
1514
+ },
1515
+ "node_modules/eslint-visitor-keys": {
1516
+ "version": "5.0.1",
1517
+ "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz",
1518
+ "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==",
1519
+ "dev": true,
1520
+ "license": "Apache-2.0",
1521
+ "engines": {
1522
+ "node": "^20.19.0 || ^22.13.0 || >=24"
1523
+ },
1524
+ "funding": {
1525
+ "url": "https://opencollective.com/eslint"
1526
+ }
1527
+ },
1528
+ "node_modules/espree": {
1529
+ "version": "11.2.0",
1530
+ "resolved": "https://registry.npmjs.org/espree/-/espree-11.2.0.tgz",
1531
+ "integrity": "sha512-7p3DrVEIopW1B1avAGLuCSh1jubc01H2JHc8B4qqGblmg5gI9yumBgACjWo4JlIc04ufug4xJ3SQI8HkS/Rgzw==",
1532
+ "dev": true,
1533
+ "license": "BSD-2-Clause",
1534
+ "dependencies": {
1535
+ "acorn": "^8.16.0",
1536
+ "acorn-jsx": "^5.3.2",
1537
+ "eslint-visitor-keys": "^5.0.1"
1538
+ },
1539
+ "engines": {
1540
+ "node": "^20.19.0 || ^22.13.0 || >=24"
1541
+ },
1542
+ "funding": {
1543
+ "url": "https://opencollective.com/eslint"
1544
+ }
1545
+ },
1546
+ "node_modules/esquery": {
1547
+ "version": "1.7.0",
1548
+ "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz",
1549
+ "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==",
1550
+ "dev": true,
1551
+ "license": "BSD-3-Clause",
1552
+ "dependencies": {
1553
+ "estraverse": "^5.1.0"
1554
+ },
1555
+ "engines": {
1556
+ "node": ">=0.10"
1557
+ }
1558
+ },
1559
+ "node_modules/esrecurse": {
1560
+ "version": "4.3.0",
1561
+ "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
1562
+ "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
1563
+ "dev": true,
1564
+ "license": "BSD-2-Clause",
1565
+ "dependencies": {
1566
+ "estraverse": "^5.2.0"
1567
+ },
1568
+ "engines": {
1569
+ "node": ">=4.0"
1570
+ }
1571
+ },
1572
+ "node_modules/estraverse": {
1573
+ "version": "5.3.0",
1574
+ "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
1575
+ "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
1576
+ "dev": true,
1577
+ "license": "BSD-2-Clause",
1578
+ "engines": {
1579
+ "node": ">=4.0"
1580
+ }
1581
+ },
1582
+ "node_modules/esutils": {
1583
+ "version": "2.0.3",
1584
+ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
1585
+ "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
1586
+ "dev": true,
1587
+ "license": "BSD-2-Clause",
1588
+ "engines": {
1589
+ "node": ">=0.10.0"
1590
+ }
1591
+ },
1592
+ "node_modules/fast-deep-equal": {
1593
+ "version": "3.1.3",
1594
+ "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
1595
+ "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
1596
+ "dev": true,
1597
+ "license": "MIT"
1598
+ },
1599
+ "node_modules/fast-json-stable-stringify": {
1600
+ "version": "2.1.0",
1601
+ "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
1602
+ "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
1603
+ "dev": true,
1604
+ "license": "MIT"
1605
+ },
1606
+ "node_modules/fast-levenshtein": {
1607
+ "version": "2.0.6",
1608
+ "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
1609
+ "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
1610
+ "dev": true,
1611
+ "license": "MIT"
1612
+ },
1613
+ "node_modules/fdir": {
1614
+ "version": "6.5.0",
1615
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
1616
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
1617
+ "dev": true,
1618
+ "license": "MIT",
1619
+ "engines": {
1620
+ "node": ">=12.0.0"
1621
+ },
1622
+ "peerDependencies": {
1623
+ "picomatch": "^3 || ^4"
1624
+ },
1625
+ "peerDependenciesMeta": {
1626
+ "picomatch": {
1627
+ "optional": true
1628
+ }
1629
+ }
1630
+ },
1631
+ "node_modules/file-entry-cache": {
1632
+ "version": "8.0.0",
1633
+ "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz",
1634
+ "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==",
1635
+ "dev": true,
1636
+ "license": "MIT",
1637
+ "dependencies": {
1638
+ "flat-cache": "^4.0.0"
1639
+ },
1640
+ "engines": {
1641
+ "node": ">=16.0.0"
1642
+ }
1643
+ },
1644
+ "node_modules/find-up": {
1645
+ "version": "5.0.0",
1646
+ "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
1647
+ "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
1648
+ "dev": true,
1649
+ "license": "MIT",
1650
+ "dependencies": {
1651
+ "locate-path": "^6.0.0",
1652
+ "path-exists": "^4.0.0"
1653
+ },
1654
+ "engines": {
1655
+ "node": ">=10"
1656
+ },
1657
+ "funding": {
1658
+ "url": "https://github.com/sponsors/sindresorhus"
1659
+ }
1660
+ },
1661
+ "node_modules/flat-cache": {
1662
+ "version": "4.0.1",
1663
+ "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz",
1664
+ "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==",
1665
+ "dev": true,
1666
+ "license": "MIT",
1667
+ "dependencies": {
1668
+ "flatted": "^3.2.9",
1669
+ "keyv": "^4.5.4"
1670
+ },
1671
+ "engines": {
1672
+ "node": ">=16"
1673
+ }
1674
+ },
1675
+ "node_modules/flatted": {
1676
+ "version": "3.4.2",
1677
+ "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz",
1678
+ "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==",
1679
+ "dev": true,
1680
+ "license": "ISC"
1681
+ },
1682
+ "node_modules/fsevents": {
1683
+ "version": "2.3.3",
1684
+ "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
1685
+ "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
1686
+ "dev": true,
1687
+ "hasInstallScript": true,
1688
+ "license": "MIT",
1689
+ "optional": true,
1690
+ "os": [
1691
+ "darwin"
1692
+ ],
1693
+ "engines": {
1694
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
1695
+ }
1696
+ },
1697
+ "node_modules/gensync": {
1698
+ "version": "1.0.0-beta.2",
1699
+ "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz",
1700
+ "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==",
1701
+ "dev": true,
1702
+ "license": "MIT",
1703
+ "engines": {
1704
+ "node": ">=6.9.0"
1705
+ }
1706
+ },
1707
+ "node_modules/glob-parent": {
1708
+ "version": "6.0.2",
1709
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
1710
+ "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
1711
+ "dev": true,
1712
+ "license": "ISC",
1713
+ "dependencies": {
1714
+ "is-glob": "^4.0.3"
1715
+ },
1716
+ "engines": {
1717
+ "node": ">=10.13.0"
1718
+ }
1719
+ },
1720
+ "node_modules/globals": {
1721
+ "version": "17.6.0",
1722
+ "resolved": "https://registry.npmjs.org/globals/-/globals-17.6.0.tgz",
1723
+ "integrity": "sha512-sepffkT8stwnIYbsMBpoCHJuJM5l98FUF2AnE07hfvE0m/qp3R586hw4jF4uadbhvg1ooIdzuu7CsfD2jzCaNA==",
1724
+ "dev": true,
1725
+ "license": "MIT",
1726
+ "engines": {
1727
+ "node": ">=18"
1728
+ },
1729
+ "funding": {
1730
+ "url": "https://github.com/sponsors/sindresorhus"
1731
+ }
1732
+ },
1733
+ "node_modules/graceful-fs": {
1734
+ "version": "4.2.11",
1735
+ "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz",
1736
+ "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==",
1737
+ "dev": true,
1738
+ "license": "ISC"
1739
+ },
1740
+ "node_modules/hermes-estree": {
1741
+ "version": "0.25.1",
1742
+ "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz",
1743
+ "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==",
1744
+ "dev": true,
1745
+ "license": "MIT"
1746
+ },
1747
+ "node_modules/hermes-parser": {
1748
+ "version": "0.25.1",
1749
+ "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz",
1750
+ "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==",
1751
+ "dev": true,
1752
+ "license": "MIT",
1753
+ "dependencies": {
1754
+ "hermes-estree": "0.25.1"
1755
+ }
1756
+ },
1757
+ "node_modules/ignore": {
1758
+ "version": "5.3.2",
1759
+ "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz",
1760
+ "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==",
1761
+ "dev": true,
1762
+ "license": "MIT",
1763
+ "engines": {
1764
+ "node": ">= 4"
1765
+ }
1766
+ },
1767
+ "node_modules/imurmurhash": {
1768
+ "version": "0.1.4",
1769
+ "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
1770
+ "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==",
1771
+ "dev": true,
1772
+ "license": "MIT",
1773
+ "engines": {
1774
+ "node": ">=0.8.19"
1775
+ }
1776
+ },
1777
+ "node_modules/is-extglob": {
1778
+ "version": "2.1.1",
1779
+ "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
1780
+ "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
1781
+ "dev": true,
1782
+ "license": "MIT",
1783
+ "engines": {
1784
+ "node": ">=0.10.0"
1785
+ }
1786
+ },
1787
+ "node_modules/is-glob": {
1788
+ "version": "4.0.3",
1789
+ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
1790
+ "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
1791
+ "dev": true,
1792
+ "license": "MIT",
1793
+ "dependencies": {
1794
+ "is-extglob": "^2.1.1"
1795
+ },
1796
+ "engines": {
1797
+ "node": ">=0.10.0"
1798
+ }
1799
+ },
1800
+ "node_modules/isexe": {
1801
+ "version": "2.0.0",
1802
+ "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
1803
+ "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
1804
+ "dev": true,
1805
+ "license": "ISC"
1806
+ },
1807
+ "node_modules/jiti": {
1808
+ "version": "2.7.0",
1809
+ "resolved": "https://registry.npmjs.org/jiti/-/jiti-2.7.0.tgz",
1810
+ "integrity": "sha512-AC/7JofJvZGrrneWNaEnJeOLUx+JlGt7tNa0wZiRPT4MY1wmfKjt2+6O2p2uz2+skll8OZZmJMNqeke7kKbNgQ==",
1811
+ "dev": true,
1812
+ "license": "MIT",
1813
+ "peer": true,
1814
+ "bin": {
1815
+ "jiti": "lib/jiti-cli.mjs"
1816
+ }
1817
+ },
1818
+ "node_modules/js-tokens": {
1819
+ "version": "4.0.0",
1820
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
1821
+ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
1822
+ "dev": true,
1823
+ "license": "MIT"
1824
+ },
1825
+ "node_modules/jsesc": {
1826
+ "version": "3.1.0",
1827
+ "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz",
1828
+ "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==",
1829
+ "dev": true,
1830
+ "license": "MIT",
1831
+ "bin": {
1832
+ "jsesc": "bin/jsesc"
1833
+ },
1834
+ "engines": {
1835
+ "node": ">=6"
1836
+ }
1837
+ },
1838
+ "node_modules/json-buffer": {
1839
+ "version": "3.0.1",
1840
+ "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
1841
+ "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
1842
+ "dev": true,
1843
+ "license": "MIT"
1844
+ },
1845
+ "node_modules/json-schema-traverse": {
1846
+ "version": "0.4.1",
1847
+ "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
1848
+ "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
1849
+ "dev": true,
1850
+ "license": "MIT"
1851
+ },
1852
+ "node_modules/json-stable-stringify-without-jsonify": {
1853
+ "version": "1.0.1",
1854
+ "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
1855
+ "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==",
1856
+ "dev": true,
1857
+ "license": "MIT"
1858
+ },
1859
+ "node_modules/json5": {
1860
+ "version": "2.2.3",
1861
+ "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
1862
+ "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
1863
+ "dev": true,
1864
+ "license": "MIT",
1865
+ "bin": {
1866
+ "json5": "lib/cli.js"
1867
+ },
1868
+ "engines": {
1869
+ "node": ">=6"
1870
+ }
1871
+ },
1872
+ "node_modules/keyv": {
1873
+ "version": "4.5.4",
1874
+ "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
1875
+ "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==",
1876
+ "dev": true,
1877
+ "license": "MIT",
1878
+ "dependencies": {
1879
+ "json-buffer": "3.0.1"
1880
+ }
1881
+ },
1882
+ "node_modules/levn": {
1883
+ "version": "0.4.1",
1884
+ "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz",
1885
+ "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==",
1886
+ "dev": true,
1887
+ "license": "MIT",
1888
+ "dependencies": {
1889
+ "prelude-ls": "^1.2.1",
1890
+ "type-check": "~0.4.0"
1891
+ },
1892
+ "engines": {
1893
+ "node": ">= 0.8.0"
1894
+ }
1895
+ },
1896
+ "node_modules/lightningcss": {
1897
+ "version": "1.32.0",
1898
+ "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz",
1899
+ "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==",
1900
+ "dev": true,
1901
+ "license": "MPL-2.0",
1902
+ "dependencies": {
1903
+ "detect-libc": "^2.0.3"
1904
+ },
1905
+ "engines": {
1906
+ "node": ">= 12.0.0"
1907
+ },
1908
+ "funding": {
1909
+ "type": "opencollective",
1910
+ "url": "https://opencollective.com/parcel"
1911
+ },
1912
+ "optionalDependencies": {
1913
+ "lightningcss-android-arm64": "1.32.0",
1914
+ "lightningcss-darwin-arm64": "1.32.0",
1915
+ "lightningcss-darwin-x64": "1.32.0",
1916
+ "lightningcss-freebsd-x64": "1.32.0",
1917
+ "lightningcss-linux-arm-gnueabihf": "1.32.0",
1918
+ "lightningcss-linux-arm64-gnu": "1.32.0",
1919
+ "lightningcss-linux-arm64-musl": "1.32.0",
1920
+ "lightningcss-linux-x64-gnu": "1.32.0",
1921
+ "lightningcss-linux-x64-musl": "1.32.0",
1922
+ "lightningcss-win32-arm64-msvc": "1.32.0",
1923
+ "lightningcss-win32-x64-msvc": "1.32.0"
1924
+ }
1925
+ },
1926
+ "node_modules/lightningcss-android-arm64": {
1927
+ "version": "1.32.0",
1928
+ "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz",
1929
+ "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==",
1930
+ "cpu": [
1931
+ "arm64"
1932
+ ],
1933
+ "dev": true,
1934
+ "license": "MPL-2.0",
1935
+ "optional": true,
1936
+ "os": [
1937
+ "android"
1938
+ ],
1939
+ "engines": {
1940
+ "node": ">= 12.0.0"
1941
+ },
1942
+ "funding": {
1943
+ "type": "opencollective",
1944
+ "url": "https://opencollective.com/parcel"
1945
+ }
1946
+ },
1947
+ "node_modules/lightningcss-darwin-arm64": {
1948
+ "version": "1.32.0",
1949
+ "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz",
1950
+ "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==",
1951
+ "cpu": [
1952
+ "arm64"
1953
+ ],
1954
+ "dev": true,
1955
+ "license": "MPL-2.0",
1956
+ "optional": true,
1957
+ "os": [
1958
+ "darwin"
1959
+ ],
1960
+ "engines": {
1961
+ "node": ">= 12.0.0"
1962
+ },
1963
+ "funding": {
1964
+ "type": "opencollective",
1965
+ "url": "https://opencollective.com/parcel"
1966
+ }
1967
+ },
1968
+ "node_modules/lightningcss-darwin-x64": {
1969
+ "version": "1.32.0",
1970
+ "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz",
1971
+ "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==",
1972
+ "cpu": [
1973
+ "x64"
1974
+ ],
1975
+ "dev": true,
1976
+ "license": "MPL-2.0",
1977
+ "optional": true,
1978
+ "os": [
1979
+ "darwin"
1980
+ ],
1981
+ "engines": {
1982
+ "node": ">= 12.0.0"
1983
+ },
1984
+ "funding": {
1985
+ "type": "opencollective",
1986
+ "url": "https://opencollective.com/parcel"
1987
+ }
1988
+ },
1989
+ "node_modules/lightningcss-freebsd-x64": {
1990
+ "version": "1.32.0",
1991
+ "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz",
1992
+ "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==",
1993
+ "cpu": [
1994
+ "x64"
1995
+ ],
1996
+ "dev": true,
1997
+ "license": "MPL-2.0",
1998
+ "optional": true,
1999
+ "os": [
2000
+ "freebsd"
2001
+ ],
2002
+ "engines": {
2003
+ "node": ">= 12.0.0"
2004
+ },
2005
+ "funding": {
2006
+ "type": "opencollective",
2007
+ "url": "https://opencollective.com/parcel"
2008
+ }
2009
+ },
2010
+ "node_modules/lightningcss-linux-arm-gnueabihf": {
2011
+ "version": "1.32.0",
2012
+ "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz",
2013
+ "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==",
2014
+ "cpu": [
2015
+ "arm"
2016
+ ],
2017
+ "dev": true,
2018
+ "license": "MPL-2.0",
2019
+ "optional": true,
2020
+ "os": [
2021
+ "linux"
2022
+ ],
2023
+ "engines": {
2024
+ "node": ">= 12.0.0"
2025
+ },
2026
+ "funding": {
2027
+ "type": "opencollective",
2028
+ "url": "https://opencollective.com/parcel"
2029
+ }
2030
+ },
2031
+ "node_modules/lightningcss-linux-arm64-gnu": {
2032
+ "version": "1.32.0",
2033
+ "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz",
2034
+ "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==",
2035
+ "cpu": [
2036
+ "arm64"
2037
+ ],
2038
+ "dev": true,
2039
+ "license": "MPL-2.0",
2040
+ "optional": true,
2041
+ "os": [
2042
+ "linux"
2043
+ ],
2044
+ "engines": {
2045
+ "node": ">= 12.0.0"
2046
+ },
2047
+ "funding": {
2048
+ "type": "opencollective",
2049
+ "url": "https://opencollective.com/parcel"
2050
+ }
2051
+ },
2052
+ "node_modules/lightningcss-linux-arm64-musl": {
2053
+ "version": "1.32.0",
2054
+ "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz",
2055
+ "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==",
2056
+ "cpu": [
2057
+ "arm64"
2058
+ ],
2059
+ "dev": true,
2060
+ "license": "MPL-2.0",
2061
+ "optional": true,
2062
+ "os": [
2063
+ "linux"
2064
+ ],
2065
+ "engines": {
2066
+ "node": ">= 12.0.0"
2067
+ },
2068
+ "funding": {
2069
+ "type": "opencollective",
2070
+ "url": "https://opencollective.com/parcel"
2071
+ }
2072
+ },
2073
+ "node_modules/lightningcss-linux-x64-gnu": {
2074
+ "version": "1.32.0",
2075
+ "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz",
2076
+ "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==",
2077
+ "cpu": [
2078
+ "x64"
2079
+ ],
2080
+ "dev": true,
2081
+ "license": "MPL-2.0",
2082
+ "optional": true,
2083
+ "os": [
2084
+ "linux"
2085
+ ],
2086
+ "engines": {
2087
+ "node": ">= 12.0.0"
2088
+ },
2089
+ "funding": {
2090
+ "type": "opencollective",
2091
+ "url": "https://opencollective.com/parcel"
2092
+ }
2093
+ },
2094
+ "node_modules/lightningcss-linux-x64-musl": {
2095
+ "version": "1.32.0",
2096
+ "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz",
2097
+ "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==",
2098
+ "cpu": [
2099
+ "x64"
2100
+ ],
2101
+ "dev": true,
2102
+ "license": "MPL-2.0",
2103
+ "optional": true,
2104
+ "os": [
2105
+ "linux"
2106
+ ],
2107
+ "engines": {
2108
+ "node": ">= 12.0.0"
2109
+ },
2110
+ "funding": {
2111
+ "type": "opencollective",
2112
+ "url": "https://opencollective.com/parcel"
2113
+ }
2114
+ },
2115
+ "node_modules/lightningcss-win32-arm64-msvc": {
2116
+ "version": "1.32.0",
2117
+ "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz",
2118
+ "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==",
2119
+ "cpu": [
2120
+ "arm64"
2121
+ ],
2122
+ "dev": true,
2123
+ "license": "MPL-2.0",
2124
+ "optional": true,
2125
+ "os": [
2126
+ "win32"
2127
+ ],
2128
+ "engines": {
2129
+ "node": ">= 12.0.0"
2130
+ },
2131
+ "funding": {
2132
+ "type": "opencollective",
2133
+ "url": "https://opencollective.com/parcel"
2134
+ }
2135
+ },
2136
+ "node_modules/lightningcss-win32-x64-msvc": {
2137
+ "version": "1.32.0",
2138
+ "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz",
2139
+ "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==",
2140
+ "cpu": [
2141
+ "x64"
2142
+ ],
2143
+ "dev": true,
2144
+ "license": "MPL-2.0",
2145
+ "optional": true,
2146
+ "os": [
2147
+ "win32"
2148
+ ],
2149
+ "engines": {
2150
+ "node": ">= 12.0.0"
2151
+ },
2152
+ "funding": {
2153
+ "type": "opencollective",
2154
+ "url": "https://opencollective.com/parcel"
2155
+ }
2156
+ },
2157
+ "node_modules/locate-path": {
2158
+ "version": "6.0.0",
2159
+ "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
2160
+ "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
2161
+ "dev": true,
2162
+ "license": "MIT",
2163
+ "dependencies": {
2164
+ "p-locate": "^5.0.0"
2165
+ },
2166
+ "engines": {
2167
+ "node": ">=10"
2168
+ },
2169
+ "funding": {
2170
+ "url": "https://github.com/sponsors/sindresorhus"
2171
+ }
2172
+ },
2173
+ "node_modules/lru-cache": {
2174
+ "version": "5.1.1",
2175
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
2176
+ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
2177
+ "dev": true,
2178
+ "license": "ISC",
2179
+ "dependencies": {
2180
+ "yallist": "^3.0.2"
2181
+ }
2182
+ },
2183
+ "node_modules/magic-string": {
2184
+ "version": "0.30.21",
2185
+ "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.21.tgz",
2186
+ "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==",
2187
+ "dev": true,
2188
+ "license": "MIT",
2189
+ "dependencies": {
2190
+ "@jridgewell/sourcemap-codec": "^1.5.5"
2191
+ }
2192
+ },
2193
+ "node_modules/minimatch": {
2194
+ "version": "10.2.5",
2195
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz",
2196
+ "integrity": "sha512-MULkVLfKGYDFYejP07QOurDLLQpcjk7Fw+7jXS2R2czRQzR56yHRveU5NDJEOviH+hETZKSkIk5c+T23GjFUMg==",
2197
+ "dev": true,
2198
+ "license": "BlueOak-1.0.0",
2199
+ "dependencies": {
2200
+ "brace-expansion": "^5.0.5"
2201
+ },
2202
+ "engines": {
2203
+ "node": "18 || 20 || >=22"
2204
+ },
2205
+ "funding": {
2206
+ "url": "https://github.com/sponsors/isaacs"
2207
+ }
2208
+ },
2209
+ "node_modules/ms": {
2210
+ "version": "2.1.3",
2211
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
2212
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
2213
+ "dev": true,
2214
+ "license": "MIT"
2215
+ },
2216
+ "node_modules/nanoid": {
2217
+ "version": "3.3.12",
2218
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz",
2219
+ "integrity": "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ==",
2220
+ "dev": true,
2221
+ "funding": [
2222
+ {
2223
+ "type": "github",
2224
+ "url": "https://github.com/sponsors/ai"
2225
+ }
2226
+ ],
2227
+ "license": "MIT",
2228
+ "bin": {
2229
+ "nanoid": "bin/nanoid.cjs"
2230
+ },
2231
+ "engines": {
2232
+ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
2233
+ }
2234
+ },
2235
+ "node_modules/natural-compare": {
2236
+ "version": "1.4.0",
2237
+ "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
2238
+ "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==",
2239
+ "dev": true,
2240
+ "license": "MIT"
2241
+ },
2242
+ "node_modules/node-releases": {
2243
+ "version": "2.0.44",
2244
+ "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.44.tgz",
2245
+ "integrity": "sha512-5WUyunoPMsvvEhS8AxHtRzP+oA8UCkJ7YRxatWKjngndhDGLiqEVAQKWjFAiAiuL8zMRGzGSJxFnLetoa43qGQ==",
2246
+ "dev": true,
2247
+ "license": "MIT"
2248
+ },
2249
+ "node_modules/optionator": {
2250
+ "version": "0.9.4",
2251
+ "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz",
2252
+ "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==",
2253
+ "dev": true,
2254
+ "license": "MIT",
2255
+ "dependencies": {
2256
+ "deep-is": "^0.1.3",
2257
+ "fast-levenshtein": "^2.0.6",
2258
+ "levn": "^0.4.1",
2259
+ "prelude-ls": "^1.2.1",
2260
+ "type-check": "^0.4.0",
2261
+ "word-wrap": "^1.2.5"
2262
+ },
2263
+ "engines": {
2264
+ "node": ">= 0.8.0"
2265
+ }
2266
+ },
2267
+ "node_modules/p-limit": {
2268
+ "version": "3.1.0",
2269
+ "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
2270
+ "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
2271
+ "dev": true,
2272
+ "license": "MIT",
2273
+ "dependencies": {
2274
+ "yocto-queue": "^0.1.0"
2275
+ },
2276
+ "engines": {
2277
+ "node": ">=10"
2278
+ },
2279
+ "funding": {
2280
+ "url": "https://github.com/sponsors/sindresorhus"
2281
+ }
2282
+ },
2283
+ "node_modules/p-locate": {
2284
+ "version": "5.0.0",
2285
+ "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
2286
+ "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
2287
+ "dev": true,
2288
+ "license": "MIT",
2289
+ "dependencies": {
2290
+ "p-limit": "^3.0.2"
2291
+ },
2292
+ "engines": {
2293
+ "node": ">=10"
2294
+ },
2295
+ "funding": {
2296
+ "url": "https://github.com/sponsors/sindresorhus"
2297
+ }
2298
+ },
2299
+ "node_modules/path-exists": {
2300
+ "version": "4.0.0",
2301
+ "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
2302
+ "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
2303
+ "dev": true,
2304
+ "license": "MIT",
2305
+ "engines": {
2306
+ "node": ">=8"
2307
+ }
2308
+ },
2309
+ "node_modules/path-key": {
2310
+ "version": "3.1.1",
2311
+ "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
2312
+ "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
2313
+ "dev": true,
2314
+ "license": "MIT",
2315
+ "engines": {
2316
+ "node": ">=8"
2317
+ }
2318
+ },
2319
+ "node_modules/picocolors": {
2320
+ "version": "1.1.1",
2321
+ "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
2322
+ "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
2323
+ "dev": true,
2324
+ "license": "ISC"
2325
+ },
2326
+ "node_modules/picomatch": {
2327
+ "version": "4.0.4",
2328
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz",
2329
+ "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==",
2330
+ "dev": true,
2331
+ "license": "MIT",
2332
+ "peer": true,
2333
+ "engines": {
2334
+ "node": ">=12"
2335
+ },
2336
+ "funding": {
2337
+ "url": "https://github.com/sponsors/jonschlinkert"
2338
+ }
2339
+ },
2340
+ "node_modules/postcss": {
2341
+ "version": "8.5.14",
2342
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.14.tgz",
2343
+ "integrity": "sha512-SoSL4+OSEtR99LHFZQiJLkT59C5B1amGO1NzTwj7TT1qCUgUO6hxOvzkOYxD+vMrXBM3XJIKzokoERdqQq/Zmg==",
2344
+ "dev": true,
2345
+ "funding": [
2346
+ {
2347
+ "type": "opencollective",
2348
+ "url": "https://opencollective.com/postcss/"
2349
+ },
2350
+ {
2351
+ "type": "tidelift",
2352
+ "url": "https://tidelift.com/funding/github/npm/postcss"
2353
+ },
2354
+ {
2355
+ "type": "github",
2356
+ "url": "https://github.com/sponsors/ai"
2357
+ }
2358
+ ],
2359
+ "license": "MIT",
2360
+ "dependencies": {
2361
+ "nanoid": "^3.3.11",
2362
+ "picocolors": "^1.1.1",
2363
+ "source-map-js": "^1.2.1"
2364
+ },
2365
+ "engines": {
2366
+ "node": "^10 || ^12 || >=14"
2367
+ }
2368
+ },
2369
+ "node_modules/prelude-ls": {
2370
+ "version": "1.2.1",
2371
+ "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
2372
+ "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
2373
+ "dev": true,
2374
+ "license": "MIT",
2375
+ "engines": {
2376
+ "node": ">= 0.8.0"
2377
+ }
2378
+ },
2379
+ "node_modules/punycode": {
2380
+ "version": "2.3.1",
2381
+ "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
2382
+ "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
2383
+ "dev": true,
2384
+ "license": "MIT",
2385
+ "engines": {
2386
+ "node": ">=6"
2387
+ }
2388
+ },
2389
+ "node_modules/react": {
2390
+ "version": "19.2.6",
2391
+ "resolved": "https://registry.npmjs.org/react/-/react-19.2.6.tgz",
2392
+ "integrity": "sha512-sfWGGfavi0xr8Pg0sVsyHMAOziVYKgPLNrS7ig+ivMNb3wbCBw3KxtflsGBAwD3gYQlE/AEZsTLgToRrSCjb0Q==",
2393
+ "license": "MIT",
2394
+ "peer": true,
2395
+ "engines": {
2396
+ "node": ">=0.10.0"
2397
+ }
2398
+ },
2399
+ "node_modules/react-dom": {
2400
+ "version": "19.2.6",
2401
+ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.6.tgz",
2402
+ "integrity": "sha512-0prMI+hvBbPjsWnxDLxlCGyM8PN6UuWjEUCYmZhO67xIV9Xasa/r/vDnq+Xyq4Lo27g8QSbO5YzARu0D1Sps3g==",
2403
+ "license": "MIT",
2404
+ "dependencies": {
2405
+ "scheduler": "^0.27.0"
2406
+ },
2407
+ "peerDependencies": {
2408
+ "react": "^19.2.6"
2409
+ }
2410
+ },
2411
+ "node_modules/rolldown": {
2412
+ "version": "1.0.0",
2413
+ "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.0.tgz",
2414
+ "integrity": "sha512-yD986aXDESFGS95spT1LAv0jssywP4npMEjmMHyN2/5+eE8qQJUype2AaKkRiLgBgyD0LFlubwAht7VmY8rGoA==",
2415
+ "dev": true,
2416
+ "license": "MIT",
2417
+ "dependencies": {
2418
+ "@oxc-project/types": "=0.129.0",
2419
+ "@rolldown/pluginutils": "1.0.0"
2420
+ },
2421
+ "bin": {
2422
+ "rolldown": "bin/cli.mjs"
2423
+ },
2424
+ "engines": {
2425
+ "node": "^20.19.0 || >=22.12.0"
2426
+ },
2427
+ "optionalDependencies": {
2428
+ "@rolldown/binding-android-arm64": "1.0.0",
2429
+ "@rolldown/binding-darwin-arm64": "1.0.0",
2430
+ "@rolldown/binding-darwin-x64": "1.0.0",
2431
+ "@rolldown/binding-freebsd-x64": "1.0.0",
2432
+ "@rolldown/binding-linux-arm-gnueabihf": "1.0.0",
2433
+ "@rolldown/binding-linux-arm64-gnu": "1.0.0",
2434
+ "@rolldown/binding-linux-arm64-musl": "1.0.0",
2435
+ "@rolldown/binding-linux-ppc64-gnu": "1.0.0",
2436
+ "@rolldown/binding-linux-s390x-gnu": "1.0.0",
2437
+ "@rolldown/binding-linux-x64-gnu": "1.0.0",
2438
+ "@rolldown/binding-linux-x64-musl": "1.0.0",
2439
+ "@rolldown/binding-openharmony-arm64": "1.0.0",
2440
+ "@rolldown/binding-wasm32-wasi": "1.0.0",
2441
+ "@rolldown/binding-win32-arm64-msvc": "1.0.0",
2442
+ "@rolldown/binding-win32-x64-msvc": "1.0.0"
2443
+ }
2444
+ },
2445
+ "node_modules/rolldown/node_modules/@rolldown/pluginutils": {
2446
+ "version": "1.0.0",
2447
+ "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0.tgz",
2448
+ "integrity": "sha512-aKs/3GSWyV0mrhNmt/96/Z3yczC3yvrzYATCiCXQebBsGyYzjNdUphRVLeJQ67ySKVXRfMxt2lm12pmXvbPFQQ==",
2449
+ "dev": true,
2450
+ "license": "MIT"
2451
+ },
2452
+ "node_modules/scheduler": {
2453
+ "version": "0.27.0",
2454
+ "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
2455
+ "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
2456
+ "license": "MIT"
2457
+ },
2458
+ "node_modules/semver": {
2459
+ "version": "6.3.1",
2460
+ "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
2461
+ "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
2462
+ "dev": true,
2463
+ "license": "ISC",
2464
+ "bin": {
2465
+ "semver": "bin/semver.js"
2466
+ }
2467
+ },
2468
+ "node_modules/shebang-command": {
2469
+ "version": "2.0.0",
2470
+ "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
2471
+ "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
2472
+ "dev": true,
2473
+ "license": "MIT",
2474
+ "dependencies": {
2475
+ "shebang-regex": "^3.0.0"
2476
+ },
2477
+ "engines": {
2478
+ "node": ">=8"
2479
+ }
2480
+ },
2481
+ "node_modules/shebang-regex": {
2482
+ "version": "3.0.0",
2483
+ "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
2484
+ "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
2485
+ "dev": true,
2486
+ "license": "MIT",
2487
+ "engines": {
2488
+ "node": ">=8"
2489
+ }
2490
+ },
2491
+ "node_modules/source-map-js": {
2492
+ "version": "1.2.1",
2493
+ "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
2494
+ "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
2495
+ "dev": true,
2496
+ "license": "BSD-3-Clause",
2497
+ "engines": {
2498
+ "node": ">=0.10.0"
2499
+ }
2500
+ },
2501
+ "node_modules/tailwindcss": {
2502
+ "version": "4.3.0",
2503
+ "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-4.3.0.tgz",
2504
+ "integrity": "sha512-y6nxMGB1nMW9R6k96e5gdIFzcfL/gTJRNaqGes1YvkLnPVXzWgbqFF2yLC0T8G774n24cx3Pe8XrKoniCOAH+Q==",
2505
+ "dev": true,
2506
+ "license": "MIT"
2507
+ },
2508
+ "node_modules/tapable": {
2509
+ "version": "2.3.3",
2510
+ "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.3.3.tgz",
2511
+ "integrity": "sha512-uxc/zpqFg6x7C8vOE7lh6Lbda8eEL9zmVm/PLeTPBRhh1xCgdWaQ+J1CUieGpIfm2HdtsUpRv+HshiasBMcc6A==",
2512
+ "dev": true,
2513
+ "license": "MIT",
2514
+ "engines": {
2515
+ "node": ">=6"
2516
+ },
2517
+ "funding": {
2518
+ "type": "opencollective",
2519
+ "url": "https://opencollective.com/webpack"
2520
+ }
2521
+ },
2522
+ "node_modules/tinyglobby": {
2523
+ "version": "0.2.16",
2524
+ "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz",
2525
+ "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==",
2526
+ "dev": true,
2527
+ "license": "MIT",
2528
+ "dependencies": {
2529
+ "fdir": "^6.5.0",
2530
+ "picomatch": "^4.0.4"
2531
+ },
2532
+ "engines": {
2533
+ "node": ">=12.0.0"
2534
+ },
2535
+ "funding": {
2536
+ "url": "https://github.com/sponsors/SuperchupuDev"
2537
+ }
2538
+ },
2539
+ "node_modules/tslib": {
2540
+ "version": "2.8.1",
2541
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
2542
+ "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==",
2543
+ "dev": true,
2544
+ "license": "0BSD",
2545
+ "optional": true
2546
+ },
2547
+ "node_modules/type-check": {
2548
+ "version": "0.4.0",
2549
+ "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz",
2550
+ "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==",
2551
+ "dev": true,
2552
+ "license": "MIT",
2553
+ "dependencies": {
2554
+ "prelude-ls": "^1.2.1"
2555
+ },
2556
+ "engines": {
2557
+ "node": ">= 0.8.0"
2558
+ }
2559
+ },
2560
+ "node_modules/update-browserslist-db": {
2561
+ "version": "1.2.3",
2562
+ "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz",
2563
+ "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==",
2564
+ "dev": true,
2565
+ "funding": [
2566
+ {
2567
+ "type": "opencollective",
2568
+ "url": "https://opencollective.com/browserslist"
2569
+ },
2570
+ {
2571
+ "type": "tidelift",
2572
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
2573
+ },
2574
+ {
2575
+ "type": "github",
2576
+ "url": "https://github.com/sponsors/ai"
2577
+ }
2578
+ ],
2579
+ "license": "MIT",
2580
+ "dependencies": {
2581
+ "escalade": "^3.2.0",
2582
+ "picocolors": "^1.1.1"
2583
+ },
2584
+ "bin": {
2585
+ "update-browserslist-db": "cli.js"
2586
+ },
2587
+ "peerDependencies": {
2588
+ "browserslist": ">= 4.21.0"
2589
+ }
2590
+ },
2591
+ "node_modules/uri-js": {
2592
+ "version": "4.4.1",
2593
+ "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
2594
+ "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
2595
+ "dev": true,
2596
+ "license": "BSD-2-Clause",
2597
+ "dependencies": {
2598
+ "punycode": "^2.1.0"
2599
+ }
2600
+ },
2601
+ "node_modules/vite": {
2602
+ "version": "8.0.12",
2603
+ "resolved": "https://registry.npmjs.org/vite/-/vite-8.0.12.tgz",
2604
+ "integrity": "sha512-w2dDofOWv2QB09ZITZBsvKTVAlYvPR4IAmrY/v0ir9KvLs0xybR7i48wxhM1/oyBWO34wPns+bPGw5ZrZqDpZg==",
2605
+ "dev": true,
2606
+ "license": "MIT",
2607
+ "peer": true,
2608
+ "dependencies": {
2609
+ "lightningcss": "^1.32.0",
2610
+ "picomatch": "^4.0.4",
2611
+ "postcss": "^8.5.14",
2612
+ "rolldown": "1.0.0",
2613
+ "tinyglobby": "^0.2.16"
2614
+ },
2615
+ "bin": {
2616
+ "vite": "bin/vite.js"
2617
+ },
2618
+ "engines": {
2619
+ "node": "^20.19.0 || >=22.12.0"
2620
+ },
2621
+ "funding": {
2622
+ "url": "https://github.com/vitejs/vite?sponsor=1"
2623
+ },
2624
+ "optionalDependencies": {
2625
+ "fsevents": "~2.3.3"
2626
+ },
2627
+ "peerDependencies": {
2628
+ "@types/node": "^20.19.0 || >=22.12.0",
2629
+ "@vitejs/devtools": "^0.1.18",
2630
+ "esbuild": "^0.27.0 || ^0.28.0",
2631
+ "jiti": ">=1.21.0",
2632
+ "less": "^4.0.0",
2633
+ "sass": "^1.70.0",
2634
+ "sass-embedded": "^1.70.0",
2635
+ "stylus": ">=0.54.8",
2636
+ "sugarss": "^5.0.0",
2637
+ "terser": "^5.16.0",
2638
+ "tsx": "^4.8.1",
2639
+ "yaml": "^2.4.2"
2640
+ },
2641
+ "peerDependenciesMeta": {
2642
+ "@types/node": {
2643
+ "optional": true
2644
+ },
2645
+ "@vitejs/devtools": {
2646
+ "optional": true
2647
+ },
2648
+ "esbuild": {
2649
+ "optional": true
2650
+ },
2651
+ "jiti": {
2652
+ "optional": true
2653
+ },
2654
+ "less": {
2655
+ "optional": true
2656
+ },
2657
+ "sass": {
2658
+ "optional": true
2659
+ },
2660
+ "sass-embedded": {
2661
+ "optional": true
2662
+ },
2663
+ "stylus": {
2664
+ "optional": true
2665
+ },
2666
+ "sugarss": {
2667
+ "optional": true
2668
+ },
2669
+ "terser": {
2670
+ "optional": true
2671
+ },
2672
+ "tsx": {
2673
+ "optional": true
2674
+ },
2675
+ "yaml": {
2676
+ "optional": true
2677
+ }
2678
+ }
2679
+ },
2680
+ "node_modules/which": {
2681
+ "version": "2.0.2",
2682
+ "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
2683
+ "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
2684
+ "dev": true,
2685
+ "license": "ISC",
2686
+ "dependencies": {
2687
+ "isexe": "^2.0.0"
2688
+ },
2689
+ "bin": {
2690
+ "node-which": "bin/node-which"
2691
+ },
2692
+ "engines": {
2693
+ "node": ">= 8"
2694
+ }
2695
+ },
2696
+ "node_modules/word-wrap": {
2697
+ "version": "1.2.5",
2698
+ "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz",
2699
+ "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==",
2700
+ "dev": true,
2701
+ "license": "MIT",
2702
+ "engines": {
2703
+ "node": ">=0.10.0"
2704
+ }
2705
+ },
2706
+ "node_modules/yallist": {
2707
+ "version": "3.1.1",
2708
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
2709
+ "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
2710
+ "dev": true,
2711
+ "license": "ISC"
2712
+ },
2713
+ "node_modules/yocto-queue": {
2714
+ "version": "0.1.0",
2715
+ "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
2716
+ "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
2717
+ "dev": true,
2718
+ "license": "MIT",
2719
+ "engines": {
2720
+ "node": ">=10"
2721
+ },
2722
+ "funding": {
2723
+ "url": "https://github.com/sponsors/sindresorhus"
2724
+ }
2725
+ },
2726
+ "node_modules/zod": {
2727
+ "version": "4.4.3",
2728
+ "resolved": "https://registry.npmjs.org/zod/-/zod-4.4.3.tgz",
2729
+ "integrity": "sha512-ytENFjIJFl2UwYglde2jchW2Hwm4GJFLDiSXWdTrJQBIN9Fcyp7n4DhxJEiWNAJMV1/BqWfW/kkg71UDcHJyTQ==",
2730
+ "dev": true,
2731
+ "license": "MIT",
2732
+ "peer": true,
2733
+ "funding": {
2734
+ "url": "https://github.com/sponsors/colinhacks"
2735
+ }
2736
+ },
2737
+ "node_modules/zod-validation-error": {
2738
+ "version": "4.0.2",
2739
+ "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz",
2740
+ "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==",
2741
+ "dev": true,
2742
+ "license": "MIT",
2743
+ "engines": {
2744
+ "node": ">=18.0.0"
2745
+ },
2746
+ "peerDependencies": {
2747
+ "zod": "^3.25.0 || ^4.0.0"
2748
+ }
2749
+ }
2750
+ }
2751
+ }
frontend/package.json ADDED
@@ -0,0 +1,29 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "name": "frontend",
3
+ "private": true,
4
+ "version": "0.0.0",
5
+ "type": "module",
6
+ "scripts": {
7
+ "dev": "vite",
8
+ "build": "vite build",
9
+ "lint": "eslint .",
10
+ "preview": "vite preview"
11
+ },
12
+ "dependencies": {
13
+ "react": "^19.2.6",
14
+ "react-dom": "^19.2.6"
15
+ },
16
+ "devDependencies": {
17
+ "@eslint/js": "^10.0.1",
18
+ "@tailwindcss/vite": "^4.3.0",
19
+ "@types/react": "^19.2.14",
20
+ "@types/react-dom": "^19.2.3",
21
+ "@vitejs/plugin-react": "^6.0.1",
22
+ "eslint": "^10.3.0",
23
+ "eslint-plugin-react-hooks": "^7.1.1",
24
+ "eslint-plugin-react-refresh": "^0.5.2",
25
+ "globals": "^17.6.0",
26
+ "tailwindcss": "^4.3.0",
27
+ "vite": "^8.0.12"
28
+ }
29
+ }
frontend/public/favicon.svg ADDED
frontend/public/icons.svg ADDED
frontend/src/App.css ADDED
@@ -0,0 +1,184 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ .counter {
2
+ font-size: 16px;
3
+ padding: 5px 10px;
4
+ border-radius: 5px;
5
+ color: var(--accent);
6
+ background: var(--accent-bg);
7
+ border: 2px solid transparent;
8
+ transition: border-color 0.3s;
9
+ margin-bottom: 24px;
10
+
11
+ &:hover {
12
+ border-color: var(--accent-border);
13
+ }
14
+ &:focus-visible {
15
+ outline: 2px solid var(--accent);
16
+ outline-offset: 2px;
17
+ }
18
+ }
19
+
20
+ .hero {
21
+ position: relative;
22
+
23
+ .base,
24
+ .framework,
25
+ .vite {
26
+ inset-inline: 0;
27
+ margin: 0 auto;
28
+ }
29
+
30
+ .base {
31
+ width: 170px;
32
+ position: relative;
33
+ z-index: 0;
34
+ }
35
+
36
+ .framework,
37
+ .vite {
38
+ position: absolute;
39
+ }
40
+
41
+ .framework {
42
+ z-index: 1;
43
+ top: 34px;
44
+ height: 28px;
45
+ transform: perspective(2000px) rotateZ(300deg) rotateX(44deg) rotateY(39deg)
46
+ scale(1.4);
47
+ }
48
+
49
+ .vite {
50
+ z-index: 0;
51
+ top: 107px;
52
+ height: 26px;
53
+ width: auto;
54
+ transform: perspective(2000px) rotateZ(300deg) rotateX(40deg) rotateY(39deg)
55
+ scale(0.8);
56
+ }
57
+ }
58
+
59
+ #center {
60
+ display: flex;
61
+ flex-direction: column;
62
+ gap: 25px;
63
+ place-content: center;
64
+ place-items: center;
65
+ flex-grow: 1;
66
+
67
+ @media (max-width: 1024px) {
68
+ padding: 32px 20px 24px;
69
+ gap: 18px;
70
+ }
71
+ }
72
+
73
+ #next-steps {
74
+ display: flex;
75
+ border-top: 1px solid var(--border);
76
+ text-align: left;
77
+
78
+ & > div {
79
+ flex: 1 1 0;
80
+ padding: 32px;
81
+ @media (max-width: 1024px) {
82
+ padding: 24px 20px;
83
+ }
84
+ }
85
+
86
+ .icon {
87
+ margin-bottom: 16px;
88
+ width: 22px;
89
+ height: 22px;
90
+ }
91
+
92
+ @media (max-width: 1024px) {
93
+ flex-direction: column;
94
+ text-align: center;
95
+ }
96
+ }
97
+
98
+ #docs {
99
+ border-right: 1px solid var(--border);
100
+
101
+ @media (max-width: 1024px) {
102
+ border-right: none;
103
+ border-bottom: 1px solid var(--border);
104
+ }
105
+ }
106
+
107
+ #next-steps ul {
108
+ list-style: none;
109
+ padding: 0;
110
+ display: flex;
111
+ gap: 8px;
112
+ margin: 32px 0 0;
113
+
114
+ .logo {
115
+ height: 18px;
116
+ }
117
+
118
+ a {
119
+ color: var(--text-h);
120
+ font-size: 16px;
121
+ border-radius: 6px;
122
+ background: var(--social-bg);
123
+ display: flex;
124
+ padding: 6px 12px;
125
+ align-items: center;
126
+ gap: 8px;
127
+ text-decoration: none;
128
+ transition: box-shadow 0.3s;
129
+
130
+ &:hover {
131
+ box-shadow: var(--shadow);
132
+ }
133
+ .button-icon {
134
+ height: 18px;
135
+ width: 18px;
136
+ }
137
+ }
138
+
139
+ @media (max-width: 1024px) {
140
+ margin-top: 20px;
141
+ flex-wrap: wrap;
142
+ justify-content: center;
143
+
144
+ li {
145
+ flex: 1 1 calc(50% - 8px);
146
+ }
147
+
148
+ a {
149
+ width: 100%;
150
+ justify-content: center;
151
+ box-sizing: border-box;
152
+ }
153
+ }
154
+ }
155
+
156
+ #spacer {
157
+ height: 88px;
158
+ border-top: 1px solid var(--border);
159
+ @media (max-width: 1024px) {
160
+ height: 48px;
161
+ }
162
+ }
163
+
164
+ .ticks {
165
+ position: relative;
166
+ width: 100%;
167
+
168
+ &::before,
169
+ &::after {
170
+ content: "";
171
+ position: absolute;
172
+ top: -4.5px;
173
+ border: 5px solid transparent;
174
+ }
175
+
176
+ &::before {
177
+ left: 0;
178
+ border-left-color: var(--border);
179
+ }
180
+ &::after {
181
+ right: 0;
182
+ border-right-color: var(--border);
183
+ }
184
+ }
frontend/src/App.jsx ADDED
@@ -0,0 +1,168 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import { useEffect, useMemo, useState } from "react";
2
+ import Header from "./components/Header";
3
+ import UploadZone from "./components/UploadZone";
4
+ import ImagePreview from "./components/ImagePreview";
5
+ import CaptionResult from "./components/CaptionResult";
6
+ import ErrorBanner from "./components/ErrorBanner";
7
+ import Spinner from "./components/Spinner";
8
+ import { captionImage } from "./services/api";
9
+
10
+ export default function App() {
11
+ const [file, setFile] = useState(null);
12
+ const [result, setResult] = useState(null);
13
+ const [error, setError] = useState(null);
14
+ const [loading, setLoading] = useState(false);
15
+
16
+ const previewUrl = useMemo(
17
+ () => (file ? URL.createObjectURL(file) : null),
18
+ [file],
19
+ );
20
+
21
+ useEffect(() => {
22
+ if (!previewUrl) return;
23
+ return () => URL.revokeObjectURL(previewUrl);
24
+ }, [previewUrl]);
25
+
26
+ const handleFileSelected = (nextFile, validationError) => {
27
+ setResult(null);
28
+ if (validationError) {
29
+ setFile(null);
30
+ setError(validationError);
31
+ return;
32
+ }
33
+ setError(null);
34
+ setFile(nextFile);
35
+ };
36
+
37
+ const handleClear = () => {
38
+ setFile(null);
39
+ setResult(null);
40
+ setError(null);
41
+ };
42
+
43
+ const handleGenerate = async () => {
44
+ if (!file || loading) return;
45
+ setLoading(true);
46
+ setError(null);
47
+ setResult(null);
48
+ try {
49
+ const data = await captionImage(file);
50
+ setResult(data);
51
+ } catch (err) {
52
+ setError(err?.message || "Caption generation failed.");
53
+ } finally {
54
+ setLoading(false);
55
+ }
56
+ };
57
+
58
+ return (
59
+ <div className="min-h-screen bg-[#0a0a0f] text-white antialiased">
60
+ <div className="pointer-events-none fixed inset-0 overflow-hidden">
61
+ <div className="absolute -top-40 -left-40 w-[480px] h-[480px] rounded-full bg-violet-600/20 blur-[120px]" />
62
+ <div className="absolute top-1/3 -right-40 w-[520px] h-[520px] rounded-full bg-fuchsia-600/10 blur-[140px]" />
63
+ <div className="absolute bottom-0 left-1/3 w-[420px] h-[420px] rounded-full bg-indigo-600/10 blur-[120px]" />
64
+ </div>
65
+
66
+ <div className="relative">
67
+ <Header />
68
+
69
+ <main className="max-w-6xl mx-auto px-6 py-10 md:py-16">
70
+ <section className="mb-10 md:mb-14">
71
+ <h1 className="text-3xl md:text-5xl font-semibold tracking-tight leading-tight">
72
+ Describe any image{" "}
73
+ <span className="bg-gradient-to-r from-violet-300 via-fuchsia-300 to-indigo-300 bg-clip-text text-transparent">
74
+ in natural language
75
+ </span>
76
+ </h1>
77
+ <p className="text-white/50 mt-3 max-w-2xl">
78
+ Upload a photo and let the model generate a concise caption.
79
+ Powered by a vision-encoder/text-decoder pipeline served over
80
+ FastAPI.
81
+ </p>
82
+ </section>
83
+
84
+ <div className="grid grid-cols-1 lg:grid-cols-5 gap-6">
85
+ <div className="lg:col-span-3 space-y-5">
86
+ {file ? (
87
+ <ImagePreview
88
+ file={file}
89
+ previewUrl={previewUrl}
90
+ onClear={handleClear}
91
+ disabled={loading}
92
+ />
93
+ ) : (
94
+ <UploadZone
95
+ onFileSelected={handleFileSelected}
96
+ disabled={loading}
97
+ />
98
+ )}
99
+
100
+ <ErrorBanner message={error} onDismiss={() => setError(null)} />
101
+ </div>
102
+
103
+ <div className="lg:col-span-2 space-y-5">
104
+ <div className="rounded-2xl border border-white/10 bg-white/[0.02] p-5">
105
+ <div className="flex items-center justify-between mb-4">
106
+ <h2 className="text-sm font-medium text-white/80 uppercase tracking-wider">
107
+ Inference
108
+ </h2>
109
+ {loading && (
110
+ <span className="text-xs text-white/40">running…</span>
111
+ )}
112
+ </div>
113
+
114
+ <button
115
+ type="button"
116
+ onClick={handleGenerate}
117
+ disabled={!file || loading}
118
+ className={[
119
+ "w-full inline-flex items-center justify-center gap-2 px-4 py-3 rounded-xl text-sm font-medium transition-all",
120
+ !file || loading
121
+ ? "bg-white/5 text-white/40 cursor-not-allowed border border-white/5"
122
+ : "bg-gradient-to-r from-violet-500 to-fuchsia-500 text-white shadow-lg shadow-violet-500/30 hover:shadow-violet-500/50 hover:from-violet-400 hover:to-fuchsia-400",
123
+ ].join(" ")}
124
+ >
125
+ {loading ? (
126
+ <>
127
+ <Spinner size="sm" />
128
+ Generating caption…
129
+ </>
130
+ ) : (
131
+ <>
132
+ <svg
133
+ viewBox="0 0 24 24"
134
+ className="w-4 h-4"
135
+ fill="none"
136
+ stroke="currentColor"
137
+ strokeWidth="2"
138
+ strokeLinecap="round"
139
+ strokeLinejoin="round"
140
+ >
141
+ <path d="M5 12h14" />
142
+ <path d="m12 5 7 7-7 7" />
143
+ </svg>
144
+ Generate caption
145
+ </>
146
+ )}
147
+ </button>
148
+
149
+ <p className="text-xs text-white/40 mt-3 leading-relaxed">
150
+ {file
151
+ ? "Click generate to send the image to the inference endpoint."
152
+ : "Upload an image to enable generation."}
153
+ </p>
154
+ </div>
155
+
156
+ {result && <CaptionResult result={result} />}
157
+ </div>
158
+ </div>
159
+ </main>
160
+
161
+ <footer className="max-w-6xl mx-auto px-6 py-8 text-xs text-white/30 border-t border-white/5 mt-10">
162
+ POST <span className="font-mono text-white/50">/v1/captions</span> ·
163
+ built with React + Vite + Tailwind
164
+ </footer>
165
+ </div>
166
+ </div>
167
+ );
168
+ }
frontend/src/assets/hero.png ADDED
frontend/src/assets/react.svg ADDED
frontend/src/assets/vite.svg ADDED
frontend/src/components/CaptionResult.jsx ADDED
@@ -0,0 +1,97 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import { useState } from "react";
2
+
3
+ function MetaPill({ label, value }) {
4
+ return (
5
+ <div className="flex items-center gap-2 px-3 py-1.5 rounded-full bg-white/[0.04] border border-white/10 text-xs">
6
+ <span className="text-white/40 uppercase tracking-wide">{label}</span>
7
+ <span className="text-white/80 font-medium">{value}</span>
8
+ </div>
9
+ );
10
+ }
11
+
12
+ export default function CaptionResult({ result }) {
13
+ const [copied, setCopied] = useState(false);
14
+
15
+ if (!result) return null;
16
+
17
+ const copy = async () => {
18
+ try {
19
+ await navigator.clipboard.writeText(result.caption);
20
+ setCopied(true);
21
+ setTimeout(() => setCopied(false), 1500);
22
+ } catch {
23
+ /* ignore */
24
+ }
25
+ };
26
+
27
+ return (
28
+ <div className="relative rounded-2xl border border-white/10 bg-gradient-to-b from-white/[0.04] to-white/[0.01] p-6 overflow-hidden">
29
+ <div className="absolute -top-24 -right-24 w-64 h-64 bg-violet-500/10 rounded-full blur-3xl pointer-events-none" />
30
+ <div className="relative">
31
+ <div className="flex items-center gap-2 mb-3">
32
+ <div className="w-1.5 h-1.5 rounded-full bg-emerald-400 shadow-[0_0_8px_rgba(52,211,153,0.7)]" />
33
+ <span className="text-xs uppercase tracking-wider text-white/40">
34
+ Generated caption
35
+ </span>
36
+ </div>
37
+
38
+ <p className="text-xl md:text-2xl text-white font-medium leading-snug">
39
+ {result.caption}
40
+ </p>
41
+
42
+ <div className="flex flex-wrap gap-2 mt-5">
43
+ <MetaPill label="Model" value={result.model_version} />
44
+ <MetaPill label="Decode" value={result.decode_strategy} />
45
+ <MetaPill
46
+ label="Latency"
47
+ value={`${result.latency_ms.toFixed(1)} ms`}
48
+ />
49
+ </div>
50
+
51
+ <div className="flex items-center justify-between mt-5 pt-4 border-t border-white/5">
52
+ <span className="text-[11px] text-white/30 font-mono truncate">
53
+ req: {result.request_id}
54
+ </span>
55
+ <button
56
+ type="button"
57
+ onClick={copy}
58
+ className="inline-flex items-center gap-1.5 text-xs text-white/70 hover:text-white px-2.5 py-1.5 rounded-md border border-white/10 hover:border-white/20 transition-colors"
59
+ >
60
+ {copied ? (
61
+ <>
62
+ <svg
63
+ viewBox="0 0 24 24"
64
+ className="w-3.5 h-3.5 text-emerald-400"
65
+ fill="none"
66
+ stroke="currentColor"
67
+ strokeWidth="2.4"
68
+ strokeLinecap="round"
69
+ strokeLinejoin="round"
70
+ >
71
+ <polyline points="20 6 9 17 4 12" />
72
+ </svg>
73
+ Copied
74
+ </>
75
+ ) : (
76
+ <>
77
+ <svg
78
+ viewBox="0 0 24 24"
79
+ className="w-3.5 h-3.5"
80
+ fill="none"
81
+ stroke="currentColor"
82
+ strokeWidth="2"
83
+ strokeLinecap="round"
84
+ strokeLinejoin="round"
85
+ >
86
+ <rect x="9" y="9" width="13" height="13" rx="2" />
87
+ <path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1" />
88
+ </svg>
89
+ Copy
90
+ </>
91
+ )}
92
+ </button>
93
+ </div>
94
+ </div>
95
+ </div>
96
+ );
97
+ }
frontend/src/components/ErrorBanner.jsx ADDED
@@ -0,0 +1,48 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ export default function ErrorBanner({ message, onDismiss }) {
2
+ if (!message) return null;
3
+ return (
4
+ <div
5
+ role="alert"
6
+ className="flex items-start gap-3 rounded-xl border border-red-500/30 bg-red-500/10 px-4 py-3"
7
+ >
8
+ <svg
9
+ viewBox="0 0 24 24"
10
+ className="w-5 h-5 text-red-400 shrink-0 mt-0.5"
11
+ fill="none"
12
+ stroke="currentColor"
13
+ strokeWidth="2"
14
+ strokeLinecap="round"
15
+ strokeLinejoin="round"
16
+ >
17
+ <circle cx="12" cy="12" r="10" />
18
+ <line x1="12" y1="8" x2="12" y2="12" />
19
+ <line x1="12" y1="16" x2="12.01" y2="16" />
20
+ </svg>
21
+ <div className="flex-1 min-w-0">
22
+ <p className="text-sm text-red-100 font-medium">Something went wrong</p>
23
+ <p className="text-sm text-red-200/80 mt-0.5 break-words">{message}</p>
24
+ </div>
25
+ {onDismiss && (
26
+ <button
27
+ type="button"
28
+ onClick={onDismiss}
29
+ className="text-red-200/60 hover:text-red-100 transition-colors"
30
+ aria-label="Dismiss error"
31
+ >
32
+ <svg
33
+ viewBox="0 0 24 24"
34
+ className="w-4 h-4"
35
+ fill="none"
36
+ stroke="currentColor"
37
+ strokeWidth="2"
38
+ strokeLinecap="round"
39
+ strokeLinejoin="round"
40
+ >
41
+ <line x1="18" y1="6" x2="6" y2="18" />
42
+ <line x1="6" y1="6" x2="18" y2="18" />
43
+ </svg>
44
+ </button>
45
+ )}
46
+ </div>
47
+ );
48
+ }
frontend/src/components/Header.jsx ADDED
@@ -0,0 +1,39 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import StatusBadge from "./StatusBadge";
2
+
3
+ export default function Header() {
4
+ return (
5
+ <header className="border-b border-white/5 bg-black/40 backdrop-blur-xl sticky top-0 z-30">
6
+ <div className="max-w-6xl mx-auto px-6 py-4 flex items-center justify-between">
7
+ <div className="flex items-center gap-3">
8
+ <div className="relative">
9
+ <div className="w-9 h-9 rounded-xl bg-gradient-to-br from-violet-500 via-fuchsia-500 to-indigo-500 flex items-center justify-center shadow-lg shadow-violet-500/30">
10
+ <svg
11
+ viewBox="0 0 24 24"
12
+ className="w-5 h-5 text-white"
13
+ fill="none"
14
+ stroke="currentColor"
15
+ strokeWidth="2.2"
16
+ strokeLinecap="round"
17
+ strokeLinejoin="round"
18
+ >
19
+ <rect x="3" y="3" width="18" height="18" rx="3" />
20
+ <circle cx="9" cy="9" r="1.5" />
21
+ <path d="m21 15-5-5L5 21" />
22
+ </svg>
23
+ </div>
24
+ <div className="absolute inset-0 rounded-xl bg-violet-500/40 blur-xl -z-10" />
25
+ </div>
26
+ <div className="flex flex-col leading-tight">
27
+ <span className="text-white font-semibold tracking-tight">
28
+ Caption Studio
29
+ </span>
30
+ <span className="text-xs text-white/40">
31
+ Vision-to-text inference
32
+ </span>
33
+ </div>
34
+ </div>
35
+ <StatusBadge />
36
+ </div>
37
+ </header>
38
+ );
39
+ }
frontend/src/components/ImagePreview.jsx ADDED
@@ -0,0 +1,53 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ function formatBytes(bytes) {
2
+ if (!bytes) return "0 B";
3
+ const units = ["B", "KB", "MB", "GB"];
4
+ const i = Math.floor(Math.log(bytes) / Math.log(1024));
5
+ return `${(bytes / Math.pow(1024, i)).toFixed(1)} ${units[i]}`;
6
+ }
7
+
8
+ export default function ImagePreview({ file, previewUrl, onClear, disabled }) {
9
+ if (!file || !previewUrl) return null;
10
+
11
+ return (
12
+ <div className="relative rounded-2xl border border-white/10 bg-white/[0.02] overflow-hidden">
13
+ <div className="aspect-video w-full bg-black flex items-center justify-center">
14
+ <img
15
+ src={previewUrl}
16
+ alt={file.name}
17
+ className="max-h-full max-w-full object-contain"
18
+ />
19
+ </div>
20
+ <div className="flex items-center justify-between px-4 py-3 border-t border-white/10 bg-black/40">
21
+ <div className="min-w-0 flex-1">
22
+ <p className="text-sm text-white/90 truncate" title={file.name}>
23
+ {file.name}
24
+ </p>
25
+ <p className="text-xs text-white/40 mt-0.5">
26
+ {formatBytes(file.size)} ·{" "}
27
+ {file.type.replace("image/", "").toUpperCase()}
28
+ </p>
29
+ </div>
30
+ <button
31
+ type="button"
32
+ onClick={onClear}
33
+ disabled={disabled}
34
+ className="ml-3 inline-flex items-center gap-1.5 text-xs text-white/60 hover:text-white px-2.5 py-1.5 rounded-md border border-white/10 hover:border-white/20 transition-colors disabled:opacity-40 disabled:cursor-not-allowed"
35
+ >
36
+ <svg
37
+ viewBox="0 0 24 24"
38
+ className="w-3.5 h-3.5"
39
+ fill="none"
40
+ stroke="currentColor"
41
+ strokeWidth="2"
42
+ strokeLinecap="round"
43
+ strokeLinejoin="round"
44
+ >
45
+ <line x1="18" y1="6" x2="6" y2="18" />
46
+ <line x1="6" y1="6" x2="18" y2="18" />
47
+ </svg>
48
+ Remove
49
+ </button>
50
+ </div>
51
+ </div>
52
+ );
53
+ }
frontend/src/components/Spinner.jsx ADDED
@@ -0,0 +1,14 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ export default function Spinner({ size = "md", className = "" }) {
2
+ const sizeMap = {
3
+ sm: "w-4 h-4 border-2",
4
+ md: "w-6 h-6 border-2",
5
+ lg: "w-10 h-10 border-[3px]",
6
+ };
7
+ return (
8
+ <div
9
+ className={`${sizeMap[size]} ${className} rounded-full border-white/15 border-t-violet-400 animate-spin`}
10
+ role="status"
11
+ aria-label="Loading"
12
+ />
13
+ );
14
+ }
frontend/src/components/StatusBadge.jsx ADDED
@@ -0,0 +1,65 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import { useEffect, useRef, useState } from "react";
2
+ import { checkHealth } from "../services/api";
3
+
4
+ const POLL_INTERVAL_MS = 10000;
5
+
6
+ export default function StatusBadge() {
7
+ const [health, setHealth] = useState(null);
8
+ const [phase, setPhase] = useState("checking"); // 'checking' | 'online' | 'offline'
9
+ const mountedRef = useRef(true);
10
+
11
+ useEffect(() => {
12
+ mountedRef.current = true;
13
+ let cancelled = false;
14
+
15
+ const poll = async () => {
16
+ const result = await checkHealth();
17
+ if (cancelled || !mountedRef.current) return;
18
+ if (result?.model_loaded) {
19
+ setHealth(result);
20
+ setPhase("online");
21
+ } else {
22
+ setHealth(result);
23
+ setPhase("offline");
24
+ }
25
+ };
26
+
27
+ poll();
28
+ const interval = setInterval(poll, POLL_INTERVAL_MS);
29
+
30
+ const onFocus = () => poll();
31
+ window.addEventListener("focus", onFocus);
32
+
33
+ return () => {
34
+ cancelled = true;
35
+ mountedRef.current = false;
36
+ clearInterval(interval);
37
+ window.removeEventListener("focus", onFocus);
38
+ };
39
+ }, []);
40
+
41
+ if (phase === "checking") {
42
+ return (
43
+ <div className="inline-flex items-center gap-2 px-3 py-1 bg-gray-700 text-gray-300 rounded-full text-sm">
44
+ <div className="w-2 h-2 bg-gray-500 rounded-full animate-pulse" />
45
+ Checking...
46
+ </div>
47
+ );
48
+ }
49
+
50
+ const isOnline = phase === "online";
51
+ const statusColor = isOnline ? "bg-green-500" : "bg-red-500";
52
+ const statusText = isOnline ? "Backend online" : "Backend offline";
53
+
54
+ return (
55
+ <div className="inline-flex items-center gap-2 px-3 py-1 bg-gray-900 border border-gray-700 rounded-full text-sm">
56
+ <div className={`w-2 h-2 ${statusColor} rounded-full`} />
57
+ <span className="text-gray-300">{statusText}</span>
58
+ {health?.api_version && (
59
+ <span className="text-gray-500 text-xs ml-1">
60
+ v{health.api_version}
61
+ </span>
62
+ )}
63
+ </div>
64
+ );
65
+ }
frontend/src/components/UploadZone.jsx ADDED
@@ -0,0 +1,104 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import { useRef, useState } from "react";
2
+
3
+ const ACCEPTED = ["image/jpeg", "image/png", "image/webp"];
4
+ const MAX_BYTES = 10 * 1024 * 1024;
5
+
6
+ export default function UploadZone({ onFileSelected, disabled = false }) {
7
+ const inputRef = useRef(null);
8
+ const [dragActive, setDragActive] = useState(false);
9
+
10
+ const validate = (file) => {
11
+ if (!file) return "No file selected.";
12
+ if (!ACCEPTED.includes(file.type)) {
13
+ return "Unsupported format. Use JPG, PNG, or WEBP.";
14
+ }
15
+ if (file.size > MAX_BYTES) {
16
+ return "File exceeds 10 MB limit.";
17
+ }
18
+ return null;
19
+ };
20
+
21
+ const handleFile = (file) => {
22
+ const error = validate(file);
23
+ onFileSelected(file ?? null, error);
24
+ };
25
+
26
+ const onDrop = (e) => {
27
+ e.preventDefault();
28
+ setDragActive(false);
29
+ if (disabled) return;
30
+ const file = e.dataTransfer?.files?.[0];
31
+ if (file) handleFile(file);
32
+ };
33
+
34
+ const onDragOver = (e) => {
35
+ e.preventDefault();
36
+ if (!disabled) setDragActive(true);
37
+ };
38
+
39
+ const onDragLeave = (e) => {
40
+ e.preventDefault();
41
+ setDragActive(false);
42
+ };
43
+
44
+ const onChange = (e) => {
45
+ const file = e.target.files?.[0];
46
+ if (file) handleFile(file);
47
+ e.target.value = "";
48
+ };
49
+
50
+ return (
51
+ <div
52
+ onDrop={onDrop}
53
+ onDragOver={onDragOver}
54
+ onDragLeave={onDragLeave}
55
+ onClick={() => !disabled && inputRef.current?.click()}
56
+ role="button"
57
+ tabIndex={0}
58
+ onKeyDown={(e) => {
59
+ if ((e.key === "Enter" || e.key === " ") && !disabled) {
60
+ e.preventDefault();
61
+ inputRef.current?.click();
62
+ }
63
+ }}
64
+ className={[
65
+ "relative w-full rounded-2xl border border-dashed transition-all duration-200 select-none",
66
+ "flex flex-col items-center justify-center text-center px-6 py-16",
67
+ disabled ? "cursor-not-allowed opacity-60" : "cursor-pointer",
68
+ dragActive
69
+ ? "border-violet-400/80 bg-violet-500/10 shadow-[0_0_0_4px_rgba(139,92,246,0.12)]"
70
+ : "border-white/15 bg-white/[0.02] hover:border-white/25 hover:bg-white/[0.04]",
71
+ ].join(" ")}
72
+ >
73
+ <input
74
+ ref={inputRef}
75
+ type="file"
76
+ accept={ACCEPTED.join(",")}
77
+ onChange={onChange}
78
+ className="hidden"
79
+ disabled={disabled}
80
+ />
81
+ <div className="w-14 h-14 rounded-2xl bg-gradient-to-br from-violet-500/20 to-fuchsia-500/20 border border-white/10 flex items-center justify-center mb-4">
82
+ <svg
83
+ viewBox="0 0 24 24"
84
+ className="w-7 h-7 text-violet-300"
85
+ fill="none"
86
+ stroke="currentColor"
87
+ strokeWidth="1.8"
88
+ strokeLinecap="round"
89
+ strokeLinejoin="round"
90
+ >
91
+ <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4" />
92
+ <polyline points="17 8 12 3 7 8" />
93
+ <line x1="12" y1="3" x2="12" y2="15" />
94
+ </svg>
95
+ </div>
96
+ <p className="text-white/90 text-base font-medium">
97
+ {dragActive ? "Drop to upload" : "Drag & drop an image"}
98
+ </p>
99
+ <p className="text-white/40 text-sm mt-1">
100
+ or click to browse — JPG, PNG, WEBP · max 10 MB
101
+ </p>
102
+ </div>
103
+ );
104
+ }
frontend/src/index.css ADDED
@@ -0,0 +1 @@
 
 
1
+ @import "tailwindcss";
frontend/src/main.jsx ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ import { StrictMode } from "react";
2
+ import { createRoot } from "react-dom/client";
3
+ import "./index.css";
4
+ import App from "./App.jsx";
5
+
6
+ createRoot(document.getElementById("root")).render(
7
+ <StrictMode>
8
+ <App />
9
+ </StrictMode>,
10
+ );
frontend/src/services/api.js ADDED
@@ -0,0 +1,90 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const DEFAULT_API_BASE = "http://127.0.0.1:8000";
2
+ const HEALTH_TIMEOUT_MS = 3000;
3
+ const CAPTION_TIMEOUT_MS = 60000;
4
+
5
+ export const API_BASE = (
6
+ import.meta.env?.VITE_API_BASE ?? DEFAULT_API_BASE
7
+ ).replace(/\/$/, "");
8
+
9
+ class ApiError extends Error {
10
+ constructor(message, { kind = "unknown", status = null, cause } = {}) {
11
+ super(message);
12
+ this.name = "ApiError";
13
+ this.kind = kind;
14
+ this.status = status;
15
+ if (cause) this.cause = cause;
16
+ }
17
+ }
18
+
19
+ const isAbortError = (err) => err?.name === "AbortError" || err?.code === 20;
20
+
21
+ const classifyFetchError = (err) => {
22
+ if (isAbortError(err)) {
23
+ return new ApiError("Request timed out.", { kind: "timeout", cause: err });
24
+ }
25
+ // Browsers surface CORS denials and network failures as a generic TypeError.
26
+ if (err instanceof TypeError) {
27
+ return new ApiError(
28
+ "Cannot reach backend. Check that the API is running and CORS allows this origin.",
29
+ { kind: "network", cause: err },
30
+ );
31
+ }
32
+ return new ApiError(err?.message || "Request failed.", {
33
+ kind: "unknown",
34
+ cause: err,
35
+ });
36
+ };
37
+
38
+ const fetchWithTimeout = async (url, { timeoutMs, ...init } = {}) => {
39
+ const controller = new AbortController();
40
+ const timer = setTimeout(() => controller.abort(), timeoutMs);
41
+ try {
42
+ return await fetch(url, { ...init, signal: controller.signal });
43
+ } finally {
44
+ clearTimeout(timer);
45
+ }
46
+ };
47
+
48
+ export const checkHealth = async () => {
49
+ try {
50
+ const response = await fetchWithTimeout(`${API_BASE}/healthz`, {
51
+ timeoutMs: HEALTH_TIMEOUT_MS,
52
+ headers: { Accept: "application/json" },
53
+ });
54
+ if (!response.ok) return null;
55
+ return await response.json();
56
+ } catch {
57
+ return null;
58
+ }
59
+ };
60
+
61
+ export const captionImage = async (imageFile) => {
62
+ const formData = new FormData();
63
+ formData.append("image", imageFile);
64
+
65
+ let response;
66
+ try {
67
+ response = await fetchWithTimeout(`${API_BASE}/v1/captions`, {
68
+ method: "POST",
69
+ body: formData,
70
+ timeoutMs: CAPTION_TIMEOUT_MS,
71
+ });
72
+ } catch (err) {
73
+ throw classifyFetchError(err);
74
+ }
75
+
76
+ if (!response.ok) {
77
+ let detail = `HTTP ${response.status}`;
78
+ try {
79
+ const errorData = await response.json();
80
+ if (errorData?.detail) detail = errorData.detail;
81
+ } catch {
82
+ /* response body was not JSON */
83
+ }
84
+ throw new ApiError(detail, { kind: "http", status: response.status });
85
+ }
86
+
87
+ return response.json();
88
+ };
89
+
90
+ export { ApiError };
frontend/vite.config.js ADDED
@@ -0,0 +1,7 @@
 
 
 
 
 
 
 
 
1
+ import { defineConfig } from "vite";
2
+ import react from "@vitejs/plugin-react";
3
+ import tailwindcss from "@tailwindcss/vite";
4
+
5
+ export default defineConfig({
6
+ plugins: [react(), tailwindcss()],
7
+ });