commit 0551fb9e20667af40da6d09dd4a6e343abc54205
Author: MaeLucia <597445175@qq.com>
Date: Tue Feb 17 20:28:53 2026 +0800
Initial commit: 入舍智能家装前端项目
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..697175d
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,36 @@
+# Dependencies
+node_modules
+.pnp
+.pnp.js
+
+# Testing
+coverage
+
+# Production
+dist
+build
+
+# Misc
+.DS_Store
+*.pem
+
+# Debug
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+
+# Local env files
+.env
+.env.local
+.env.development.local
+.env.test.local
+.env.production.local
+
+# IDE
+.idea
+.vscode
+*.swp
+*.swo
+
+# Trae
+.trae
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..d2e7761
--- /dev/null
+++ b/README.md
@@ -0,0 +1,73 @@
+# React + TypeScript + Vite
+
+This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
+
+Currently, two official plugins are available:
+
+- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Babel](https://babeljs.io/) (or [oxc](https://oxc.rs) when used in [rolldown-vite](https://vite.dev/guide/rolldown)) for Fast Refresh
+- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh
+
+## React Compiler
+
+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).
+
+## Expanding the ESLint configuration
+
+If you are developing a production application, we recommend updating the configuration to enable type-aware lint rules:
+
+```js
+export default defineConfig([
+ globalIgnores(['dist']),
+ {
+ files: ['**/*.{ts,tsx}'],
+ extends: [
+ // Other configs...
+
+ // Remove tseslint.configs.recommended and replace with this
+ tseslint.configs.recommendedTypeChecked,
+ // Alternatively, use this for stricter rules
+ tseslint.configs.strictTypeChecked,
+ // Optionally, add this for stylistic rules
+ tseslint.configs.stylisticTypeChecked,
+
+ // Other configs...
+ ],
+ languageOptions: {
+ parserOptions: {
+ project: ['./tsconfig.node.json', './tsconfig.app.json'],
+ tsconfigRootDir: import.meta.dirname,
+ },
+ // other options...
+ },
+ },
+])
+```
+
+You can also install [eslint-plugin-react-x](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-x) and [eslint-plugin-react-dom](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-dom) for React-specific lint rules:
+
+```js
+// eslint.config.js
+import reactX from 'eslint-plugin-react-x'
+import reactDom from 'eslint-plugin-react-dom'
+
+export default defineConfig([
+ globalIgnores(['dist']),
+ {
+ files: ['**/*.{ts,tsx}'],
+ extends: [
+ // Other configs...
+ // Enable lint rules for React
+ reactX.configs['recommended-typescript'],
+ // Enable lint rules for React DOM
+ reactDom.configs.recommended,
+ ],
+ languageOptions: {
+ parserOptions: {
+ project: ['./tsconfig.node.json', './tsconfig.app.json'],
+ tsconfigRootDir: import.meta.dirname,
+ },
+ // other options...
+ },
+ },
+])
+```
diff --git a/eslint.config.js b/eslint.config.js
new file mode 100644
index 0000000..5e6b472
--- /dev/null
+++ b/eslint.config.js
@@ -0,0 +1,23 @@
+import js from '@eslint/js'
+import globals from 'globals'
+import reactHooks from 'eslint-plugin-react-hooks'
+import reactRefresh from 'eslint-plugin-react-refresh'
+import tseslint from 'typescript-eslint'
+import { defineConfig, globalIgnores } from 'eslint/config'
+
+export default defineConfig([
+ globalIgnores(['dist']),
+ {
+ files: ['**/*.{ts,tsx}'],
+ extends: [
+ js.configs.recommended,
+ tseslint.configs.recommended,
+ reactHooks.configs.flat.recommended,
+ reactRefresh.configs.vite,
+ ],
+ languageOptions: {
+ ecmaVersion: 2020,
+ globals: globals.browser,
+ },
+ },
+])
diff --git a/index.html b/index.html
new file mode 100644
index 0000000..f5d30f4
--- /dev/null
+++ b/index.html
@@ -0,0 +1,14 @@
+
+
+
+
+
+
+
+ 入舍 - 智能家装
+
+
+
+
+
+
diff --git a/package-lock.json b/package-lock.json
new file mode 100644
index 0000000..54996ab
--- /dev/null
+++ b/package-lock.json
@@ -0,0 +1,2621 @@
+{
+ "name": "rushe-ui",
+ "version": "0.0.0",
+ "lockfileVersion": 3,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "rushe-ui",
+ "version": "0.0.0",
+ "dependencies": {
+ "clsx": "^2.1.0",
+ "framer-motion": "^11.0.0",
+ "lucide-react": "^0.323.0",
+ "react": "^18.2.0",
+ "react-dom": "^18.2.0",
+ "react-router-dom": "^6.22.0",
+ "zustand": "^4.5.0"
+ },
+ "devDependencies": {
+ "@types/react": "^18.2.55",
+ "@types/react-dom": "^18.2.19",
+ "@vitejs/plugin-react": "^4.2.1",
+ "autoprefixer": "^10.4.17",
+ "postcss": "^8.4.35",
+ "tailwindcss": "^3.4.1",
+ "typescript": "^5.2.2",
+ "vite": "^5.1.0"
+ }
+ },
+ "node_modules/@alloc/quick-lru": {
+ "version": "5.2.0",
+ "resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz",
+ "integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==",
+ "dev": true,
+ "engines": {
+ "node": ">=10"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/@babel/code-frame": {
+ "version": "7.29.0",
+ "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz",
+ "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==",
+ "dev": true,
+ "dependencies": {
+ "@babel/helper-validator-identifier": "^7.28.5",
+ "js-tokens": "^4.0.0",
+ "picocolors": "^1.1.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/compat-data": {
+ "version": "7.29.0",
+ "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.0.tgz",
+ "integrity": "sha512-T1NCJqT/j9+cn8fvkt7jtwbLBfLC/1y1c7NtCeXFRgzGTsafi68MRv8yzkYSapBnFA6L3U2VSc02ciDzoAJhJg==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/core": {
+ "version": "7.29.0",
+ "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz",
+ "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==",
+ "dev": true,
+ "dependencies": {
+ "@babel/code-frame": "^7.29.0",
+ "@babel/generator": "^7.29.0",
+ "@babel/helper-compilation-targets": "^7.28.6",
+ "@babel/helper-module-transforms": "^7.28.6",
+ "@babel/helpers": "^7.28.6",
+ "@babel/parser": "^7.29.0",
+ "@babel/template": "^7.28.6",
+ "@babel/traverse": "^7.29.0",
+ "@babel/types": "^7.29.0",
+ "@jridgewell/remapping": "^2.3.5",
+ "convert-source-map": "^2.0.0",
+ "debug": "^4.1.0",
+ "gensync": "^1.0.0-beta.2",
+ "json5": "^2.2.3",
+ "semver": "^6.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/babel"
+ }
+ },
+ "node_modules/@babel/generator": {
+ "version": "7.29.1",
+ "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz",
+ "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==",
+ "dev": true,
+ "dependencies": {
+ "@babel/parser": "^7.29.0",
+ "@babel/types": "^7.29.0",
+ "@jridgewell/gen-mapping": "^0.3.12",
+ "@jridgewell/trace-mapping": "^0.3.28",
+ "jsesc": "^3.0.2"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-compilation-targets": {
+ "version": "7.28.6",
+ "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz",
+ "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==",
+ "dev": true,
+ "dependencies": {
+ "@babel/compat-data": "^7.28.6",
+ "@babel/helper-validator-option": "^7.27.1",
+ "browserslist": "^4.24.0",
+ "lru-cache": "^5.1.1",
+ "semver": "^6.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-globals": {
+ "version": "7.28.0",
+ "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz",
+ "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-module-imports": {
+ "version": "7.28.6",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz",
+ "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==",
+ "dev": true,
+ "dependencies": {
+ "@babel/traverse": "^7.28.6",
+ "@babel/types": "^7.28.6"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-module-transforms": {
+ "version": "7.28.6",
+ "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz",
+ "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==",
+ "dev": true,
+ "dependencies": {
+ "@babel/helper-module-imports": "^7.28.6",
+ "@babel/helper-validator-identifier": "^7.28.5",
+ "@babel/traverse": "^7.28.6"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0"
+ }
+ },
+ "node_modules/@babel/helper-plugin-utils": {
+ "version": "7.28.6",
+ "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.28.6.tgz",
+ "integrity": "sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-string-parser": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz",
+ "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-validator-identifier": {
+ "version": "7.28.5",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
+ "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helper-validator-option": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz",
+ "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/helpers": {
+ "version": "7.28.6",
+ "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.6.tgz",
+ "integrity": "sha512-xOBvwq86HHdB7WUDTfKfT/Vuxh7gElQ+Sfti2Cy6yIWNW05P8iUslOVcZ4/sKbE+/jQaukQAdz/gf3724kYdqw==",
+ "dev": true,
+ "dependencies": {
+ "@babel/template": "^7.28.6",
+ "@babel/types": "^7.28.6"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/parser": {
+ "version": "7.29.0",
+ "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.0.tgz",
+ "integrity": "sha512-IyDgFV5GeDUVX4YdF/3CPULtVGSXXMLh1xVIgdCgxApktqnQV0r7/8Nqthg+8YLGaAtdyIlo2qIdZrbCv4+7ww==",
+ "dev": true,
+ "dependencies": {
+ "@babel/types": "^7.29.0"
+ },
+ "bin": {
+ "parser": "bin/babel-parser.js"
+ },
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@babel/plugin-transform-react-jsx-self": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.27.1.tgz",
+ "integrity": "sha512-6UzkCs+ejGdZ5mFFC/OCUrv028ab2fp1znZmCZjAOBKiBK2jXD1O+BPSfX8X2qjJ75fZBMSnQn3Rq2mrBJK2mw==",
+ "dev": true,
+ "dependencies": {
+ "@babel/helper-plugin-utils": "^7.27.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0-0"
+ }
+ },
+ "node_modules/@babel/plugin-transform-react-jsx-source": {
+ "version": "7.27.1",
+ "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.27.1.tgz",
+ "integrity": "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw==",
+ "dev": true,
+ "dependencies": {
+ "@babel/helper-plugin-utils": "^7.27.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ },
+ "peerDependencies": {
+ "@babel/core": "^7.0.0-0"
+ }
+ },
+ "node_modules/@babel/template": {
+ "version": "7.28.6",
+ "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz",
+ "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==",
+ "dev": true,
+ "dependencies": {
+ "@babel/code-frame": "^7.28.6",
+ "@babel/parser": "^7.28.6",
+ "@babel/types": "^7.28.6"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/traverse": {
+ "version": "7.29.0",
+ "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz",
+ "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==",
+ "dev": true,
+ "dependencies": {
+ "@babel/code-frame": "^7.29.0",
+ "@babel/generator": "^7.29.0",
+ "@babel/helper-globals": "^7.28.0",
+ "@babel/parser": "^7.29.0",
+ "@babel/template": "^7.28.6",
+ "@babel/types": "^7.29.0",
+ "debug": "^4.3.1"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@babel/types": {
+ "version": "7.29.0",
+ "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz",
+ "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==",
+ "dev": true,
+ "dependencies": {
+ "@babel/helper-string-parser": "^7.27.1",
+ "@babel/helper-validator-identifier": "^7.28.5"
+ },
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/@esbuild/aix-ppc64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz",
+ "integrity": "sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "aix"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/android-arm": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.21.5.tgz",
+ "integrity": "sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/android-arm64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz",
+ "integrity": "sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/android-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.21.5.tgz",
+ "integrity": "sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/darwin-arm64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz",
+ "integrity": "sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/darwin-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz",
+ "integrity": "sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/freebsd-arm64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz",
+ "integrity": "sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/freebsd-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz",
+ "integrity": "sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-arm": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz",
+ "integrity": "sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-arm64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz",
+ "integrity": "sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-ia32": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz",
+ "integrity": "sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-loong64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz",
+ "integrity": "sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-mips64el": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz",
+ "integrity": "sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==",
+ "cpu": [
+ "mips64el"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-ppc64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz",
+ "integrity": "sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-riscv64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz",
+ "integrity": "sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-s390x": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz",
+ "integrity": "sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==",
+ "cpu": [
+ "s390x"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/linux-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz",
+ "integrity": "sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/netbsd-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz",
+ "integrity": "sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "netbsd"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/openbsd-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz",
+ "integrity": "sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "openbsd"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/sunos-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz",
+ "integrity": "sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "sunos"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/win32-arm64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz",
+ "integrity": "sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/win32-ia32": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz",
+ "integrity": "sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@esbuild/win32-x64": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz",
+ "integrity": "sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/@jridgewell/gen-mapping": {
+ "version": "0.3.13",
+ "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
+ "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
+ "dev": true,
+ "dependencies": {
+ "@jridgewell/sourcemap-codec": "^1.5.0",
+ "@jridgewell/trace-mapping": "^0.3.24"
+ }
+ },
+ "node_modules/@jridgewell/remapping": {
+ "version": "2.3.5",
+ "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
+ "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
+ "dev": true,
+ "dependencies": {
+ "@jridgewell/gen-mapping": "^0.3.5",
+ "@jridgewell/trace-mapping": "^0.3.24"
+ }
+ },
+ "node_modules/@jridgewell/resolve-uri": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
+ "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/@jridgewell/sourcemap-codec": {
+ "version": "1.5.5",
+ "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
+ "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
+ "dev": true
+ },
+ "node_modules/@jridgewell/trace-mapping": {
+ "version": "0.3.31",
+ "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
+ "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
+ "dev": true,
+ "dependencies": {
+ "@jridgewell/resolve-uri": "^3.1.0",
+ "@jridgewell/sourcemap-codec": "^1.4.14"
+ }
+ },
+ "node_modules/@nodelib/fs.scandir": {
+ "version": "2.1.5",
+ "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
+ "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==",
+ "dev": true,
+ "dependencies": {
+ "@nodelib/fs.stat": "2.0.5",
+ "run-parallel": "^1.1.9"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@nodelib/fs.stat": {
+ "version": "2.0.5",
+ "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz",
+ "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==",
+ "dev": true,
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@nodelib/fs.walk": {
+ "version": "1.2.8",
+ "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz",
+ "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==",
+ "dev": true,
+ "dependencies": {
+ "@nodelib/fs.scandir": "2.1.5",
+ "fastq": "^1.6.0"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/@remix-run/router": {
+ "version": "1.23.2",
+ "resolved": "https://registry.npmjs.org/@remix-run/router/-/router-1.23.2.tgz",
+ "integrity": "sha512-Ic6m2U/rMjTkhERIa/0ZtXJP17QUi2CbWE7cqx4J58M8aA3QTfW+2UlQ4psvTX9IO1RfNVhK3pcpdjej7L+t2w==",
+ "engines": {
+ "node": ">=14.0.0"
+ }
+ },
+ "node_modules/@rolldown/pluginutils": {
+ "version": "1.0.0-beta.27",
+ "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.27.tgz",
+ "integrity": "sha512-+d0F4MKMCbeVUJwG96uQ4SgAznZNSq93I3V+9NHA4OpvqG8mRCpGdKmK8l/dl02h2CCDHwW2FqilnTyDcAnqjA==",
+ "dev": true
+ },
+ "node_modules/@rollup/rollup-android-arm-eabi": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.57.1.tgz",
+ "integrity": "sha512-A6ehUVSiSaaliTxai040ZpZ2zTevHYbvu/lDoeAteHI8QnaosIzm4qwtezfRg1jOYaUmnzLX1AOD6Z+UJjtifg==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "android"
+ ]
+ },
+ "node_modules/@rollup/rollup-android-arm64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.57.1.tgz",
+ "integrity": "sha512-dQaAddCY9YgkFHZcFNS/606Exo8vcLHwArFZ7vxXq4rigo2bb494/xKMMwRRQW6ug7Js6yXmBZhSBRuBvCCQ3w==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "android"
+ ]
+ },
+ "node_modules/@rollup/rollup-darwin-arm64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.57.1.tgz",
+ "integrity": "sha512-crNPrwJOrRxagUYeMn/DZwqN88SDmwaJ8Cvi/TN1HnWBU7GwknckyosC2gd0IqYRsHDEnXf328o9/HC6OkPgOg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "darwin"
+ ]
+ },
+ "node_modules/@rollup/rollup-darwin-x64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.57.1.tgz",
+ "integrity": "sha512-Ji8g8ChVbKrhFtig5QBV7iMaJrGtpHelkB3lsaKzadFBe58gmjfGXAOfI5FV0lYMH8wiqsxKQ1C9B0YTRXVy4w==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "darwin"
+ ]
+ },
+ "node_modules/@rollup/rollup-freebsd-arm64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.57.1.tgz",
+ "integrity": "sha512-R+/WwhsjmwodAcz65guCGFRkMb4gKWTcIeLy60JJQbXrJ97BOXHxnkPFrP+YwFlaS0m+uWJTstrUA9o+UchFug==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "freebsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-freebsd-x64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.57.1.tgz",
+ "integrity": "sha512-IEQTCHeiTOnAUC3IDQdzRAGj3jOAYNr9kBguI7MQAAZK3caezRrg0GxAb6Hchg4lxdZEI5Oq3iov/w/hnFWY9Q==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "freebsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm-gnueabihf": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.57.1.tgz",
+ "integrity": "sha512-F8sWbhZ7tyuEfsmOxwc2giKDQzN3+kuBLPwwZGyVkLlKGdV1nvnNwYD0fKQ8+XS6hp9nY7B+ZeK01EBUE7aHaw==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm-musleabihf": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.57.1.tgz",
+ "integrity": "sha512-rGfNUfn0GIeXtBP1wL5MnzSj98+PZe/AXaGBCRmT0ts80lU5CATYGxXukeTX39XBKsxzFpEeK+Mrp9faXOlmrw==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm64-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.57.1.tgz",
+ "integrity": "sha512-MMtej3YHWeg/0klK2Qodf3yrNzz6CGjo2UntLvk2RSPlhzgLvYEB3frRvbEF2wRKh1Z2fDIg9KRPe1fawv7C+g==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-arm64-musl": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.57.1.tgz",
+ "integrity": "sha512-1a/qhaaOXhqXGpMFMET9VqwZakkljWHLmZOX48R0I/YLbhdxr1m4gtG1Hq7++VhVUmf+L3sTAf9op4JlhQ5u1Q==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-loong64-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.57.1.tgz",
+ "integrity": "sha512-QWO6RQTZ/cqYtJMtxhkRkidoNGXc7ERPbZN7dVW5SdURuLeVU7lwKMpo18XdcmpWYd0qsP1bwKPf7DNSUinhvA==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-loong64-musl": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.57.1.tgz",
+ "integrity": "sha512-xpObYIf+8gprgWaPP32xiN5RVTi/s5FCR+XMXSKmhfoJjrpRAjCuuqQXyxUa/eJTdAE6eJ+KDKaoEqjZQxh3Gw==",
+ "cpu": [
+ "loong64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-ppc64-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.57.1.tgz",
+ "integrity": "sha512-4BrCgrpZo4hvzMDKRqEaW1zeecScDCR+2nZ86ATLhAoJ5FQ+lbHVD3ttKe74/c7tNT9c6F2viwB3ufwp01Oh2w==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-ppc64-musl": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.57.1.tgz",
+ "integrity": "sha512-NOlUuzesGauESAyEYFSe3QTUguL+lvrN1HtwEEsU2rOwdUDeTMJdO5dUYl/2hKf9jWydJrO9OL/XSSf65R5+Xw==",
+ "cpu": [
+ "ppc64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-riscv64-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.57.1.tgz",
+ "integrity": "sha512-ptA88htVp0AwUUqhVghwDIKlvJMD/fmL/wrQj99PRHFRAG6Z5nbWoWG4o81Nt9FT+IuqUQi+L31ZKAFeJ5Is+A==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-riscv64-musl": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.57.1.tgz",
+ "integrity": "sha512-S51t7aMMTNdmAMPpBg7OOsTdn4tySRQvklmL3RpDRyknk87+Sp3xaumlatU+ppQ+5raY7sSTcC2beGgvhENfuw==",
+ "cpu": [
+ "riscv64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-s390x-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.57.1.tgz",
+ "integrity": "sha512-Bl00OFnVFkL82FHbEqy3k5CUCKH6OEJL54KCyx2oqsmZnFTR8IoNqBF+mjQVcRCT5sB6yOvK8A37LNm/kPJiZg==",
+ "cpu": [
+ "s390x"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-x64-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.57.1.tgz",
+ "integrity": "sha512-ABca4ceT4N+Tv/GtotnWAeXZUZuM/9AQyCyKYyKnpk4yoA7QIAuBt6Hkgpw8kActYlew2mvckXkvx0FfoInnLg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-linux-x64-musl": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.57.1.tgz",
+ "integrity": "sha512-HFps0JeGtuOR2convgRRkHCekD7j+gdAuXM+/i6kGzQtFhlCtQkpwtNzkNj6QhCDp7DRJ7+qC/1Vg2jt5iSOFw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "linux"
+ ]
+ },
+ "node_modules/@rollup/rollup-openbsd-x64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.57.1.tgz",
+ "integrity": "sha512-H+hXEv9gdVQuDTgnqD+SQffoWoc0Of59AStSzTEj/feWTBAnSfSD3+Dql1ZruJQxmykT/JVY0dE8Ka7z0DH1hw==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "openbsd"
+ ]
+ },
+ "node_modules/@rollup/rollup-openharmony-arm64": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.57.1.tgz",
+ "integrity": "sha512-4wYoDpNg6o/oPximyc/NG+mYUejZrCU2q+2w6YZqrAs2UcNUChIZXjtafAiiZSUc7On8v5NyNj34Kzj/Ltk6dQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "openharmony"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-arm64-msvc": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.57.1.tgz",
+ "integrity": "sha512-O54mtsV/6LW3P8qdTcamQmuC990HDfR71lo44oZMZlXU4tzLrbvTii87Ni9opq60ds0YzuAlEr/GNwuNluZyMQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-ia32-msvc": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.57.1.tgz",
+ "integrity": "sha512-P3dLS+IerxCT/7D2q2FYcRdWRl22dNbrbBEtxdWhXrfIMPP9lQhb5h4Du04mdl5Woq05jVCDPCMF7Ub0NAjIew==",
+ "cpu": [
+ "ia32"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-x64-gnu": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.57.1.tgz",
+ "integrity": "sha512-VMBH2eOOaKGtIJYleXsi2B8CPVADrh+TyNxJ4mWPnKfLB/DBUmzW+5m1xUrcwWoMfSLagIRpjUFeW5CO5hyciQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@rollup/rollup-win32-x64-msvc": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.57.1.tgz",
+ "integrity": "sha512-mxRFDdHIWRxg3UfIIAwCm6NzvxG0jDX/wBN6KsQFTvKFqqg9vTrWUE68qEjHt19A5wwx5X5aUi2zuZT7YR0jrA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "optional": true,
+ "os": [
+ "win32"
+ ]
+ },
+ "node_modules/@types/babel__core": {
+ "version": "7.20.5",
+ "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz",
+ "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==",
+ "dev": true,
+ "dependencies": {
+ "@babel/parser": "^7.20.7",
+ "@babel/types": "^7.20.7",
+ "@types/babel__generator": "*",
+ "@types/babel__template": "*",
+ "@types/babel__traverse": "*"
+ }
+ },
+ "node_modules/@types/babel__generator": {
+ "version": "7.27.0",
+ "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz",
+ "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==",
+ "dev": true,
+ "dependencies": {
+ "@babel/types": "^7.0.0"
+ }
+ },
+ "node_modules/@types/babel__template": {
+ "version": "7.4.4",
+ "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz",
+ "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==",
+ "dev": true,
+ "dependencies": {
+ "@babel/parser": "^7.1.0",
+ "@babel/types": "^7.0.0"
+ }
+ },
+ "node_modules/@types/babel__traverse": {
+ "version": "7.28.0",
+ "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz",
+ "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==",
+ "dev": true,
+ "dependencies": {
+ "@babel/types": "^7.28.2"
+ }
+ },
+ "node_modules/@types/estree": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
+ "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
+ "dev": true
+ },
+ "node_modules/@types/prop-types": {
+ "version": "15.7.15",
+ "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz",
+ "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==",
+ "devOptional": true
+ },
+ "node_modules/@types/react": {
+ "version": "18.3.28",
+ "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.28.tgz",
+ "integrity": "sha512-z9VXpC7MWrhfWipitjNdgCauoMLRdIILQsAEV+ZesIzBq/oUlxk0m3ApZuMFCXdnS4U7KrI+l3WRUEGQ8K1QKw==",
+ "devOptional": true,
+ "dependencies": {
+ "@types/prop-types": "*",
+ "csstype": "^3.2.2"
+ }
+ },
+ "node_modules/@types/react-dom": {
+ "version": "18.3.7",
+ "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.7.tgz",
+ "integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==",
+ "dev": true,
+ "peerDependencies": {
+ "@types/react": "^18.0.0"
+ }
+ },
+ "node_modules/@vitejs/plugin-react": {
+ "version": "4.7.0",
+ "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-4.7.0.tgz",
+ "integrity": "sha512-gUu9hwfWvvEDBBmgtAowQCojwZmJ5mcLn3aufeCsitijs3+f2NsrPtlAWIR6OPiqljl96GVCUbLe0HyqIpVaoA==",
+ "dev": true,
+ "dependencies": {
+ "@babel/core": "^7.28.0",
+ "@babel/plugin-transform-react-jsx-self": "^7.27.1",
+ "@babel/plugin-transform-react-jsx-source": "^7.27.1",
+ "@rolldown/pluginutils": "1.0.0-beta.27",
+ "@types/babel__core": "^7.20.5",
+ "react-refresh": "^0.17.0"
+ },
+ "engines": {
+ "node": "^14.18.0 || >=16.0.0"
+ },
+ "peerDependencies": {
+ "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0"
+ }
+ },
+ "node_modules/any-promise": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
+ "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==",
+ "dev": true
+ },
+ "node_modules/anymatch": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
+ "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
+ "dev": true,
+ "dependencies": {
+ "normalize-path": "^3.0.0",
+ "picomatch": "^2.0.4"
+ },
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/arg": {
+ "version": "5.0.2",
+ "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz",
+ "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==",
+ "dev": true
+ },
+ "node_modules/autoprefixer": {
+ "version": "10.4.24",
+ "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.4.24.tgz",
+ "integrity": "sha512-uHZg7N9ULTVbutaIsDRoUkoS8/h3bdsmVJYZ5l3wv8Cp/6UIIoRDm90hZ+BwxUj/hGBEzLxdHNSKuFpn8WOyZw==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/autoprefixer"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "browserslist": "^4.28.1",
+ "caniuse-lite": "^1.0.30001766",
+ "fraction.js": "^5.3.4",
+ "picocolors": "^1.1.1",
+ "postcss-value-parser": "^4.2.0"
+ },
+ "bin": {
+ "autoprefixer": "bin/autoprefixer"
+ },
+ "engines": {
+ "node": "^10 || ^12 || >=14"
+ },
+ "peerDependencies": {
+ "postcss": "^8.1.0"
+ }
+ },
+ "node_modules/baseline-browser-mapping": {
+ "version": "2.9.19",
+ "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.19.tgz",
+ "integrity": "sha512-ipDqC8FrAl/76p2SSWKSI+H9tFwm7vYqXQrItCuiVPt26Km0jS+NzSsBWAaBusvSbQcfJG+JitdMm+wZAgTYqg==",
+ "dev": true,
+ "bin": {
+ "baseline-browser-mapping": "dist/cli.js"
+ }
+ },
+ "node_modules/binary-extensions": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz",
+ "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==",
+ "dev": true,
+ "engines": {
+ "node": ">=8"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/braces": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz",
+ "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==",
+ "dev": true,
+ "dependencies": {
+ "fill-range": "^7.1.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/browserslist": {
+ "version": "4.28.1",
+ "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz",
+ "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "baseline-browser-mapping": "^2.9.0",
+ "caniuse-lite": "^1.0.30001759",
+ "electron-to-chromium": "^1.5.263",
+ "node-releases": "^2.0.27",
+ "update-browserslist-db": "^1.2.0"
+ },
+ "bin": {
+ "browserslist": "cli.js"
+ },
+ "engines": {
+ "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
+ }
+ },
+ "node_modules/camelcase-css": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz",
+ "integrity": "sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==",
+ "dev": true,
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/caniuse-lite": {
+ "version": "1.0.30001770",
+ "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001770.tgz",
+ "integrity": "sha512-x/2CLQ1jHENRbHg5PSId2sXq1CIO1CISvwWAj027ltMVG2UNgW+w9oH2+HzgEIRFembL8bUlXtfbBHR1fCg2xw==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/caniuse-lite"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ]
+ },
+ "node_modules/chokidar": {
+ "version": "3.6.0",
+ "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz",
+ "integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==",
+ "dev": true,
+ "dependencies": {
+ "anymatch": "~3.1.2",
+ "braces": "~3.0.2",
+ "glob-parent": "~5.1.2",
+ "is-binary-path": "~2.1.0",
+ "is-glob": "~4.0.1",
+ "normalize-path": "~3.0.0",
+ "readdirp": "~3.6.0"
+ },
+ "engines": {
+ "node": ">= 8.10.0"
+ },
+ "funding": {
+ "url": "https://paulmillr.com/funding/"
+ },
+ "optionalDependencies": {
+ "fsevents": "~2.3.2"
+ }
+ },
+ "node_modules/chokidar/node_modules/glob-parent": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+ "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+ "dev": true,
+ "dependencies": {
+ "is-glob": "^4.0.1"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/clsx": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz",
+ "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/commander": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz",
+ "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==",
+ "dev": true,
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/convert-source-map": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
+ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==",
+ "dev": true
+ },
+ "node_modules/cssesc": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz",
+ "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==",
+ "dev": true,
+ "bin": {
+ "cssesc": "bin/cssesc"
+ },
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/csstype": {
+ "version": "3.2.3",
+ "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
+ "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==",
+ "devOptional": true
+ },
+ "node_modules/debug": {
+ "version": "4.4.3",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
+ "dev": true,
+ "dependencies": {
+ "ms": "^2.1.3"
+ },
+ "engines": {
+ "node": ">=6.0"
+ },
+ "peerDependenciesMeta": {
+ "supports-color": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/didyoumean": {
+ "version": "1.2.2",
+ "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz",
+ "integrity": "sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==",
+ "dev": true
+ },
+ "node_modules/dlv": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz",
+ "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==",
+ "dev": true
+ },
+ "node_modules/electron-to-chromium": {
+ "version": "1.5.286",
+ "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.286.tgz",
+ "integrity": "sha512-9tfDXhJ4RKFNerfjdCcZfufu49vg620741MNs26a9+bhLThdB+plgMeou98CAaHu/WATj2iHOOHTp1hWtABj2A==",
+ "dev": true
+ },
+ "node_modules/esbuild": {
+ "version": "0.21.5",
+ "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz",
+ "integrity": "sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==",
+ "dev": true,
+ "hasInstallScript": true,
+ "bin": {
+ "esbuild": "bin/esbuild"
+ },
+ "engines": {
+ "node": ">=12"
+ },
+ "optionalDependencies": {
+ "@esbuild/aix-ppc64": "0.21.5",
+ "@esbuild/android-arm": "0.21.5",
+ "@esbuild/android-arm64": "0.21.5",
+ "@esbuild/android-x64": "0.21.5",
+ "@esbuild/darwin-arm64": "0.21.5",
+ "@esbuild/darwin-x64": "0.21.5",
+ "@esbuild/freebsd-arm64": "0.21.5",
+ "@esbuild/freebsd-x64": "0.21.5",
+ "@esbuild/linux-arm": "0.21.5",
+ "@esbuild/linux-arm64": "0.21.5",
+ "@esbuild/linux-ia32": "0.21.5",
+ "@esbuild/linux-loong64": "0.21.5",
+ "@esbuild/linux-mips64el": "0.21.5",
+ "@esbuild/linux-ppc64": "0.21.5",
+ "@esbuild/linux-riscv64": "0.21.5",
+ "@esbuild/linux-s390x": "0.21.5",
+ "@esbuild/linux-x64": "0.21.5",
+ "@esbuild/netbsd-x64": "0.21.5",
+ "@esbuild/openbsd-x64": "0.21.5",
+ "@esbuild/sunos-x64": "0.21.5",
+ "@esbuild/win32-arm64": "0.21.5",
+ "@esbuild/win32-ia32": "0.21.5",
+ "@esbuild/win32-x64": "0.21.5"
+ }
+ },
+ "node_modules/escalade": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
+ "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
+ "dev": true,
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/fast-glob": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz",
+ "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==",
+ "dev": true,
+ "dependencies": {
+ "@nodelib/fs.stat": "^2.0.2",
+ "@nodelib/fs.walk": "^1.2.3",
+ "glob-parent": "^5.1.2",
+ "merge2": "^1.3.0",
+ "micromatch": "^4.0.8"
+ },
+ "engines": {
+ "node": ">=8.6.0"
+ }
+ },
+ "node_modules/fast-glob/node_modules/glob-parent": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
+ "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
+ "dev": true,
+ "dependencies": {
+ "is-glob": "^4.0.1"
+ },
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/fastq": {
+ "version": "1.20.1",
+ "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.20.1.tgz",
+ "integrity": "sha512-GGToxJ/w1x32s/D2EKND7kTil4n8OVk/9mycTc4VDza13lOvpUZTGX3mFSCtV9ksdGBVzvsyAVLM6mHFThxXxw==",
+ "dev": true,
+ "dependencies": {
+ "reusify": "^1.0.4"
+ }
+ },
+ "node_modules/fill-range": {
+ "version": "7.1.1",
+ "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz",
+ "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==",
+ "dev": true,
+ "dependencies": {
+ "to-regex-range": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/fraction.js": {
+ "version": "5.3.4",
+ "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-5.3.4.tgz",
+ "integrity": "sha512-1X1NTtiJphryn/uLQz3whtY6jK3fTqoE3ohKs0tT+Ujr1W59oopxmoEh7Lu5p6vBaPbgoM0bzveAW4Qi5RyWDQ==",
+ "dev": true,
+ "engines": {
+ "node": "*"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/rawify"
+ }
+ },
+ "node_modules/framer-motion": {
+ "version": "11.18.2",
+ "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-11.18.2.tgz",
+ "integrity": "sha512-5F5Och7wrvtLVElIpclDT0CBzMVg3dL22B64aZwHtsIY8RB4mXICLrkajK4G9R+ieSAGcgrLeae2SeUTg2pr6w==",
+ "dependencies": {
+ "motion-dom": "^11.18.1",
+ "motion-utils": "^11.18.1",
+ "tslib": "^2.4.0"
+ },
+ "peerDependencies": {
+ "@emotion/is-prop-valid": "*",
+ "react": "^18.0.0 || ^19.0.0",
+ "react-dom": "^18.0.0 || ^19.0.0"
+ },
+ "peerDependenciesMeta": {
+ "@emotion/is-prop-valid": {
+ "optional": true
+ },
+ "react": {
+ "optional": true
+ },
+ "react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/fsevents": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
+ "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
+ "dev": true,
+ "hasInstallScript": true,
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
+ }
+ },
+ "node_modules/function-bind": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
+ "dev": true,
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/gensync": {
+ "version": "1.0.0-beta.2",
+ "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz",
+ "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==",
+ "dev": true,
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
+ "node_modules/glob-parent": {
+ "version": "6.0.2",
+ "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
+ "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
+ "dev": true,
+ "dependencies": {
+ "is-glob": "^4.0.3"
+ },
+ "engines": {
+ "node": ">=10.13.0"
+ }
+ },
+ "node_modules/hasown": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
+ "dev": true,
+ "dependencies": {
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/is-binary-path": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
+ "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
+ "dev": true,
+ "dependencies": {
+ "binary-extensions": "^2.0.0"
+ },
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/is-core-module": {
+ "version": "2.16.1",
+ "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz",
+ "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==",
+ "dev": true,
+ "dependencies": {
+ "hasown": "^2.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/is-extglob": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+ "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/is-glob": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
+ "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
+ "dev": true,
+ "dependencies": {
+ "is-extglob": "^2.1.1"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/is-number": {
+ "version": "7.0.0",
+ "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
+ "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.12.0"
+ }
+ },
+ "node_modules/jiti": {
+ "version": "1.21.7",
+ "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.7.tgz",
+ "integrity": "sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==",
+ "dev": true,
+ "bin": {
+ "jiti": "bin/jiti.js"
+ }
+ },
+ "node_modules/js-tokens": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
+ "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ=="
+ },
+ "node_modules/jsesc": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz",
+ "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==",
+ "dev": true,
+ "bin": {
+ "jsesc": "bin/jsesc"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/json5": {
+ "version": "2.2.3",
+ "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
+ "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
+ "dev": true,
+ "bin": {
+ "json5": "lib/cli.js"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/lilconfig": {
+ "version": "3.1.3",
+ "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-3.1.3.tgz",
+ "integrity": "sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw==",
+ "dev": true,
+ "engines": {
+ "node": ">=14"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/antonk52"
+ }
+ },
+ "node_modules/lines-and-columns": {
+ "version": "1.2.4",
+ "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz",
+ "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==",
+ "dev": true
+ },
+ "node_modules/loose-envify": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz",
+ "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==",
+ "dependencies": {
+ "js-tokens": "^3.0.0 || ^4.0.0"
+ },
+ "bin": {
+ "loose-envify": "cli.js"
+ }
+ },
+ "node_modules/lru-cache": {
+ "version": "5.1.1",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
+ "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
+ "dev": true,
+ "dependencies": {
+ "yallist": "^3.0.2"
+ }
+ },
+ "node_modules/lucide-react": {
+ "version": "0.323.0",
+ "resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-0.323.0.tgz",
+ "integrity": "sha512-rTXZFILl2Y4d1SG9p1Mdcf17AcPvPvpc/egFIzUrp7IUy60MUQo3Oi1mu8LGYXUVwuRZYsSMt3csHRW5mAovJg==",
+ "peerDependencies": {
+ "react": "^16.5.1 || ^17.0.0 || ^18.0.0"
+ }
+ },
+ "node_modules/merge2": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
+ "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==",
+ "dev": true,
+ "engines": {
+ "node": ">= 8"
+ }
+ },
+ "node_modules/micromatch": {
+ "version": "4.0.8",
+ "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz",
+ "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==",
+ "dev": true,
+ "dependencies": {
+ "braces": "^3.0.3",
+ "picomatch": "^2.3.1"
+ },
+ "engines": {
+ "node": ">=8.6"
+ }
+ },
+ "node_modules/motion-dom": {
+ "version": "11.18.1",
+ "resolved": "https://registry.npmjs.org/motion-dom/-/motion-dom-11.18.1.tgz",
+ "integrity": "sha512-g76KvA001z+atjfxczdRtw/RXOM3OMSdd1f4DL77qCTF/+avrRJiawSG4yDibEQ215sr9kpinSlX2pCTJ9zbhw==",
+ "dependencies": {
+ "motion-utils": "^11.18.1"
+ }
+ },
+ "node_modules/motion-utils": {
+ "version": "11.18.1",
+ "resolved": "https://registry.npmjs.org/motion-utils/-/motion-utils-11.18.1.tgz",
+ "integrity": "sha512-49Kt+HKjtbJKLtgO/LKj9Ld+6vw9BjH5d9sc40R/kVyH8GLAXgT42M2NnuPcJNuA3s9ZfZBUcwIgpmZWGEE+hA=="
+ },
+ "node_modules/ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+ "dev": true
+ },
+ "node_modules/mz": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
+ "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==",
+ "dev": true,
+ "dependencies": {
+ "any-promise": "^1.0.0",
+ "object-assign": "^4.0.1",
+ "thenify-all": "^1.0.0"
+ }
+ },
+ "node_modules/nanoid": {
+ "version": "3.3.11",
+ "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
+ "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "bin": {
+ "nanoid": "bin/nanoid.cjs"
+ },
+ "engines": {
+ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
+ }
+ },
+ "node_modules/node-releases": {
+ "version": "2.0.27",
+ "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz",
+ "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==",
+ "dev": true
+ },
+ "node_modules/normalize-path": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
+ "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/object-hash": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz",
+ "integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==",
+ "dev": true,
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/path-parse": {
+ "version": "1.0.7",
+ "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
+ "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
+ "dev": true
+ },
+ "node_modules/picocolors": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
+ "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
+ "dev": true
+ },
+ "node_modules/picomatch": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz",
+ "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==",
+ "dev": true,
+ "engines": {
+ "node": ">=8.6"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
+ "node_modules/pify": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
+ "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/pirates": {
+ "version": "4.0.7",
+ "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz",
+ "integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==",
+ "dev": true,
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/postcss": {
+ "version": "8.5.6",
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz",
+ "integrity": "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/postcss"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "nanoid": "^3.3.11",
+ "picocolors": "^1.1.1",
+ "source-map-js": "^1.2.1"
+ },
+ "engines": {
+ "node": "^10 || ^12 || >=14"
+ }
+ },
+ "node_modules/postcss-import": {
+ "version": "15.1.0",
+ "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz",
+ "integrity": "sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==",
+ "dev": true,
+ "dependencies": {
+ "postcss-value-parser": "^4.0.0",
+ "read-cache": "^1.0.0",
+ "resolve": "^1.1.7"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ },
+ "peerDependencies": {
+ "postcss": "^8.0.0"
+ }
+ },
+ "node_modules/postcss-js": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/postcss-js/-/postcss-js-4.1.0.tgz",
+ "integrity": "sha512-oIAOTqgIo7q2EOwbhb8UalYePMvYoIeRY2YKntdpFQXNosSu3vLrniGgmH9OKs/qAkfoj5oB3le/7mINW1LCfw==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "camelcase-css": "^2.0.1"
+ },
+ "engines": {
+ "node": "^12 || ^14 || >= 16"
+ },
+ "peerDependencies": {
+ "postcss": "^8.4.21"
+ }
+ },
+ "node_modules/postcss-load-config": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-6.0.1.tgz",
+ "integrity": "sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "lilconfig": "^3.1.1"
+ },
+ "engines": {
+ "node": ">= 18"
+ },
+ "peerDependencies": {
+ "jiti": ">=1.21.0",
+ "postcss": ">=8.0.9",
+ "tsx": "^4.8.1",
+ "yaml": "^2.4.2"
+ },
+ "peerDependenciesMeta": {
+ "jiti": {
+ "optional": true
+ },
+ "postcss": {
+ "optional": true
+ },
+ "tsx": {
+ "optional": true
+ },
+ "yaml": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/postcss-nested": {
+ "version": "6.2.0",
+ "resolved": "https://registry.npmjs.org/postcss-nested/-/postcss-nested-6.2.0.tgz",
+ "integrity": "sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/postcss/"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "postcss-selector-parser": "^6.1.1"
+ },
+ "engines": {
+ "node": ">=12.0"
+ },
+ "peerDependencies": {
+ "postcss": "^8.2.14"
+ }
+ },
+ "node_modules/postcss-selector-parser": {
+ "version": "6.1.2",
+ "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz",
+ "integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==",
+ "dev": true,
+ "dependencies": {
+ "cssesc": "^3.0.0",
+ "util-deprecate": "^1.0.2"
+ },
+ "engines": {
+ "node": ">=4"
+ }
+ },
+ "node_modules/postcss-value-parser": {
+ "version": "4.2.0",
+ "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz",
+ "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==",
+ "dev": true
+ },
+ "node_modules/queue-microtask": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
+ "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ]
+ },
+ "node_modules/react": {
+ "version": "18.3.1",
+ "resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz",
+ "integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==",
+ "dependencies": {
+ "loose-envify": "^1.1.0"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/react-dom": {
+ "version": "18.3.1",
+ "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz",
+ "integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==",
+ "dependencies": {
+ "loose-envify": "^1.1.0",
+ "scheduler": "^0.23.2"
+ },
+ "peerDependencies": {
+ "react": "^18.3.1"
+ }
+ },
+ "node_modules/react-refresh": {
+ "version": "0.17.0",
+ "resolved": "https://registry.npmjs.org/react-refresh/-/react-refresh-0.17.0.tgz",
+ "integrity": "sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/react-router": {
+ "version": "6.30.3",
+ "resolved": "https://registry.npmjs.org/react-router/-/react-router-6.30.3.tgz",
+ "integrity": "sha512-XRnlbKMTmktBkjCLE8/XcZFlnHvr2Ltdr1eJX4idL55/9BbORzyZEaIkBFDhFGCEWBBItsVrDxwx3gnisMitdw==",
+ "dependencies": {
+ "@remix-run/router": "1.23.2"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.8"
+ }
+ },
+ "node_modules/react-router-dom": {
+ "version": "6.30.3",
+ "resolved": "https://registry.npmjs.org/react-router-dom/-/react-router-dom-6.30.3.tgz",
+ "integrity": "sha512-pxPcv1AczD4vso7G4Z3TKcvlxK7g7TNt3/FNGMhfqyntocvYKj+GCatfigGDjbLozC4baguJ0ReCigoDJXb0ag==",
+ "dependencies": {
+ "@remix-run/router": "1.23.2",
+ "react-router": "6.30.3"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.8",
+ "react-dom": ">=16.8"
+ }
+ },
+ "node_modules/read-cache": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
+ "integrity": "sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==",
+ "dev": true,
+ "dependencies": {
+ "pify": "^2.3.0"
+ }
+ },
+ "node_modules/readdirp": {
+ "version": "3.6.0",
+ "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
+ "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
+ "dev": true,
+ "dependencies": {
+ "picomatch": "^2.2.1"
+ },
+ "engines": {
+ "node": ">=8.10.0"
+ }
+ },
+ "node_modules/resolve": {
+ "version": "1.22.11",
+ "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz",
+ "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==",
+ "dev": true,
+ "dependencies": {
+ "is-core-module": "^2.16.1",
+ "path-parse": "^1.0.7",
+ "supports-preserve-symlinks-flag": "^1.0.0"
+ },
+ "bin": {
+ "resolve": "bin/resolve"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/reusify": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz",
+ "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==",
+ "dev": true,
+ "engines": {
+ "iojs": ">=1.0.0",
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/rollup": {
+ "version": "4.57.1",
+ "resolved": "https://registry.npmjs.org/rollup/-/rollup-4.57.1.tgz",
+ "integrity": "sha512-oQL6lgK3e2QZeQ7gcgIkS2YZPg5slw37hYufJ3edKlfQSGGm8ICoxswK15ntSzF/a8+h7ekRy7k7oWc3BQ7y8A==",
+ "dev": true,
+ "dependencies": {
+ "@types/estree": "1.0.8"
+ },
+ "bin": {
+ "rollup": "dist/bin/rollup"
+ },
+ "engines": {
+ "node": ">=18.0.0",
+ "npm": ">=8.0.0"
+ },
+ "optionalDependencies": {
+ "@rollup/rollup-android-arm-eabi": "4.57.1",
+ "@rollup/rollup-android-arm64": "4.57.1",
+ "@rollup/rollup-darwin-arm64": "4.57.1",
+ "@rollup/rollup-darwin-x64": "4.57.1",
+ "@rollup/rollup-freebsd-arm64": "4.57.1",
+ "@rollup/rollup-freebsd-x64": "4.57.1",
+ "@rollup/rollup-linux-arm-gnueabihf": "4.57.1",
+ "@rollup/rollup-linux-arm-musleabihf": "4.57.1",
+ "@rollup/rollup-linux-arm64-gnu": "4.57.1",
+ "@rollup/rollup-linux-arm64-musl": "4.57.1",
+ "@rollup/rollup-linux-loong64-gnu": "4.57.1",
+ "@rollup/rollup-linux-loong64-musl": "4.57.1",
+ "@rollup/rollup-linux-ppc64-gnu": "4.57.1",
+ "@rollup/rollup-linux-ppc64-musl": "4.57.1",
+ "@rollup/rollup-linux-riscv64-gnu": "4.57.1",
+ "@rollup/rollup-linux-riscv64-musl": "4.57.1",
+ "@rollup/rollup-linux-s390x-gnu": "4.57.1",
+ "@rollup/rollup-linux-x64-gnu": "4.57.1",
+ "@rollup/rollup-linux-x64-musl": "4.57.1",
+ "@rollup/rollup-openbsd-x64": "4.57.1",
+ "@rollup/rollup-openharmony-arm64": "4.57.1",
+ "@rollup/rollup-win32-arm64-msvc": "4.57.1",
+ "@rollup/rollup-win32-ia32-msvc": "4.57.1",
+ "@rollup/rollup-win32-x64-gnu": "4.57.1",
+ "@rollup/rollup-win32-x64-msvc": "4.57.1",
+ "fsevents": "~2.3.2"
+ }
+ },
+ "node_modules/run-parallel": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz",
+ "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "dependencies": {
+ "queue-microtask": "^1.2.2"
+ }
+ },
+ "node_modules/scheduler": {
+ "version": "0.23.2",
+ "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz",
+ "integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==",
+ "dependencies": {
+ "loose-envify": "^1.1.0"
+ }
+ },
+ "node_modules/semver": {
+ "version": "6.3.1",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
+ "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
+ "dev": true,
+ "bin": {
+ "semver": "bin/semver.js"
+ }
+ },
+ "node_modules/source-map-js": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
+ "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
+ "dev": true,
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/sucrase": {
+ "version": "3.35.1",
+ "resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.35.1.tgz",
+ "integrity": "sha512-DhuTmvZWux4H1UOnWMB3sk0sbaCVOoQZjv8u1rDoTV0HTdGem9hkAZtl4JZy8P2z4Bg0nT+YMeOFyVr4zcG5Tw==",
+ "dev": true,
+ "dependencies": {
+ "@jridgewell/gen-mapping": "^0.3.2",
+ "commander": "^4.0.0",
+ "lines-and-columns": "^1.1.6",
+ "mz": "^2.7.0",
+ "pirates": "^4.0.1",
+ "tinyglobby": "^0.2.11",
+ "ts-interface-checker": "^0.1.9"
+ },
+ "bin": {
+ "sucrase": "bin/sucrase",
+ "sucrase-node": "bin/sucrase-node"
+ },
+ "engines": {
+ "node": ">=16 || 14 >=14.17"
+ }
+ },
+ "node_modules/supports-preserve-symlinks-flag": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
+ "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
+ "dev": true,
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/tailwindcss": {
+ "version": "3.4.19",
+ "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.19.tgz",
+ "integrity": "sha512-3ofp+LL8E+pK/JuPLPggVAIaEuhvIz4qNcf3nA1Xn2o/7fb7s/TYpHhwGDv1ZU3PkBluUVaF8PyCHcm48cKLWQ==",
+ "dev": true,
+ "dependencies": {
+ "@alloc/quick-lru": "^5.2.0",
+ "arg": "^5.0.2",
+ "chokidar": "^3.6.0",
+ "didyoumean": "^1.2.2",
+ "dlv": "^1.1.3",
+ "fast-glob": "^3.3.2",
+ "glob-parent": "^6.0.2",
+ "is-glob": "^4.0.3",
+ "jiti": "^1.21.7",
+ "lilconfig": "^3.1.3",
+ "micromatch": "^4.0.8",
+ "normalize-path": "^3.0.0",
+ "object-hash": "^3.0.0",
+ "picocolors": "^1.1.1",
+ "postcss": "^8.4.47",
+ "postcss-import": "^15.1.0",
+ "postcss-js": "^4.0.1",
+ "postcss-load-config": "^4.0.2 || ^5.0 || ^6.0",
+ "postcss-nested": "^6.2.0",
+ "postcss-selector-parser": "^6.1.2",
+ "resolve": "^1.22.8",
+ "sucrase": "^3.35.0"
+ },
+ "bin": {
+ "tailwind": "lib/cli.js",
+ "tailwindcss": "lib/cli.js"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ }
+ },
+ "node_modules/thenify": {
+ "version": "3.3.1",
+ "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
+ "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==",
+ "dev": true,
+ "dependencies": {
+ "any-promise": "^1.0.0"
+ }
+ },
+ "node_modules/thenify-all": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz",
+ "integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==",
+ "dev": true,
+ "dependencies": {
+ "thenify": ">= 3.1.0 < 4"
+ },
+ "engines": {
+ "node": ">=0.8"
+ }
+ },
+ "node_modules/tinyglobby": {
+ "version": "0.2.15",
+ "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.15.tgz",
+ "integrity": "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ==",
+ "dev": true,
+ "dependencies": {
+ "fdir": "^6.5.0",
+ "picomatch": "^4.0.3"
+ },
+ "engines": {
+ "node": ">=12.0.0"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/SuperchupuDev"
+ }
+ },
+ "node_modules/tinyglobby/node_modules/fdir": {
+ "version": "6.5.0",
+ "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
+ "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
+ "dev": true,
+ "engines": {
+ "node": ">=12.0.0"
+ },
+ "peerDependencies": {
+ "picomatch": "^3 || ^4"
+ },
+ "peerDependenciesMeta": {
+ "picomatch": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/tinyglobby/node_modules/picomatch": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz",
+ "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==",
+ "dev": true,
+ "engines": {
+ "node": ">=12"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/jonschlinkert"
+ }
+ },
+ "node_modules/to-regex-range": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
+ "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
+ "dev": true,
+ "dependencies": {
+ "is-number": "^7.0.0"
+ },
+ "engines": {
+ "node": ">=8.0"
+ }
+ },
+ "node_modules/ts-interface-checker": {
+ "version": "0.1.13",
+ "resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz",
+ "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==",
+ "dev": true
+ },
+ "node_modules/tslib": {
+ "version": "2.8.1",
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
+ "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="
+ },
+ "node_modules/typescript": {
+ "version": "5.9.3",
+ "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
+ "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==",
+ "dev": true,
+ "bin": {
+ "tsc": "bin/tsc",
+ "tsserver": "bin/tsserver"
+ },
+ "engines": {
+ "node": ">=14.17"
+ }
+ },
+ "node_modules/update-browserslist-db": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz",
+ "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==",
+ "dev": true,
+ "funding": [
+ {
+ "type": "opencollective",
+ "url": "https://opencollective.com/browserslist"
+ },
+ {
+ "type": "tidelift",
+ "url": "https://tidelift.com/funding/github/npm/browserslist"
+ },
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/ai"
+ }
+ ],
+ "dependencies": {
+ "escalade": "^3.2.0",
+ "picocolors": "^1.1.1"
+ },
+ "bin": {
+ "update-browserslist-db": "cli.js"
+ },
+ "peerDependencies": {
+ "browserslist": ">= 4.21.0"
+ }
+ },
+ "node_modules/use-sync-external-store": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmjs.org/use-sync-external-store/-/use-sync-external-store-1.6.0.tgz",
+ "integrity": "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==",
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/util-deprecate": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
+ "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
+ "dev": true
+ },
+ "node_modules/vite": {
+ "version": "5.4.21",
+ "resolved": "https://registry.npmjs.org/vite/-/vite-5.4.21.tgz",
+ "integrity": "sha512-o5a9xKjbtuhY6Bi5S3+HvbRERmouabWbyUcpXXUA1u+GNUKoROi9byOJ8M0nHbHYHkYICiMlqxkg1KkYmm25Sw==",
+ "dev": true,
+ "dependencies": {
+ "esbuild": "^0.21.3",
+ "postcss": "^8.4.43",
+ "rollup": "^4.20.0"
+ },
+ "bin": {
+ "vite": "bin/vite.js"
+ },
+ "engines": {
+ "node": "^18.0.0 || >=20.0.0"
+ },
+ "funding": {
+ "url": "https://github.com/vitejs/vite?sponsor=1"
+ },
+ "optionalDependencies": {
+ "fsevents": "~2.3.3"
+ },
+ "peerDependencies": {
+ "@types/node": "^18.0.0 || >=20.0.0",
+ "less": "*",
+ "lightningcss": "^1.21.0",
+ "sass": "*",
+ "sass-embedded": "*",
+ "stylus": "*",
+ "sugarss": "*",
+ "terser": "^5.4.0"
+ },
+ "peerDependenciesMeta": {
+ "@types/node": {
+ "optional": true
+ },
+ "less": {
+ "optional": true
+ },
+ "lightningcss": {
+ "optional": true
+ },
+ "sass": {
+ "optional": true
+ },
+ "sass-embedded": {
+ "optional": true
+ },
+ "stylus": {
+ "optional": true
+ },
+ "sugarss": {
+ "optional": true
+ },
+ "terser": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/yallist": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+ "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+ "dev": true
+ },
+ "node_modules/zustand": {
+ "version": "4.5.7",
+ "resolved": "https://registry.npmjs.org/zustand/-/zustand-4.5.7.tgz",
+ "integrity": "sha512-CHOUy7mu3lbD6o6LJLfllpjkzhHXSBlX8B9+qPddUsIfeF5S/UZ5q0kmCsnRqT1UHFQZchNFDDzMbQsuesHWlw==",
+ "dependencies": {
+ "use-sync-external-store": "^1.2.2"
+ },
+ "engines": {
+ "node": ">=12.7.0"
+ },
+ "peerDependencies": {
+ "@types/react": ">=16.8",
+ "immer": ">=9.0.6",
+ "react": ">=16.8"
+ },
+ "peerDependenciesMeta": {
+ "@types/react": {
+ "optional": true
+ },
+ "immer": {
+ "optional": true
+ },
+ "react": {
+ "optional": true
+ }
+ }
+ }
+ }
+}
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..ddad5cf
--- /dev/null
+++ b/package.json
@@ -0,0 +1,31 @@
+{
+ "name": "rushe-ui",
+ "private": true,
+ "version": "0.0.0",
+ "type": "module",
+ "scripts": {
+ "dev": "vite",
+ "build": "tsc && vite build",
+ "lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0",
+ "preview": "vite preview"
+ },
+ "dependencies": {
+ "react": "^18.2.0",
+ "react-dom": "^18.2.0",
+ "react-router-dom": "^6.22.0",
+ "zustand": "^4.5.0",
+ "framer-motion": "^11.0.0",
+ "lucide-react": "^0.323.0",
+ "clsx": "^2.1.0"
+ },
+ "devDependencies": {
+ "@types/react": "^18.2.55",
+ "@types/react-dom": "^18.2.19",
+ "@vitejs/plugin-react": "^4.2.1",
+ "autoprefixer": "^10.4.17",
+ "postcss": "^8.4.35",
+ "tailwindcss": "^3.4.1",
+ "typescript": "^5.2.2",
+ "vite": "^5.1.0"
+ }
+}
\ No newline at end of file
diff --git a/postcss.config.js b/postcss.config.js
new file mode 100644
index 0000000..2e7af2b
--- /dev/null
+++ b/postcss.config.js
@@ -0,0 +1,6 @@
+export default {
+ plugins: {
+ tailwindcss: {},
+ autoprefixer: {},
+ },
+}
diff --git a/src/App.tsx b/src/App.tsx
new file mode 100644
index 0000000..57098ef
--- /dev/null
+++ b/src/App.tsx
@@ -0,0 +1,61 @@
+import { Navigate, Route, Routes } from 'react-router-dom'
+import AdminLayout from './layouts/AdminLayout'
+import UserLayout from './layouts/UserLayout'
+import AdminMinePage from './pages/admin/mine/MinePage'
+import ToolboxPage from './pages/admin/toolbox/ToolboxPage'
+import WorkspacePage from './pages/admin/workspace/WorkspacePage'
+import AIBudgetPage from './pages/user/ai/AIBudgetPage'
+import AIPage from './pages/user/ai/AIPage'
+import BudgetDetailPage from './pages/user/ai/BudgetDetailPage'
+import ExplorePage from './pages/user/explore/ExplorePage'
+import BusBookingPage from './pages/user/group/BusBookingPage'
+import GroupBuyPage from './pages/user/group/GroupBuyPage'
+import GroupStorePage from './pages/user/group/GroupStorePage'
+import PackageDetailPage from './pages/user/group/PackageDetailPage'
+import BalconyPage from './pages/user/home/BalconyPage'
+import DesignerPage from './pages/user/home/DesignerPage'
+import HomePage from './pages/user/home/HomePage'
+import HomePage2 from './pages/user/home/HomePage2'
+import DataAssetsPage from './pages/user/mine/DataAssetsPage'
+import DesignSchemesPage from './pages/user/mine/DesignSchemesPage'
+import MinePage from './pages/user/mine/MinePage'
+import PartnerCenterPage from './pages/user/mine/PartnerCenterPage'
+import SettingsPage from './pages/user/mine/SettingsPage'
+import ProgressPage from './pages/user/progress/ProgressPage'
+
+function App() {
+ return (
+
+ } />
+
+ }>
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+ } />
+
+
+ }>
+ } />
+ } />
+ } />
+
+
+ )
+}
+
+export default App
diff --git a/src/components/AIChatModal.tsx b/src/components/AIChatModal.tsx
new file mode 100644
index 0000000..a4f6001
--- /dev/null
+++ b/src/components/AIChatModal.tsx
@@ -0,0 +1,227 @@
+import { AnimatePresence, motion } from 'framer-motion'
+import { Calculator, Send, Sparkles, X } from 'lucide-react'
+import { useRef, useState } from 'react'
+import { useNavigate } from 'react-router-dom'
+
+interface Message {
+ id: number
+ type: 'ai' | 'user'
+ content: string
+ time: string
+}
+
+interface AIChatModalProps {
+ isOpen: boolean
+ onClose: () => void
+}
+
+const initialMessages: Message[] = [
+ {
+ id: 1,
+ type: 'ai',
+ content: '您好!我是入舍智能家装助手,可以为您提供装修咨询、预算估算、风格推荐等服务。有什么可以帮助您的吗?',
+ time: '刚刚',
+ },
+]
+
+const quickQuestions = [
+ '装修大概需要多少钱?',
+ '现代简约风格特点',
+ '如何选择装修材料?',
+ '小户型怎么设计?',
+]
+
+export default function AIChatModal({ isOpen, onClose }: AIChatModalProps) {
+ const navigate = useNavigate()
+ const [messages, setMessages] = useState(initialMessages)
+ const [inputValue, setInputValue] = useState('')
+ const [isTyping, setIsTyping] = useState(false)
+ const messagesEndRef = useRef(null)
+
+ const scrollToBottom = () => {
+ messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' })
+ }
+
+ const handleSend = () => {
+ if (!inputValue.trim()) return
+
+ const userMessage: Message = {
+ id: Date.now(),
+ type: 'user',
+ content: inputValue,
+ time: '刚刚',
+ }
+
+ setMessages((prev) => [...prev, userMessage])
+ setInputValue('')
+ setIsTyping(true)
+
+ setTimeout(() => {
+ const aiResponses = [
+ '根据您的需求,我建议您可以先了解一下我们的装修套餐,从基础装修到全屋定制,价格区间在5万到30万不等。您想了解哪个档次的装修方案呢?',
+ '这是一个很好的问题!现代简约风格注重简洁的线条和功能性,通常使用中性色调,搭配少量亮色点缀。家具选择以实用为主,避免过多装饰。',
+ '选择装修材料时,建议关注以下几点:1. 环保等级(E0/E1级)2. 耐用性 3. 性价比 4. 售后服务。我们可以为您提供一线品牌材料,品质有保障。',
+ '小户型设计的关键是"轻装修、重装饰",建议:1. 选择浅色系墙面 2. 利用镜面增加空间感 3. 定制收纳家具 4. 选择多功能家具。',
+ ]
+
+ const aiMessage: Message = {
+ id: Date.now(),
+ type: 'ai',
+ content: aiResponses[Math.floor(Math.random() * aiResponses.length)],
+ time: '刚刚',
+ }
+
+ setMessages((prev) => [...prev, aiMessage])
+ setIsTyping(false)
+ setTimeout(scrollToBottom, 100)
+ }, 1500)
+
+ setTimeout(scrollToBottom, 100)
+ }
+
+ const handleQuickQuestion = (question: string) => {
+ setInputValue(question)
+ }
+
+ const handleBudgetClick = () => {
+ onClose()
+ navigate('/user/ai/budget')
+ }
+
+ return (
+
+ {isOpen && (
+ <>
+
+
+
+
+
+
+
+
+
+
+
+ {messages.map((message) => (
+
+ {message.type === 'ai' && (
+
+
+
+ )}
+
+
{message.content}
+
+ {message.time}
+
+
+
+ ))}
+
+ {isTyping && (
+
+
+
+
+
+
+ )}
+
+
+
+
+
+
+ {quickQuestions.map((question, index) => (
+
+ ))}
+
+
+
+ setInputValue(e.target.value)}
+ onKeyPress={(e) => e.key === 'Enter' && handleSend()}
+ placeholder="输入您的问题..."
+ className="flex-1 px-4 py-2 rounded-full text-sm outline-none"
+ style={{ backgroundColor: 'var(--bg-secondary)', color: 'var(--text-primary)' }}
+ />
+
+
+
+
+ >
+ )}
+
+ )
+}
diff --git a/src/components/NotificationModal.tsx b/src/components/NotificationModal.tsx
new file mode 100644
index 0000000..aa16c99
--- /dev/null
+++ b/src/components/NotificationModal.tsx
@@ -0,0 +1,198 @@
+import { AnimatePresence, motion } from 'framer-motion'
+import { Bell, Check, Filter, Package, Percent, Settings, Trash2, X } from 'lucide-react'
+import { useState } from 'react'
+import { notifications as initialNotifications } from '@/mock/homeData'
+
+interface Notification {
+ id: number
+ type: string
+ title: string
+ content: string
+ time: string
+ isRead: boolean
+}
+
+interface NotificationModalProps {
+ isOpen: boolean
+ onClose: () => void
+}
+
+const typeConfig: Record = {
+ order: { icon: , color: 'bg-blue-500', label: '订单' },
+ promotion: { icon: , color: 'bg-orange-500', label: '优惠' },
+ system: { icon: , color: 'bg-gray-500', label: '系统' },
+ design: { icon: , color: 'bg-purple-500', label: '设计' },
+}
+
+export default function NotificationModal({ isOpen, onClose }: NotificationModalProps) {
+ const [notifications, setNotifications] = useState(initialNotifications)
+ const [activeFilter, setActiveFilter] = useState('all')
+
+ const filters = [
+ { key: 'all', label: '全部' },
+ { key: 'unread', label: '未读' },
+ { key: 'order', label: '订单' },
+ { key: 'promotion', label: '优惠' },
+ { key: 'system', label: '系统' },
+ ]
+
+ const filteredNotifications = notifications.filter((n) => {
+ if (activeFilter === 'all') return true
+ if (activeFilter === 'unread') return !n.isRead
+ return n.type === activeFilter
+ })
+
+ const unreadCount = notifications.filter((n) => !n.isRead).length
+
+ const markAsRead = (id: number) => {
+ setNotifications((prev) =>
+ prev.map((n) => (n.id === id ? { ...n, isRead: true } : n))
+ )
+ }
+
+ const markAllAsRead = () => {
+ setNotifications((prev) => prev.map((n) => ({ ...n, isRead: true })))
+ }
+
+ const deleteNotification = (id: number) => {
+ setNotifications((prev) => prev.filter((n) => n.id !== id))
+ }
+
+ return (
+
+ {isOpen && (
+ <>
+
+
+
+
+
+
通知中心
+ {unreadCount > 0 && (
+
+ {unreadCount}
+
+ )}
+
+
+ {unreadCount > 0 && (
+
+ )}
+
+
+
+
+
+
+ {filters.map((filter) => (
+
+ ))}
+
+
+
+ {filteredNotifications.length === 0 ? (
+
+ ) : (
+
+ {filteredNotifications.map((notification, index) => (
+
+ {!notification.isRead && (
+
+ )}
+
+
+ {typeConfig[notification.type]?.icon || }
+
+
+
+
+ {notification.title}
+
+
+ {notification.time.split(' ')[1]}
+
+
+
+ {notification.content}
+
+
+ {!notification.isRead && (
+
+ )}
+
+
+
+
+
+ ))}
+
+ )}
+
+
+
+ >
+ )}
+
+ )
+}
diff --git a/src/components/ThemeProvider.tsx b/src/components/ThemeProvider.tsx
new file mode 100644
index 0000000..48e3f3b
--- /dev/null
+++ b/src/components/ThemeProvider.tsx
@@ -0,0 +1,33 @@
+import { useEffect } from 'react'
+import { useThemeStore, ThemeMode } from '@/store/themeStore'
+
+interface ThemeProviderProps {
+ children: React.ReactNode
+}
+
+const getThemeClass = (theme: ThemeMode): string => {
+ switch (theme) {
+ case 'dark':
+ return 'dark'
+ case 'deepBlue':
+ return 'deep-blue'
+ default:
+ return ''
+ }
+}
+
+export default function ThemeProvider({ children }: ThemeProviderProps) {
+ const { theme } = useThemeStore()
+
+ useEffect(() => {
+ const root = document.documentElement
+ root.classList.remove('dark', 'deep-blue')
+
+ const themeClass = getThemeClass(theme)
+ if (themeClass) {
+ root.classList.add(themeClass)
+ }
+ }, [theme])
+
+ return <>{children}>
+}
diff --git a/src/index.css b/src/index.css
new file mode 100644
index 0000000..9089903
--- /dev/null
+++ b/src/index.css
@@ -0,0 +1,128 @@
+@tailwind base;
+@tailwind components;
+@tailwind utilities;
+
+:root {
+ --safe-area-inset-top: env(safe-area-inset-top);
+ --safe-area-inset-bottom: env(safe-area-inset-bottom);
+}
+
+:root {
+ --bg-primary: #ffffff;
+ --bg-secondary: #f5f5f5;
+ --bg-tertiary: #f9fafb;
+ --text-primary: #1f2937;
+ --text-secondary: #6b7280;
+ --text-tertiary: #9ca3af;
+ --border-color: #e5e7eb;
+ --shadow-color: rgba(0, 0, 0, 0.05);
+ --gradient-start: #f97316;
+ --gradient-end: #ea580c;
+}
+
+.dark {
+ --bg-primary: #1f2937;
+ --bg-secondary: #111827;
+ --bg-tertiary: #374151;
+ --text-primary: #f9fafb;
+ --text-secondary: #d1d5db;
+ --text-tertiary: #9ca3af;
+ --border-color: #374151;
+ --shadow-color: rgba(0, 0, 0, 0.3);
+ --gradient-start: #f97316;
+ --gradient-end: #ea580c;
+}
+
+.deep-blue {
+ --bg-primary: #0f172a;
+ --bg-secondary: #020617;
+ --bg-tertiary: #1e293b;
+ --text-primary: #f1f5f9;
+ --text-secondary: #cbd5e1;
+ --text-tertiary: #94a3b8;
+ --border-color: #334155;
+ --shadow-color: rgba(0, 0, 0, 0.4);
+ --gradient-start: #3b82f6;
+ --gradient-end: #2563eb;
+}
+
+* {
+ margin: 0;
+ padding: 0;
+ box-sizing: border-box;
+ -webkit-tap-highlight-color: transparent;
+}
+
+html,
+body {
+ height: 100%;
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'PingFang SC', 'Microsoft YaHei', 'Helvetica Neue', Arial, sans-serif;
+ background-color: var(--bg-secondary);
+ color: var(--text-primary);
+ line-height: 1.5;
+ -webkit-font-smoothing: antialiased;
+ -moz-osx-font-smoothing: grayscale;
+ transition: background-color 0.3s ease, color 0.3s ease;
+}
+
+#root {
+ height: 100%;
+}
+
+.page-container {
+ min-height: 100vh;
+ padding-bottom: calc(60px + env(safe-area-inset-bottom));
+ background-color: var(--bg-secondary);
+ transition: background-color 0.3s ease;
+}
+
+.page-content {
+ padding: 12px;
+}
+
+.card {
+ background: var(--bg-primary);
+ border-radius: 12px;
+ padding: 16px;
+ margin-bottom: 12px;
+ box-shadow: 0 1px 3px var(--shadow-color);
+ transition: background-color 0.3s ease, box-shadow 0.3s ease;
+}
+
+.btn-primary {
+ @apply bg-primary-500 text-white px-4 py-2 rounded-lg font-medium transition-all duration-200;
+ @apply hover:bg-primary-600 active:scale-95;
+}
+
+.btn-secondary {
+ @apply bg-gray-100 text-gray-700 px-4 py-2 rounded-lg font-medium transition-all duration-200;
+ @apply hover:bg-gray-200 active:scale-95;
+}
+
+.input-field {
+ @apply w-full px-4 py-3 border border-gray-200 rounded-lg text-base;
+ @apply focus:outline-none focus:border-primary-500 focus:ring-1 focus:ring-primary-500;
+}
+
+@layer utilities {
+ .text-gradient {
+ @apply bg-gradient-to-r from-primary-500 to-primary-600 bg-clip-text text-transparent;
+ }
+
+ .safe-bottom {
+ padding-bottom: env(safe-area-inset-bottom);
+ }
+
+ .safe-top {
+ padding-top: env(safe-area-inset-top);
+ }
+
+ .scrollbar-hide {
+ -ms-overflow-style: none;
+ scrollbar-width: none;
+ }
+
+ .scrollbar-hide::-webkit-scrollbar {
+ display: none;
+ }
+}
\ No newline at end of file
diff --git a/src/layouts/AdminLayout.tsx b/src/layouts/AdminLayout.tsx
new file mode 100644
index 0000000..4fbd75c
--- /dev/null
+++ b/src/layouts/AdminLayout.tsx
@@ -0,0 +1,60 @@
+import { Outlet, useLocation, useNavigate } from 'react-router-dom'
+import { Briefcase, Wrench, User } from 'lucide-react'
+import { motion } from 'framer-motion'
+
+const tabs = [
+ { path: '/admin/workspace', icon: Briefcase, label: '工作台' },
+ { path: '/admin/toolbox', icon: Wrench, label: '百宝箱' },
+ { path: '/admin/mine', icon: User, label: '我的' },
+]
+
+export default function AdminLayout() {
+ const location = useLocation()
+ const navigate = useNavigate()
+
+ const isActive = (path: string) => location.pathname.startsWith(path)
+
+ return (
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/layouts/UserLayout.tsx b/src/layouts/UserLayout.tsx
new file mode 100644
index 0000000..1cda047
--- /dev/null
+++ b/src/layouts/UserLayout.tsx
@@ -0,0 +1,59 @@
+import { Outlet, useLocation, useNavigate } from 'react-router-dom'
+import { Compass, Home, ShoppingBag, Sparkles, User } from 'lucide-react'
+import { motion } from 'framer-motion'
+
+const tabs = [
+ { path: '/user/home', icon: Home, label: '首页' },
+ { path: '/user/ai', icon: Sparkles, label: '智能管家' },
+ { path: '/user/explore', icon: Compass, label: '逛逛' },
+ { path: '/user/group', icon: ShoppingBag, label: '团购' },
+ { path: '/user/mine', icon: User, label: '我的' },
+]
+
+export default function UserLayout() {
+ const location = useLocation()
+ const navigate = useNavigate()
+
+ const isActive = (path: string) => {
+ if (path === '/user/home') {
+ return location.pathname === '/user/home' || location.pathname === '/user/home2'
+ }
+ return location.pathname.startsWith(path)
+ }
+
+ return (
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/main.tsx b/src/main.tsx
new file mode 100644
index 0000000..43ea122
--- /dev/null
+++ b/src/main.tsx
@@ -0,0 +1,16 @@
+import React from 'react'
+import ReactDOM from 'react-dom/client'
+import { BrowserRouter } from 'react-router-dom'
+import App from './App'
+import ThemeProvider from './components/ThemeProvider'
+import './index.css'
+
+ReactDOM.createRoot(document.getElementById('root')!).render(
+
+
+
+
+
+
+ ,
+)
diff --git a/src/mock/adminData.ts b/src/mock/adminData.ts
new file mode 100644
index 0000000..ace43cc
--- /dev/null
+++ b/src/mock/adminData.ts
@@ -0,0 +1,90 @@
+export const workspaceTasks = [
+ {
+ id: 1,
+ title: '张先生 - 阳光花园',
+ type: '报价待确认',
+ status: 'urgent',
+ amount: 158000,
+ createTime: '2024-01-28 10:30',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ },
+ {
+ id: 2,
+ title: '李女士 - 翠湖花园',
+ type: '二期款待收',
+ status: 'pending',
+ amount: 45000,
+ createTime: '2024-01-27 15:20',
+ avatar: 'https://images.unsplash.com/photo-1438761681033-6461ffad8d80?w=200',
+ },
+ {
+ id: 3,
+ title: '王先生 - 金色家园',
+ type: '方案待审核',
+ status: 'normal',
+ amount: 0,
+ createTime: '2024-01-26 09:15',
+ avatar: 'https://images.unsplash.com/photo-1507003211169-0a1dd7228f2d?w=200',
+ },
+ {
+ id: 4,
+ title: '赵女士 - 幸福小区',
+ type: '验收待安排',
+ status: 'normal',
+ amount: 0,
+ createTime: '2024-01-25 14:00',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ },
+]
+
+export const aiNotifications = [
+ {
+ id: 1,
+ type: 'payment',
+ title: 'AI催款提醒',
+ content: '李女士的二期款已逾期3天,建议及时跟进',
+ action: '立即催款',
+ amount: 45000,
+ },
+ {
+ id: 2,
+ type: 'quote',
+ title: 'AI报价推送',
+ content: '张先生的报价已生成,等待客户确认',
+ action: '查看报价',
+ amount: 158000,
+ },
+ {
+ id: 3,
+ type: 'bonus',
+ title: '挽回奖金',
+ content: '成功挽回王先生的订单,奖金已到账',
+ action: '查看详情',
+ amount: 2000,
+ },
+]
+
+export const statistics = {
+ todayOrders: 5,
+ monthOrders: 28,
+ monthAmount: 458000,
+ completionRate: 92,
+}
+
+export const toolboxItems = [
+ { id: 1, title: '报价计算器', icon: 'calculator', desc: '快速生成报价单' },
+ { id: 2, title: '合同模板', icon: 'file-text', desc: '标准合同下载' },
+ { id: 3, title: '材料清单', icon: 'list', desc: '材料用量计算' },
+ { id: 4, title: '进度模板', icon: 'calendar', desc: '施工进度表' },
+ { id: 5, title: '验收标准', icon: 'check-square', desc: '验收规范查询' },
+ { id: 6, title: '客户管理', icon: 'users', desc: '客户信息管理' },
+]
+
+export const adminUserInfo = {
+ id: 1,
+ name: '项目经理张',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ phone: '139****9999',
+ department: '工程部',
+ position: '项目经理',
+}
diff --git a/src/mock/aiData.ts b/src/mock/aiData.ts
new file mode 100644
index 0000000..1fd5708
--- /dev/null
+++ b/src/mock/aiData.ts
@@ -0,0 +1,63 @@
+export const budgetCategories = [
+ { id: 'kitchen', name: '厨房', icon: '🍳', count: 15 },
+ { id: 'bathroom', name: '卫浴', icon: '🛁', count: 12 },
+ { id: 'living', name: '厅房', icon: '🛋️', count: 18 },
+ { id: 'balcony', name: '阳台', icon: '🌿', count: 8 },
+ { id: 'garden', name: '花园', icon: '🌸', count: 6 },
+]
+
+export const budgetItems = {
+ kitchen: [
+ { id: 1, name: '整体橱柜', brand: '欧派', spec: '定制', price: 15000, unit: '套' },
+ { id: 2, name: '油烟机', brand: '方太', spec: '侧吸式', price: 3500, unit: '台' },
+ { id: 3, name: '燃气灶', brand: '老板', spec: '嵌入式', price: 2000, unit: '台' },
+ { id: 4, name: '消毒柜', brand: '美的', spec: '嵌入式', price: 1500, unit: '台' },
+ { id: 5, name: '水槽龙头', brand: '九牧', spec: '不锈钢', price: 800, unit: '套' },
+ { id: 6, name: '厨房吊顶', brand: '奥普', spec: '铝扣板', price: 1200, unit: '㎡' },
+ { id: 7, name: '厨房瓷砖', brand: '东鹏', spec: '300x600', price: 45, unit: '片' },
+ { id: 8, name: '厨房防水', brand: '东方雨虹', spec: '柔性', price: 65, unit: '㎡' },
+ ],
+ bathroom: [
+ { id: 1, name: '马桶', brand: 'TOTO', spec: '智能款', price: 4500, unit: '个' },
+ { id: 2, name: '浴室柜', brand: '箭牌', spec: '实木', price: 3500, unit: '套' },
+ { id: 3, name: '淋浴花洒', brand: '汉斯格雅', spec: '恒温', price: 2800, unit: '套' },
+ { id: 4, name: '浴缸', brand: '科勒', spec: '亚克力', price: 6000, unit: '个' },
+ { id: 5, name: '浴室镜', brand: '定制', spec: '带灯', price: 800, unit: '面' },
+ { id: 6, name: '卫浴五金', brand: '九牧', spec: '套装', price: 1200, unit: '套' },
+ { id: 7, name: '卫生间瓷砖', brand: '马可波罗', spec: '300x300', price: 35, unit: '片' },
+ { id: 8, name: '卫生间防水', brand: '东方雨虹', spec: '柔性', price: 75, unit: '㎡' },
+ ],
+ living: [
+ { id: 1, name: '客厅沙发', brand: '顾家', spec: '真皮', price: 12000, unit: '套' },
+ { id: 2, name: '电视柜', brand: '全友', spec: '实木', price: 3500, unit: '个' },
+ { id: 3, name: '茶几', brand: '林氏木业', spec: '岩板', price: 1800, unit: '个' },
+ { id: 4, name: '餐桌餐椅', brand: '曲美', spec: '一桌六椅', price: 4500, unit: '套' },
+ { id: 5, name: '床', brand: '喜临门', spec: '1.8米', price: 5000, unit: '张' },
+ { id: 6, name: '床垫', brand: '席梦思', spec: '乳胶', price: 6000, unit: '张' },
+ { id: 7, name: '衣柜', brand: '索菲亚', spec: '定制', price: 12000, unit: '套' },
+ { id: 8, name: '书桌', brand: '宜家', spec: '实木', price: 1500, unit: '张' },
+ ],
+ balcony: [
+ { id: 1, name: '阳台推拉门', brand: '凤铝', spec: '断桥铝', price: 800, unit: '㎡' },
+ { id: 2, name: '阳台护栏', brand: '定制', spec: '不锈钢', price: 300, unit: '米' },
+ { id: 3, name: '阳台地砖', brand: '东鹏', spec: '防滑', price: 55, unit: '片' },
+ { id: 4, name: '阳台吊柜', brand: '定制', spec: '防水', price: 800, unit: '米' },
+ { id: 5, name: '晾衣架', brand: '好太太', spec: '电动', price: 2000, unit: '套' },
+ { id: 6, name: '阳台防水', brand: '东方雨虹', spec: '柔性', price: 55, unit: '㎡' },
+ ],
+ garden: [
+ { id: 1, name: '花园围栏', brand: '定制', spec: '防腐木', price: 350, unit: '米' },
+ { id: 2, name: '花园地砖', brand: '户外专用', spec: '透水砖', price: 85, unit: '㎡' },
+ { id: 3, name: '花架', brand: '定制', spec: '防腐木', price: 3000, unit: '套' },
+ { id: 4, name: '花园灯具', brand: '欧普', spec: '太阳能', price: 150, unit: '个' },
+ { id: 5, name: '自动灌溉系统', brand: '雨鸟', spec: '智能', price: 5000, unit: '套' },
+ { id: 6, name: '花园座椅', brand: '户外专用', spec: '藤编', price: 2500, unit: '套' },
+ ],
+}
+
+export const selectedItems = [
+ { categoryId: 'kitchen', itemId: 1, quantity: 1 },
+ { categoryId: 'kitchen', itemId: 2, quantity: 1 },
+ { categoryId: 'bathroom', itemId: 1, quantity: 2 },
+ { categoryId: 'living', itemId: 1, quantity: 1 },
+]
diff --git a/src/mock/groupData.ts b/src/mock/groupData.ts
new file mode 100644
index 0000000..3f0c41d
--- /dev/null
+++ b/src/mock/groupData.ts
@@ -0,0 +1,98 @@
+export const groupBuyPackages = [
+ {
+ id: 1,
+ title: '极速焕新套餐',
+ subtitle: '7天快速改造',
+ price: 29999,
+ originalPrice: 39999,
+ image: 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ features: ['全屋墙面翻新', '地板更换', '厨卫改造', '软装搭配'],
+ sales: 128,
+ rating: 4.8,
+ },
+ {
+ id: 2,
+ title: '品质整装套餐',
+ subtitle: '一站式服务',
+ price: 89999,
+ originalPrice: 129999,
+ image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ features: ['全屋设计', '主材包', '施工服务', '家具软装'],
+ sales: 86,
+ rating: 4.9,
+ },
+ {
+ id: 3,
+ title: '智能家装套餐',
+ subtitle: '科技赋能生活',
+ price: 159999,
+ originalPrice: 199999,
+ image: 'https://images.unsplash.com/photo-1600585154340-be6161a56a0c?w=600',
+ features: ['全屋智能', '灯光系统', '安防系统', '环境控制'],
+ sales: 45,
+ rating: 4.7,
+ },
+]
+
+export const factoryStores = [
+ {
+ id: 1,
+ name: '佛山陶瓷工厂店',
+ address: '佛山市禅城区南庄镇',
+ distance: '15km',
+ image: 'https://images.unsplash.com/photo-1600566753086-00f18fb6b3ea?w=600',
+ rating: 4.8,
+ products: ['瓷砖', '大理石', '马赛克'],
+ busAvailable: true,
+ busTime: '每周六 9:00',
+ },
+ {
+ id: 2,
+ name: '顺德家具工厂店',
+ address: '佛山市顺德区龙江镇',
+ distance: '25km',
+ image: 'https://images.unsplash.com/photo-1600573472592-401b489a3cdc?w=600',
+ rating: 4.9,
+ products: ['沙发', '床具', '餐桌'],
+ busAvailable: true,
+ busTime: '每周日 9:00',
+ },
+ {
+ id: 3,
+ name: '中山灯饰工厂店',
+ address: '中山市古镇镇',
+ distance: '35km',
+ image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ rating: 4.7,
+ products: ['吊灯', '台灯', '智能灯'],
+ busAvailable: false,
+ busTime: '',
+ },
+]
+
+export const busSeats = Array.from({ length: 45 }, (_, i) => ({
+ id: i + 1,
+ row: Math.floor(i / 4) + 1,
+ col: (i % 4) + 1,
+ status: Math.random() > 0.3 ? 'available' : 'occupied',
+ price: 0,
+}))
+
+export const renovationSchemes = [
+ {
+ id: 1,
+ title: '现代简约方案',
+ image: 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ description: '简约不简单,追求功能与美学的完美平衡',
+ price: 15000,
+ duration: '30天',
+ },
+ {
+ id: 2,
+ title: '北欧风格方案',
+ image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ description: '自然清新,营造温馨舒适的居家氛围',
+ price: 18000,
+ duration: '35天',
+ },
+]
diff --git a/src/mock/homeData.ts b/src/mock/homeData.ts
new file mode 100644
index 0000000..9e82ac4
--- /dev/null
+++ b/src/mock/homeData.ts
@@ -0,0 +1,299 @@
+export const homeBanners = [
+ {
+ id: 1,
+ title: '春季家装节',
+ subtitle: '限时优惠',
+ image: 'https://images.unsplash.com/photo-1586023492125-27b2c045efd7?w=800',
+ link: '/user/group',
+ },
+ {
+ id: 2,
+ title: 'AI智能设计',
+ subtitle: '免费体验',
+ image: 'https://images.unsplash.com/photo-1618221195710-dd6b41faaea6?w=800',
+ link: '/user/ai',
+ },
+ {
+ id: 3,
+ title: '品质团购',
+ subtitle: '工厂直供',
+ image: 'https://images.unsplash.com/photo-1560448204-e02f11c3d0e2?w=800',
+ link: '/user/group',
+ },
+]
+
+export const quickActions = [
+ { id: 1, icon: 'calculator', title: '装修预算', desc: 'AI智能计算', link: '/user/ai/budget' },
+ { id: 2, icon: 'palette', title: '设计方案', desc: '专业设计师', link: '/user/mine/designs' },
+ { id: 3, icon: 'truck', title: '团购优惠', desc: '工厂直供', link: '/user/group' },
+ { id: 4, icon: 'calendar', title: '预约服务', desc: '上门量房', link: '/user/home' },
+]
+
+export const designers = [
+ {
+ id: 1,
+ name: '张设计',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ title: '首席设计师',
+ experience: '10年',
+ style: '现代简约',
+ rating: 4.9,
+ projects: 128,
+ },
+ {
+ id: 2,
+ name: '李雅',
+ avatar: 'https://images.unsplash.com/photo-1438761681033-6461ffad8d80?w=200',
+ title: '资深设计师',
+ experience: '8年',
+ style: '北欧风格',
+ rating: 4.8,
+ projects: 96,
+ },
+ {
+ id: 3,
+ name: '王明',
+ avatar: 'https://images.unsplash.com/photo-1507003211169-0a1dd7228f2d?w=200',
+ title: '高级设计师',
+ experience: '6年',
+ style: '中式风格',
+ rating: 4.7,
+ projects: 72,
+ },
+]
+
+export const groupPackages = [
+ {
+ id: 1,
+ name: '全屋定制套餐',
+ image: 'https://images.unsplash.com/photo-1556909114-f6e7ad7d3136?w=400',
+ originalPrice: 89999,
+ groupPrice: 59999,
+ sold: 128,
+ tag: '限时秒杀',
+ },
+ {
+ id: 2,
+ name: '厨房焕新套餐',
+ image: 'https://images.unsplash.com/photo-1556909114-44e3e70034e2?w=400',
+ originalPrice: 39999,
+ groupPrice: 25999,
+ sold: 96,
+ tag: '爆款',
+ },
+ {
+ id: 3,
+ name: '卫浴升级套餐',
+ image: 'https://images.unsplash.com/photo-1552321554-5fefe8c9ef14?w=400',
+ originalPrice: 29999,
+ groupPrice: 19999,
+ sold: 72,
+ tag: '热销',
+ },
+]
+
+export const featuredProjects = [
+ {
+ id: 1,
+ title: '现代简约三居室',
+ area: '120㎡',
+ style: '现代简约',
+ budget: '15-20万',
+ image: 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ },
+ {
+ id: 2,
+ title: '北欧风两居室',
+ area: '85㎡',
+ style: '北欧风格',
+ budget: '10-15万',
+ image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ },
+ {
+ id: 3,
+ title: '新中式别墅',
+ area: '280㎡',
+ style: '新中式',
+ budget: '50-80万',
+ image: 'https://images.unsplash.com/photo-1600566753190-17f0baa2a6c3?w=600',
+ },
+]
+
+export const balconyDesigns = [
+ {
+ id: 1,
+ title: '休闲阳台',
+ desc: '打造惬意休闲空间',
+ image: 'https://images.unsplash.com/photo-1600585154340-be6161a56a0c?w=600',
+ features: ['绿植装饰', '休闲座椅', '遮阳设施'],
+ price: '3000-8000',
+ },
+ {
+ id: 2,
+ title: '洗衣阳台',
+ desc: '实用功能型设计',
+ image: 'https://images.unsplash.com/photo-1600566753086-00f18fb6b3ea?w=600',
+ features: ['洗衣区', '收纳柜', '晾晒区'],
+ price: '2000-5000',
+ },
+ {
+ id: 3,
+ title: '花园阳台',
+ desc: '都市中的小花园',
+ image: 'https://images.unsplash.com/photo-1600573472592-401b489a3cdc?w=600',
+ features: ['花架', '园艺工具', '自动灌溉'],
+ price: '5000-15000',
+ },
+]
+
+export const notifications = [
+ {
+ id: 1,
+ type: 'order',
+ title: '订单发货通知',
+ content: '您的订单 #20240315001 已发货,预计3天内送达',
+ time: '2024-03-15 14:30',
+ isRead: false,
+ },
+ {
+ id: 2,
+ type: 'promotion',
+ title: '限时优惠活动',
+ content: '春季家装节火热进行中,全场套餐低至6折起',
+ time: '2024-03-15 10:00',
+ isRead: false,
+ },
+ {
+ id: 3,
+ type: 'system',
+ title: '系统升级通知',
+ content: '系统将于今晚22:00-23:00进行维护升级,届时部分功能暂不可用',
+ time: '2024-03-14 18:00',
+ isRead: true,
+ },
+ {
+ id: 4,
+ type: 'design',
+ title: '设计方案完成',
+ content: '您预约的设计方案"现代简约三居室"已完成,点击查看详情',
+ time: '2024-03-14 15:20',
+ isRead: true,
+ },
+ {
+ id: 5,
+ type: 'order',
+ title: '订单支付成功',
+ content: '您已成功支付订单 #20240313002,金额¥25999',
+ time: '2024-03-13 09:45',
+ isRead: true,
+ },
+ {
+ id: 6,
+ type: 'promotion',
+ title: '优惠券到账',
+ content: '恭喜您获得200元优惠券一张,有效期30天',
+ time: '2024-03-12 16:30',
+ isRead: true,
+ },
+]
+
+export const exploreContents = [
+ {
+ id: 1,
+ type: 'designer',
+ author: {
+ name: '李雅设计',
+ avatar: 'https://images.unsplash.com/photo-1438761681033-6461ffad8d80?w=200',
+ title: '资深设计师',
+ },
+ title: '120㎡现代简约,打造温馨三口之家',
+ content: '这套案例以白色和原木色为主调,搭配简洁的线条设计,营造出明亮通透的居住空间。客厅采用大面积落地窗,引入充足自然光...',
+ images: [
+ 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ ],
+ likes: 328,
+ comments: 56,
+ shares: 23,
+ time: '2小时前',
+ tags: ['现代简约', '三居室', '120㎡'],
+ },
+ {
+ id: 2,
+ type: 'owner',
+ author: {
+ name: '小王装修记',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ title: '装修达人',
+ },
+ title: '历时3个月,我的北欧风小窝终于完工啦!',
+ content: '从毛坯到入住,记录每一个装修细节。厨房采用了U型布局,最大化利用空间;卧室选用了浅灰色墙面,搭配原木家具...',
+ images: [
+ 'https://images.unsplash.com/photo-1600566753086-00f18fb6b3ea?w=600',
+ 'https://images.unsplash.com/photo-1600585154340-be6161a56a0c?w=600',
+ 'https://images.unsplash.com/photo-1600573472592-401b489a3cdc?w=600',
+ ],
+ likes: 892,
+ comments: 134,
+ shares: 67,
+ time: '5小时前',
+ tags: ['北欧风', '装修日记', '两居室'],
+ },
+ {
+ id: 3,
+ type: 'designer',
+ author: {
+ name: '张设计工作室',
+ avatar: 'https://images.unsplash.com/photo-1507003211169-0a1dd7228f2d?w=200',
+ title: '首席设计师',
+ },
+ title: '新中式别墅设计,传承与现代的完美融合',
+ content: '本案将传统中式元素与现代设计手法相结合,通过木质格栅、水墨画背景墙、禅意茶室等设计,打造出既有东方韵味又不失现代感的居住空间...',
+ images: [
+ 'https://images.unsplash.com/photo-1600566753190-17f0baa2a6c3?w=600',
+ ],
+ likes: 567,
+ comments: 89,
+ shares: 45,
+ time: '昨天',
+ tags: ['新中式', '别墅', '280㎡'],
+ },
+ {
+ id: 4,
+ type: 'owner',
+ author: {
+ name: '装修小白成长记',
+ avatar: 'https://images.unsplash.com/photo-1544005313-94ddf0286df2?w=200',
+ title: '业主',
+ },
+ title: '小户型逆袭!35㎡变身精致单身公寓',
+ content: '谁说小户型不能有品质?通过巧妙的空间规划和收纳设计,我的35㎡小窝实现了客餐厨卫卧一应俱全,分享我的改造心得...',
+ images: [
+ 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ ],
+ likes: 1256,
+ comments: 234,
+ shares: 189,
+ time: '昨天',
+ tags: ['小户型', '单身公寓', '收纳设计'],
+ },
+ {
+ id: 5,
+ type: 'video',
+ author: {
+ name: '入舍设计官方',
+ avatar: 'https://images.unsplash.com/photo-1568602471122-7832951cc4c5?w=200',
+ title: '官方认证',
+ },
+ title: '【视频】全屋定制避坑指南,设计师亲授',
+ content: '全屋定制有哪些坑?如何避免踩雷?本期视频邀请资深设计师为大家详细讲解全屋定制的注意事项...',
+ video: 'https://example.com/video.mp4',
+ videoCover: 'https://images.unsplash.com/photo-1556909114-f6e7ad7d3136?w=600',
+ likes: 2341,
+ comments: 456,
+ shares: 321,
+ time: '3天前',
+ tags: ['全屋定制', '避坑指南', '设计师'],
+ },
+]
diff --git a/src/mock/mineData.ts b/src/mock/mineData.ts
new file mode 100644
index 0000000..4902f21
--- /dev/null
+++ b/src/mock/mineData.ts
@@ -0,0 +1,88 @@
+export const userInfo = {
+ id: 1,
+ name: '用户小明',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ phone: '138****8888',
+ level: 'VIP会员',
+ points: 2580,
+ coupons: 5,
+}
+
+export const dataAssets = [
+ { id: 1, title: '设计方案', count: 3, icon: 'palette' },
+ { id: 2, title: '报价单', count: 2, icon: 'file-text' },
+ { id: 3, title: '合同文件', count: 1, icon: 'file-signature' },
+ { id: 4, title: '验收报告', count: 0, icon: 'check-circle' },
+]
+
+export const designSchemes = [
+ {
+ id: 1,
+ title: '现代简约三居室',
+ status: '进行中',
+ progress: 65,
+ image: 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ designer: '张设计',
+ updateDate: '2024-01-28',
+ },
+ {
+ id: 2,
+ title: '北欧风两居室',
+ status: '已完成',
+ progress: 100,
+ image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ designer: '李雅',
+ updateDate: '2024-01-15',
+ },
+ {
+ id: 3,
+ title: '新中式方案',
+ status: '待确认',
+ progress: 80,
+ image: 'https://images.unsplash.com/photo-1600585154340-be6161a56a0c?w=600',
+ designer: '王明',
+ updateDate: '2024-01-20',
+ },
+]
+
+export const partnerInfo = {
+ level: '金牌合伙人',
+ commission: 15800,
+ orders: 28,
+ teamSize: 15,
+ monthlyTarget: 50000,
+ monthlyProgress: 32000,
+}
+
+export const partnerProducts = [
+ {
+ id: 1,
+ title: '极速焕新套餐',
+ commission: 1500,
+ sales: 12,
+ image: 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=600',
+ },
+ {
+ id: 2,
+ title: '品质整装套餐',
+ commission: 4500,
+ sales: 8,
+ image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=600',
+ },
+ {
+ id: 3,
+ title: '智能家装套餐',
+ commission: 8000,
+ sales: 5,
+ image: 'https://images.unsplash.com/photo-1600585154340-be6161a56a0c?w=600',
+ },
+]
+
+export const menuItems = [
+ { id: 1, title: '装修进度', icon: 'clock', link: '/user/progress' },
+ { id: 2, title: '我的订单', icon: 'shopping-bag', link: '/user/mine/orders' },
+ { id: 3, title: '我的收藏', icon: 'heart', link: '/user/mine/favorites' },
+ { id: 4, title: '我的地址', icon: 'map-pin', link: '/user/mine/address' },
+ { id: 5, title: '帮助中心', icon: 'help-circle', link: '/user/mine/help' },
+ { id: 6, title: '设置', icon: 'settings', link: '/user/mine/settings' },
+]
diff --git a/src/mock/progressData.ts b/src/mock/progressData.ts
new file mode 100644
index 0000000..f05c3ae
--- /dev/null
+++ b/src/mock/progressData.ts
@@ -0,0 +1,78 @@
+export const progressStages = [
+ { id: 'prepare', name: '准备阶段', icon: '📋' },
+ { id: 'design', name: '方案阶段', icon: '✏️' },
+ { id: 'implement', name: '落地阶段', icon: '🔨' },
+ { id: 'visit', name: '上门阶段', icon: '🏠' },
+ { id: 'deliver', name: '交付阶段', icon: '🎉' },
+]
+
+export const progressDetails = {
+ prepare: {
+ title: '准备阶段',
+ status: 'completed',
+ tasks: [
+ { id: 1, title: '需求沟通', status: 'completed', date: '2024-01-15' },
+ { id: 2, title: '上门量房', status: 'completed', date: '2024-01-18' },
+ { id: 3, title: '方案初稿', status: 'completed', date: '2024-01-22' },
+ ],
+ tips: '准备阶段已完成,设计师已获取房屋详细信息。',
+ },
+ design: {
+ title: '方案阶段',
+ status: 'in_progress',
+ tasks: [
+ { id: 1, title: '平面布局', status: 'completed', date: '2024-01-25' },
+ { id: 2, title: '效果图设计', status: 'in_progress', date: '2024-01-28' },
+ { id: 3, title: '施工图绘制', status: 'pending', date: '' },
+ { id: 4, title: '预算报价', status: 'pending', date: '' },
+ ],
+ tips: '效果图设计中,预计3天内完成。',
+ },
+ implement: {
+ title: '落地阶段',
+ status: 'pending',
+ tasks: [
+ { id: 1, title: '材料采购', status: 'pending', date: '' },
+ { id: 2, title: '水电改造', status: 'pending', date: '' },
+ { id: 3, title: '泥瓦工程', status: 'pending', date: '' },
+ { id: 4, title: '木工工程', status: 'pending', date: '' },
+ { id: 5, title: '油漆工程', status: 'pending', date: '' },
+ ],
+ tips: '等待方案确认后开始施工。',
+ },
+ visit: {
+ title: '上门阶段',
+ status: 'pending',
+ tasks: [
+ { id: 1, title: '中期验收', status: 'pending', date: '' },
+ { id: 2, title: '安装调试', status: 'pending', date: '' },
+ { id: 3, title: '软装进场', status: 'pending', date: '' },
+ ],
+ tips: '等待施工完成后进行上门验收。',
+ },
+ deliver: {
+ title: '交付阶段',
+ status: 'pending',
+ tasks: [
+ { id: 1, title: '最终验收', status: 'pending', date: '' },
+ { id: 2, title: '交付使用', status: 'pending', date: '' },
+ { id: 3, title: '售后服务', status: 'pending', date: '' },
+ ],
+ tips: '等待所有工程完成后交付。',
+ },
+}
+
+export const projectInfo = {
+ name: '阳光花园3栋1201室',
+ area: '120㎡',
+ style: '现代简约',
+ budget: '15-20万',
+ startDate: '2024-01-15',
+ expectedEndDate: '2024-04-15',
+ designer: {
+ name: '张设计',
+ avatar: 'https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?w=200',
+ phone: '138****8888',
+ },
+ progress: 35,
+}
diff --git a/src/pages/admin/mine/MinePage.tsx b/src/pages/admin/mine/MinePage.tsx
new file mode 100644
index 0000000..e3665bb
--- /dev/null
+++ b/src/pages/admin/mine/MinePage.tsx
@@ -0,0 +1,67 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronRight, Settings, HelpCircle, Shield, LogOut } from 'lucide-react'
+import { adminUserInfo } from '@/mock/adminData'
+
+export default function MinePage() {
+ const navigate = useNavigate()
+
+ const menuItems = [
+ { id: 1, title: '账号设置', icon: Settings, link: '/admin/mine/settings' },
+ { id: 2, title: '帮助中心', icon: HelpCircle, link: '/admin/mine/help' },
+ { id: 3, title: '隐私政策', icon: Shield, link: '/admin/mine/privacy' },
+ ]
+
+ return (
+
+
+
+

+
+
{adminUserInfo.name}
+
{adminUserInfo.phone}
+
+ {adminUserInfo.department} · {adminUserInfo.position}
+
+
+
+
+
+
+
+
+ {menuItems.map((item, index) => (
+
navigate(item.link)}
+ className={`w-full flex items-center justify-between p-4 ${
+ index !== menuItems.length - 1 ? 'border-b border-gray-50' : ''
+ }`}
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+
+ ))}
+
+
+
+
+ 退出登录
+
+
+
+ )
+}
diff --git a/src/pages/admin/toolbox/ToolboxPage.tsx b/src/pages/admin/toolbox/ToolboxPage.tsx
new file mode 100644
index 0000000..128c0ff
--- /dev/null
+++ b/src/pages/admin/toolbox/ToolboxPage.tsx
@@ -0,0 +1,81 @@
+import { motion } from 'framer-motion'
+import { Calculator, FileText, List, Calendar, CheckSquare, Users } from 'lucide-react'
+import { toolboxItems } from '@/mock/adminData'
+
+export default function ToolboxPage() {
+ const getIcon = (icon: string) => {
+ const icons: Record = {
+ 'calculator': ,
+ 'file-text': ,
+ 'list':
,
+ 'calendar': ,
+ 'check-square': ,
+ 'users': ,
+ }
+ return icons[icon] ||
+ }
+
+ return (
+
+
+
+
+
+
+ {toolboxItems.map((item) => (
+
+
+ {getIcon(item.icon)}
+
+ {item.title}
+ {item.desc}
+
+ ))}
+
+
+
+
+
+
+ )
+}
diff --git a/src/pages/admin/workspace/WorkspacePage.tsx b/src/pages/admin/workspace/WorkspacePage.tsx
new file mode 100644
index 0000000..5b7caef
--- /dev/null
+++ b/src/pages/admin/workspace/WorkspacePage.tsx
@@ -0,0 +1,160 @@
+import { useState } from 'react'
+import { motion } from 'framer-motion'
+import { Bell, ChevronRight, AlertTriangle, DollarSign, FileText, CheckCircle } from 'lucide-react'
+import { workspaceTasks, aiNotifications, statistics } from '@/mock/adminData'
+
+export default function WorkspacePage() {
+ const [showNotifications, setShowNotifications] = useState(false)
+
+ const getStatusColor = (status: string) => {
+ switch (status) {
+ case 'urgent':
+ return 'bg-red-50 text-red-500'
+ case 'pending':
+ return 'bg-yellow-50 text-yellow-500'
+ default:
+ return 'bg-gray-50 text-gray-500'
+ }
+ }
+
+ const getNotificationIcon = (type: string) => {
+ switch (type) {
+ case 'payment':
+ return
+ case 'quote':
+ return
+ case 'bonus':
+ return
+ default:
+ return
+ }
+ }
+
+ return (
+
+
+
+
工作台
+ setShowNotifications(!showNotifications)}
+ className="relative p-2"
+ whileTap={{ scale: 0.9 }}
+ >
+
+
+
+
+
+
+
+
{statistics.todayOrders}
+
今日订单
+
+
+
{statistics.monthOrders}
+
本月订单
+
+
+
{(statistics.monthAmount / 10000).toFixed(1)}万
+
本月金额
+
+
+
{statistics.completionRate}%
+
完成率
+
+
+
+
+ {showNotifications && (
+
+
+
AI智能提醒
+
+ {aiNotifications.map((notification) => (
+
+
+ {getNotificationIcon(notification.type)}
+
+
+
{notification.title}
+
{notification.content}
+
+ {notification.amount > 0 && (
+
+ ¥{notification.amount.toLocaleString()}
+
+ )}
+
+ {notification.action}
+
+
+
+
+ ))}
+
+
+
+ )}
+
+
+
+
+
待办任务
+ {workspaceTasks.length}项
+
+
+
+ {workspaceTasks.map((task) => (
+
+
+
+
{task.title}
+
{task.createTime}
+
+
+
+ {task.type}
+
+ {task.amount > 0 && (
+
+ ¥{task.amount.toLocaleString()}
+
+ )}
+
+
+
+ ))}
+
+
+
+
+
+
+ 今日有2笔款项待跟进,建议优先处理逾期订单,可提高回款率15%。
+
+
+
+
+ )
+}
diff --git a/src/pages/user/ai/AIBudgetPage.tsx b/src/pages/user/ai/AIBudgetPage.tsx
new file mode 100644
index 0000000..0d83794
--- /dev/null
+++ b/src/pages/user/ai/AIBudgetPage.tsx
@@ -0,0 +1,134 @@
+import { budgetCategories, budgetItems, selectedItems as initialSelectedItems } from '@/mock/aiData'
+import { motion } from 'framer-motion'
+import { ChevronLeft, ShoppingCart, Trash2 } from 'lucide-react'
+import { useState } from 'react'
+import { useNavigate } from 'react-router-dom'
+
+export default function AIBudgetPage() {
+ const navigate = useNavigate()
+ const [selectedItems, setSelectedItems] = useState(initialSelectedItems)
+
+ const getTotalPrice = () => {
+ return selectedItems.reduce((total, item) => {
+ const categoryItems = budgetItems[item.categoryId as keyof typeof budgetItems]
+ const budgetItem = categoryItems?.find((i) => i.id === item.itemId)
+ return total + (budgetItem?.price || 0) * item.quantity
+ }, 0)
+ }
+
+ const removeItem = (categoryId: string, itemId: number) => {
+ setSelectedItems(
+ selectedItems.filter(
+ (item) => !(item.categoryId === categoryId && item.itemId === itemId)
+ )
+ )
+ }
+
+ return (
+
+
+
+
装修预算
+
+
+
+
+
+
选择空间
+
+ {budgetCategories.map((category) => (
+ navigate(`/user/ai/budget/${category.id}`)}
+ className="flex flex-col items-center gap-1 p-2 rounded-xl bg-gray-50"
+ whileTap={{ scale: 0.95 }}
+ >
+ {category.icon}
+ {category.name}
+
+ ))}
+
+
+
+
+
+
已选清单
+ {selectedItems.length}项
+
+
+ {selectedItems.length === 0 ? (
+
+ ) : (
+
+ {selectedItems.map((item) => {
+ const categoryItems = budgetItems[item.categoryId as keyof typeof budgetItems]
+ const budgetItem = categoryItems?.find((i) => i.id === item.itemId)
+ const category = budgetCategories.find((c) => c.id === item.categoryId)
+
+ if (!budgetItem) return null
+
+ return (
+
+ {category?.icon}
+
+
{budgetItem.name}
+
+ {budgetItem.brand} · {budgetItem.spec}
+
+
+
+
+ ¥{budgetItem.price.toLocaleString()}
+
+
x{item.quantity}{budgetItem.unit}
+
+ removeItem(item.categoryId, item.itemId)}
+ className="p-1 text-gray-400 hover:text-red-500"
+ whileTap={{ scale: 0.9 }}
+ >
+
+
+
+ )
+ })}
+
+ )}
+
+
+
+
+
+
预估总价
+
¥{getTotalPrice().toLocaleString()}
+
+
+ 生成报价单
+
+
+
+
+
+
AI智能推荐
+
+ 根据您的选择,建议增加厨房吊顶和卫生间防水项目,确保装修质量。
+
+
+
+
+ )
+}
diff --git a/src/pages/user/ai/AIPage.tsx b/src/pages/user/ai/AIPage.tsx
new file mode 100644
index 0000000..4e38b26
--- /dev/null
+++ b/src/pages/user/ai/AIPage.tsx
@@ -0,0 +1,170 @@
+import { motion } from 'framer-motion'
+import { Calculator, ChevronRight, Mic, Send, Volume2 } from 'lucide-react'
+import { useState } from 'react'
+import { useNavigate } from 'react-router-dom'
+
+export default function AIPage() {
+ const navigate = useNavigate()
+ const [inputText, setInputText] = useState('')
+ const [isListening, setIsListening] = useState(false)
+ const [messages, setMessages] = useState([
+ {
+ id: 1,
+ type: 'ai',
+ content: '您好!我是入舍智能管家,我可以帮您:\n\n• 计算装修预算\n• 推荐设计方案\n• 解答装修问题\n• 预约上门服务\n\n请问有什么可以帮您的?',
+ },
+ ])
+
+ const quickQuestions = [
+ '帮我计算装修预算',
+ '推荐现代简约风格',
+ '如何选择装修材料',
+ '预约设计师上门',
+ ]
+
+ const handleSend = () => {
+ if (!inputText.trim()) return
+
+ setMessages([
+ ...messages,
+ { id: Date.now(), type: 'user', content: inputText },
+ {
+ id: Date.now() + 1,
+ type: 'ai',
+ content: '好的,我来帮您处理这个问题。请问您的房屋面积是多少平方米?',
+ },
+ ])
+ setInputText('')
+ }
+
+ const handleMicClick = () => {
+ setIsListening(!isListening)
+ if (!isListening) {
+ setTimeout(() => {
+ setIsListening(false)
+ setInputText('我想计算一下120平米房子的装修预算')
+ }, 2000)
+ }
+ }
+
+ return (
+
+
+
智能管家
+
+
navigate('/user/ai/budget')}
+ className="flex flex-col items-center gap-2"
+ whileTap={{ scale: 0.95 }}
+ >
+
+
+
+ 装修预算
+
+
+
+
+
+ 语音播报
+
+
+
+
+
+ 语音输入
+
+
+
+
+
+ {messages.map((message) => (
+
+
+
+ ))}
+
+
+
快捷问题
+
+ {quickQuestions.map((question, index) => (
+ {
+ if (index === 0) {
+ navigate('/user/ai/budget')
+ } else {
+ setInputText(question)
+ }
+ }}
+ className="w-full flex items-center justify-between p-3 rounded-lg"
+ style={{ backgroundColor: 'var(--bg-tertiary)' }}
+ whileTap={{ scale: 0.98 }}
+ >
+ {question}
+
+
+ ))}
+
+
+
+
+
+
+
+
+
+
+ setInputText(e.target.value)}
+ placeholder="请输入您的问题..."
+ className="w-full h-12 pl-4 pr-12 rounded-full border focus:outline-none focus:border-primary-500"
+ style={{
+ backgroundColor: 'var(--bg-primary)',
+ borderColor: 'var(--border-color)',
+ color: 'var(--text-primary)'
+ }}
+ onKeyPress={(e) => e.key === 'Enter' && handleSend()}
+ />
+
+
+
+
+
+
+
+ )
+}
diff --git a/src/pages/user/ai/BudgetDetailPage.tsx b/src/pages/user/ai/BudgetDetailPage.tsx
new file mode 100644
index 0000000..75b3334
--- /dev/null
+++ b/src/pages/user/ai/BudgetDetailPage.tsx
@@ -0,0 +1,154 @@
+import { useState } from 'react'
+import { useNavigate, useParams } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Plus, Minus, Check } from 'lucide-react'
+import { budgetCategories, budgetItems } from '@/mock/aiData'
+
+export default function BudgetDetailPage() {
+ const navigate = useNavigate()
+ const { category } = useParams<{ category: string }>()
+ const [quantities, setQuantities] = useState>({})
+
+ const categoryInfo = budgetCategories.find((c) => c.id === category)
+ const items = budgetItems[category as keyof typeof budgetItems] || []
+
+ const updateQuantity = (itemId: number, delta: number) => {
+ setQuantities((prev) => ({
+ ...prev,
+ [itemId]: Math.max(0, (prev[itemId] || 0) + delta),
+ }))
+ }
+
+ const handleConfirm = () => {
+ const selected = Object.entries(quantities)
+ .filter(([_, qty]) => qty > 0)
+ .map(([id, qty]) => ({
+ categoryId: category!,
+ itemId: parseInt(id),
+ quantity: qty,
+ }))
+
+ if (selected.length > 0) {
+ alert(`已添加 ${selected.length} 个项目到清单`)
+ }
+ navigate('/user/ai/budget')
+ }
+
+ const getTotalPrice = () => {
+ return items.reduce((total, item) => {
+ return total + item.price * (quantities[item.id] || 0)
+ }, 0)
+ }
+
+ return (
+
+
+
+
+ {categoryInfo?.name || '装修预算'}
+
+
+
+
+
+
+ {budgetCategories.map((cat) => (
+ navigate(`/user/ai/budget/${cat.id}`)}
+ className={`flex-shrink-0 px-4 py-2 rounded-full text-sm ${
+ cat.id === category
+ ? 'bg-primary-500 text-white'
+ : 'bg-white text-gray-600 border border-gray-200'
+ }`}
+ whileTap={{ scale: 0.95 }}
+ >
+ {cat.icon} {cat.name}
+
+ ))}
+
+
+
+ {items.map((item) => (
+
+
+
+
{item.name}
+
+ {item.brand} · {item.spec}
+
+
+
+ ¥{item.price.toLocaleString()}
+ /{item.unit}
+
+
+
+
+
+
updateQuantity(item.id, -1)}
+ disabled={!quantities[item.id]}
+ className={`w-8 h-8 rounded-full flex items-center justify-center ${
+ quantities[item.id]
+ ? 'bg-gray-100 text-gray-600'
+ : 'bg-gray-50 text-gray-300'
+ }`}
+ whileTap={{ scale: 0.9 }}
+ >
+
+
+
+ {quantities[item.id] || 0}
+
+
updateQuantity(item.id, 1)}
+ className="w-8 h-8 rounded-full bg-primary-500 text-white flex items-center justify-center"
+ whileTap={{ scale: 0.9 }}
+ >
+
+
+
+
+ {quantities[item.id] > 0 && (
+
+
+
+ )}
+
+
+ ))}
+
+
+
+
+
+
+
已选 {Object.values(quantities).filter(q => q > 0).length } 项
+
+ ¥{getTotalPrice().toLocaleString()}
+
+
+
+ 确认添加
+
+
+
+
+ )
+}
diff --git a/src/pages/user/explore/ExplorePage.tsx b/src/pages/user/explore/ExplorePage.tsx
new file mode 100644
index 0000000..b9618c9
--- /dev/null
+++ b/src/pages/user/explore/ExplorePage.tsx
@@ -0,0 +1,160 @@
+import { exploreContents } from '@/mock/homeData'
+import { motion } from 'framer-motion'
+import { Heart, MessageCircle, Play, Share2 } from 'lucide-react'
+import { useState } from 'react'
+
+export default function ExplorePage() {
+ const [activeTab, setActiveTab] = useState('all')
+ const [likedPosts, setLikedPosts] = useState([])
+
+ const tabs = [
+ { key: 'all', label: '推荐' },
+ { key: 'designer', label: '设计师' },
+ { key: 'owner', label: '业主分享' },
+ { key: 'video', label: '视频' },
+ ]
+
+ const filteredContents = activeTab === 'all'
+ ? exploreContents
+ : exploreContents.filter((c) => c.type === activeTab)
+
+ const toggleLike = (id: number) => {
+ setLikedPosts((prev) =>
+ prev.includes(id) ? prev.filter((i) => i !== id) : [...prev, id]
+ )
+ }
+
+ return (
+
+
+
逛逛
+
+ {tabs.map((tab) => (
+
+ ))}
+
+
+
+
+ {filteredContents.map((content, index) => (
+
+
+
+

+
+
+
{content.author.name}
+ {content.type === 'designer' && (
+ 设计师
+ )}
+ {content.type === 'video' && (
+ 官方
+ )}
+
+
{content.author.title}
+
+
{content.time}
+
+
+
{content.title}
+
{content.content}
+
+ {content.type === 'video' && content.videoCover ? (
+
+

+
+
+ ) : content.images && content.images.length > 0 && (
+
+ {content.images.map((image, imgIndex) => (
+

+ ))}
+
+ )}
+
+
+ {content.tags.map((tag, tagIndex) => (
+
+ #{tag}
+
+ ))}
+
+
+
+ toggleLike(content.id)}
+ className="flex items-center gap-1"
+ whileTap={{ scale: 0.9 }}
+ >
+
+
+ {likedPosts.includes(content.id) ? content.likes + 1 : content.likes}
+
+
+
+
+
+ {content.comments}
+
+
+
+
+ {content.shares}
+
+
+
+
+ ))}
+
+ {filteredContents.length === 0 && (
+
+ )}
+
+
+ )
+}
diff --git a/src/pages/user/group/BusBookingPage.tsx b/src/pages/user/group/BusBookingPage.tsx
new file mode 100644
index 0000000..b610631
--- /dev/null
+++ b/src/pages/user/group/BusBookingPage.tsx
@@ -0,0 +1,174 @@
+import { useState } from 'react'
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, User, Check } from 'lucide-react'
+import { busSeats } from '@/mock/groupData'
+
+export default function BusBookingPage() {
+ const navigate = useNavigate()
+ const [selectedSeats, setSelectedSeats] = useState([])
+ const [passengerName, setPassengerName] = useState('')
+ const [passengerPhone, setPassengerPhone] = useState('')
+
+ const toggleSeat = (seatId: number, status: string) => {
+ if (status === 'occupied') return
+
+ setSelectedSeats((prev) =>
+ prev.includes(seatId)
+ ? prev.filter((id) => id !== seatId)
+ : prev.length < 4
+ ? [...prev, seatId]
+ : prev
+ )
+ }
+
+ const handleSubmit = () => {
+ if (selectedSeats.length === 0) {
+ alert('请选择座位')
+ return
+ }
+ if (!passengerName || !passengerPhone) {
+ alert('请填写乘客信息')
+ return
+ }
+ alert(`预约成功!已为您预留 ${selectedSeats.length} 个座位`)
+ navigate('/user/group')
+ }
+
+ const renderSeat = (seat: typeof busSeats[0]) => {
+ const isSelected = selectedSeats.includes(seat.id)
+ const isOccupied = seat.status === 'occupied'
+
+ return (
+ toggleSeat(seat.id, seat.status)}
+ disabled={isOccupied}
+ className={`w-10 h-10 rounded-lg flex items-center justify-center text-xs ${
+ isOccupied
+ ? 'bg-gray-200 text-gray-400 cursor-not-allowed'
+ : isSelected
+ ? 'bg-primary-500 text-white'
+ : 'bg-gray-100 text-gray-600 hover:bg-gray-200'
+ }`}
+ whileTap={!isOccupied ? { scale: 0.95 } : {}}
+ >
+ {isSelected ? : seat.id}
+
+ )
+ }
+
+ const rows = []
+ for (let i = 0; i < busSeats.length; i += 4) {
+ const rowSeats = busSeats.slice(i, i + 4)
+ rows.push(rowSeats)
+ }
+
+ return (
+
+
+
+
预约大巴座位
+
+
+
+
+
+
+
+
+
+
+
佛山陶瓷工厂店
+
每周六 9:00 发车
+
+
+
+ 出发:广州市天河区
+ |
+ 全程约1小时
+
+
+
+
+
选择座位
+
+
+
+
+
+
+ {rows.map((row, rowIndex) => (
+
+ {renderSeat(row[0])}
+ {renderSeat(row[1])}
+
+ {renderSeat(row[2])}
+ {renderSeat(row[3])}
+
+ ))}
+
+
+
+ 已选 {selectedSeats.length} 个座位(最多可选4个)
+
+
+
+
+
+
+
+
+ 确认预约
+
+
+
+ )
+}
diff --git a/src/pages/user/group/GroupBuyPage.tsx b/src/pages/user/group/GroupBuyPage.tsx
new file mode 100644
index 0000000..b75b034
--- /dev/null
+++ b/src/pages/user/group/GroupBuyPage.tsx
@@ -0,0 +1,122 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronRight, Star, MapPin, Clock, Users } from 'lucide-react'
+import { groupBuyPackages, factoryStores } from '@/mock/groupData'
+
+export default function GroupBuyPage() {
+ const navigate = useNavigate()
+
+ return (
+
+
+
品质团购
+
工厂直供 · 品质保障 · 专属优惠
+
+
+
+
+
热门套餐
+
+ {groupBuyPackages.map((pkg) => (
+
navigate(`/user/group/package/${pkg.id}`)}
+ className="flex gap-3 p-3 bg-gray-50 rounded-xl"
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+
{pkg.title}
+
{pkg.subtitle}
+
+
+ {pkg.rating}
+ | 已售{pkg.sales}
+
+
+
+ ¥{pkg.price.toLocaleString()}
+
+
+ ¥{pkg.originalPrice.toLocaleString()}
+
+
+
+
+
+ ))}
+
+
+
+
+
+
工厂店展厅
+
+
+
+ {factoryStores.slice(0, 2).map((store) => (
+
navigate('/user/group/store')}
+ className="flex gap-3 p-3 bg-gray-50 rounded-xl"
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+
{store.name}
+
+
+ {store.address}
+
+
+
+
+ {store.rating}
+
+
{store.distance}
+
+ {store.busAvailable && (
+
+
+ 免费大巴 · {store.busTime}
+
+ )}
+
+
+ ))}
+
+
+
+
navigate('/user/group/bus-booking')}
+ className="w-full bg-gradient-to-r from-primary-500 to-primary-600 text-white rounded-xl p-4 flex items-center justify-between"
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+
+
+
+ )
+}
diff --git a/src/pages/user/group/GroupStorePage.tsx b/src/pages/user/group/GroupStorePage.tsx
new file mode 100644
index 0000000..8daade2
--- /dev/null
+++ b/src/pages/user/group/GroupStorePage.tsx
@@ -0,0 +1,84 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Star, MapPin, Clock, Bus } from 'lucide-react'
+import { factoryStores } from '@/mock/groupData'
+
+export default function GroupStorePage() {
+ const navigate = useNavigate()
+
+ return (
+
+
+
+
工厂店展厅
+
+
+
+
+ {factoryStores.map((store) => (
+
+
+
+
{store.name}
+
+
+ {store.address}
+ |
+ {store.distance}
+
+
+
+
+
+ {store.products.map((product, index) => (
+
+ {product}
+
+ ))}
+
+
+ {store.busAvailable && (
+
+
+
+
+
+
免费大巴
+
{store.busTime}
+
+
+
navigate('/user/group/bus-booking')}
+ className="px-4 py-2 bg-primary-500 text-white rounded-lg text-sm"
+ whileTap={{ scale: 0.95 }}
+ >
+ 预约座位
+
+
+
+ )}
+
+
+ ))}
+
+
+ )
+}
diff --git a/src/pages/user/group/PackageDetailPage.tsx b/src/pages/user/group/PackageDetailPage.tsx
new file mode 100644
index 0000000..0a19fe8
--- /dev/null
+++ b/src/pages/user/group/PackageDetailPage.tsx
@@ -0,0 +1,133 @@
+import { useNavigate, useParams } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Star, Check, Share2 } from 'lucide-react'
+import { groupBuyPackages, renovationSchemes } from '@/mock/groupData'
+
+export default function PackageDetailPage() {
+ const navigate = useNavigate()
+ const { id } = useParams()
+
+ const pkg = groupBuyPackages.find((p) => p.id === parseInt(id || '1')) || groupBuyPackages[0]
+
+ return (
+
+
+

+
+
+
+
+
+
+
+
+
+
{pkg.title}
+
{pkg.subtitle}
+
+
+
+
+
+
+
+
+
+ {pkg.rating}
+
+
| 已售{pkg.sales}套
+
+
+
+
+ ¥{pkg.price.toLocaleString()}
+
+
+ ¥{pkg.originalPrice.toLocaleString()}
+
+
+ 省¥{(pkg.originalPrice - pkg.price).toLocaleString()}
+
+
+
+
+
+
套餐包含
+
+ {pkg.features.map((feature, index) => (
+
+
+ {feature}
+
+ ))}
+
+
+
+
+
改造方案
+
+ {renovationSchemes.map((scheme) => (
+
+
+
+
{scheme.title}
+
{scheme.description}
+
+
+ ¥{scheme.price.toLocaleString()}
+
+ {scheme.duration}
+
+
+
+ ))}
+
+
+
+
+
服务保障
+
+ - · 30天无理由退换
+ - · 终身质保服务
+ - · 专业施工团队
+ - · 一站式售后服务
+
+
+
+
+
+
+
+ 加入购物车
+
+
+ 立即购买
+
+
+
+
+ )
+}
diff --git a/src/pages/user/home/BalconyPage.tsx b/src/pages/user/home/BalconyPage.tsx
new file mode 100644
index 0000000..921d074
--- /dev/null
+++ b/src/pages/user/home/BalconyPage.tsx
@@ -0,0 +1,71 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Check } from 'lucide-react'
+import { balconyDesigns } from '@/mock/homeData'
+
+export default function BalconyPage() {
+ const navigate = useNavigate()
+
+ return (
+
+
+
+
阳台设计
+
+
+
+
+ {balconyDesigns.map((design) => (
+
+
+
+
+
{design.title}
+ ¥{design.price}
+
+
{design.desc}
+
+ {design.features.map((feature, index) => (
+
+ {feature}
+
+ ))}
+
+
+
+ ))}
+
+
+
阳台设计要点
+
+ {[
+ '合理规划功能区域',
+ '选择防水防晒材料',
+ '注意承重安全',
+ '考虑采光通风',
+ '绿植搭配技巧',
+ ].map((item, index) => (
+ -
+
+ {item}
+
+ ))}
+
+
+
+
+ )
+}
diff --git a/src/pages/user/home/DesignerPage.tsx b/src/pages/user/home/DesignerPage.tsx
new file mode 100644
index 0000000..dd819c2
--- /dev/null
+++ b/src/pages/user/home/DesignerPage.tsx
@@ -0,0 +1,139 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Star, MessageCircle, Phone, MapPin } from 'lucide-react'
+import { designers } from '@/mock/homeData'
+
+export default function DesignerPage() {
+ const navigate = useNavigate()
+ const designer = designers[0]
+
+ const works = [
+ { id: 1, image: 'https://images.unsplash.com/photo-1600210492486-724fe5c67fb0?w=400' },
+ { id: 2, image: 'https://images.unsplash.com/photo-1600607687939-ce8a6c25118c?w=400' },
+ { id: 3, image: 'https://images.unsplash.com/photo-1600585154340-be6161a56a0c?w=400' },
+ { id: 4, image: 'https://images.unsplash.com/photo-1600566753086-00f18fb6b3ea?w=400' },
+ ]
+
+ return (
+
+
+
+
+
设计师主页
+
+
+
+
+
+

+
+
{designer.name}
+
{designer.title}
+
+
+
+ {designer.rating}
+
+
|
+
{designer.projects}套作品
+
+
+
+
+
+
+
+
+
+
+
{designer.experience}
+
从业经验
+
+
+
{designer.style}
+
擅长风格
+
+
+
+
+
+
+
+
+
个人简介
+
+ 拥有10年室内设计经验,专注于现代简约、北欧风格设计。曾获多项设计大奖,
+ 作品多次刊登于《家居廊》《Elle Decoration》等知名杂志。擅长将空间功能与美学完美结合,
+ 为每一位客户打造独一无二的理想之家。
+
+
+
+
+
+
+
设计作品
+
+ {works.map((work) => (
+
+
+
+ ))}
+
+
+
+
+
+
+
服务范围
+
+
+ 广州市全区及周边城市
+
+
+
+
+
+
+
+
+ 咨询
+
+
+
+ 电话
+
+
navigate('/user/home2')}
+ >
+ 预约设计
+
+
+
+
+ )
+}
diff --git a/src/pages/user/home/HomePage.tsx b/src/pages/user/home/HomePage.tsx
new file mode 100644
index 0000000..aec2bfa
--- /dev/null
+++ b/src/pages/user/home/HomePage.tsx
@@ -0,0 +1,313 @@
+import AIChatModal from '@/components/AIChatModal'
+import NotificationModal from '@/components/NotificationModal'
+import { featuredProjects, groupPackages, homeBanners, quickActions } from '@/mock/homeData'
+import { AnimatePresence, motion } from 'framer-motion'
+import {
+ Bell,
+ Calculator,
+ Calendar,
+ ChevronRight,
+ Palette,
+ Sparkles,
+ Truck
+} from 'lucide-react'
+import { useEffect, useRef, useState } from 'react'
+import { useNavigate } from 'react-router-dom'
+
+function useHorizontalScroll() {
+ const scrollRef = useRef(null)
+
+ useEffect(() => {
+ const el = scrollRef.current
+ if (!el) return
+
+ let isDown = false
+ let startX = 0
+ let scrollLeft = 0
+
+ const handleMouseDown = (e: MouseEvent) => {
+ isDown = true
+ el.style.cursor = 'grabbing'
+ startX = e.pageX - el.offsetLeft
+ scrollLeft = el.scrollLeft
+ }
+
+ const handleMouseLeave = () => {
+ isDown = false
+ el.style.cursor = 'grab'
+ }
+
+ const handleMouseUp = () => {
+ isDown = false
+ el.style.cursor = 'grab'
+ }
+
+ const handleMouseMove = (e: MouseEvent) => {
+ if (!isDown) return
+ e.preventDefault()
+ const x = e.pageX - el.offsetLeft
+ const walk = (x - startX) * 1.5
+ el.scrollLeft = scrollLeft - walk
+ }
+
+ const handleWheel = (e: WheelEvent) => {
+ if (Math.abs(e.deltaY) > Math.abs(e.deltaX)) {
+ e.preventDefault()
+ el.scrollLeft += e.deltaY
+ }
+ }
+
+ el.style.cursor = 'grab'
+ el.addEventListener('mousedown', handleMouseDown)
+ el.addEventListener('mouseleave', handleMouseLeave)
+ el.addEventListener('mouseup', handleMouseUp)
+ el.addEventListener('mousemove', handleMouseMove)
+ el.addEventListener('wheel', handleWheel, { passive: false })
+
+ return () => {
+ el.removeEventListener('mousedown', handleMouseDown)
+ el.removeEventListener('mouseleave', handleMouseLeave)
+ el.removeEventListener('mouseup', handleMouseUp)
+ el.removeEventListener('mousemove', handleMouseMove)
+ el.removeEventListener('wheel', handleWheel)
+ }
+ }, [])
+
+ return scrollRef
+}
+
+export default function HomePage() {
+ const navigate = useNavigate()
+ const [currentBanner, setCurrentBanner] = useState(0)
+ const [countdown, setCountdown] = useState({ days: 0, hours: 0, minutes: 0, seconds: 0 })
+ const [showNotifications, setShowNotifications] = useState(false)
+ const [showAIChat, setShowAIChat] = useState(false)
+ const scrollRef = useHorizontalScroll()
+
+ useEffect(() => {
+ const timer = setInterval(() => {
+ setCurrentBanner((prev) => (prev + 1) % homeBanners.length)
+ }, 4000)
+ return () => clearInterval(timer)
+ }, [])
+
+ useEffect(() => {
+ const endTime = new Date()
+ endTime.setDate(endTime.getDate() + 3)
+ endTime.setHours(23, 59, 59, 999)
+
+ const updateCountdown = () => {
+ const now = new Date()
+ const diff = endTime.getTime() - now.getTime()
+
+ if (diff <= 0) {
+ setCountdown({ days: 0, hours: 0, minutes: 0, seconds: 0 })
+ return
+ }
+
+ const days = Math.floor(diff / (1000 * 60 * 60 * 24))
+ const hours = Math.floor((diff % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60))
+ const minutes = Math.floor((diff % (1000 * 60 * 60)) / (1000 * 60))
+ const seconds = Math.floor((diff % (1000 * 60)) / 1000)
+
+ setCountdown({ days, hours, minutes, seconds })
+ }
+
+ updateCountdown()
+ const timer = setInterval(updateCountdown, 1000)
+ return () => clearInterval(timer)
+ }, [])
+
+ const getActionIcon = (icon: string) => {
+ const icons: Record = {
+ calculator: ,
+ palette: ,
+ truck: ,
+ calendar: ,
+ }
+ return icons[icon] ||
+ }
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
{homeBanners[currentBanner].title}
+
{homeBanners[currentBanner].subtitle}
+
+
+
+
+
+ {homeBanners.map((_, index) => (
+
+ ))}
+
+
+
+
+
+
+
+ {quickActions.map((action) => (
+
navigate(action.link)}
+ className="flex flex-col items-center"
+ whileTap={{ scale: 0.95 }}
+ >
+
+ {getActionIcon(action.icon)}
+
+ {action.title}
+ {action.desc}
+
+ ))}
+
+
+
+
+
+
+
+
团购秒杀
+
+ 距结束
+
+ {String(countdown.days).padStart(2, '0')}
+
+ 天
+
+ {String(countdown.hours).padStart(2, '0')}
+
+ 时
+
+ {String(countdown.minutes).padStart(2, '0')}
+
+ 分
+
+ {String(countdown.seconds).padStart(2, '0')}
+
+ 秒
+
+
+
+
+
+
+ {groupPackages.map((pkg) => (
+
navigate('/user/group')}
+ className="flex-shrink-0 w-36 rounded-xl overflow-hidden shadow-sm"
+ style={{ backgroundColor: 'var(--bg-primary)' }}
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+
{pkg.name}
+
+ ¥{pkg.groupPrice}
+ ¥{pkg.originalPrice}
+
+
+ {pkg.tag}
+ 已售{pkg.sold}
+
+
+
+ ))}
+
+
+
+
+
+
精选案例
+
+
+
+
+ {featuredProjects.map((project) => (
+
+
+
+
{project.title}
+
+ {project.area}
+ |
+ {project.style}
+
+
{project.budget}
+
+
+ ))}
+
+
+
+
setShowNotifications(false)} />
+ setShowAIChat(false)} />
+
+ )
+}
diff --git a/src/pages/user/home/HomePage2.tsx b/src/pages/user/home/HomePage2.tsx
new file mode 100644
index 0000000..88aa1f6
--- /dev/null
+++ b/src/pages/user/home/HomePage2.tsx
@@ -0,0 +1,153 @@
+import { useState } from 'react'
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, ChevronRight, Check } from 'lucide-react'
+
+const timeSlots = [
+ { id: 1, time: '09:00-11:00', available: true },
+ { id: 2, time: '11:00-13:00', available: true },
+ { id: 3, time: '14:00-16:00', available: false },
+ { id: 4, time: '16:00-18:00', available: true },
+]
+
+const dates = Array.from({ length: 7 }, (_, i) => {
+ const date = new Date()
+ date.setDate(date.getDate() + i + 1)
+ return {
+ id: i + 1,
+ day: date.getDate(),
+ week: ['日', '一', '二', '三', '四', '五', '六'][date.getDay()],
+ fullDate: date.toLocaleDateString('zh-CN'),
+ }
+})
+
+export default function HomePage2() {
+ const navigate = useNavigate()
+ const [selectedDate, setSelectedDate] = useState(1)
+ const [selectedTime, setSelectedTime] = useState(null)
+ const [address, setAddress] = useState('')
+ const [phone, setPhone] = useState('')
+ const [remark, setRemark] = useState('')
+
+ const handleSubmit = () => {
+ if (!selectedTime || !address || !phone) {
+ alert('请填写完整信息')
+ return
+ }
+ alert('预约成功!我们将尽快与您联系确认。')
+ navigate('/user/home')
+ }
+
+ return (
+
+
+
+
预约管家上门
+
+
+
+
+
+
选择日期
+
+ {dates.map((date) => (
+ setSelectedDate(date.id)}
+ className={`flex-shrink-0 w-14 h-16 rounded-xl flex flex-col items-center justify-center ${
+ selectedDate === date.id
+ ? 'bg-primary-500 text-white'
+ : 'bg-gray-50 text-gray-600'
+ }`}
+ whileTap={{ scale: 0.95 }}
+ >
+ {date.day}
+ 周{date.week}
+
+ ))}
+
+
+
+
+
选择时间
+
+ {timeSlots.map((slot) => (
+ slot.available && setSelectedTime(slot.id)}
+ disabled={!slot.available}
+ className={`h-12 rounded-xl flex items-center justify-center gap-2 ${
+ !slot.available
+ ? 'bg-gray-100 text-gray-400'
+ : selectedTime === slot.id
+ ? 'bg-primary-500 text-white'
+ : 'bg-gray-50 text-gray-600 border border-gray-200'
+ }`}
+ whileTap={slot.available ? { scale: 0.95 } : {}}
+ >
+ {slot.time}
+ {selectedTime === slot.id && }
+
+ ))}
+
+
+
+
+
联系信息
+
+
+
+ setAddress(e.target.value)}
+ placeholder="请输入详细地址"
+ className="input-field"
+ />
+
+
+
+ setPhone(e.target.value)}
+ placeholder="请输入联系电话"
+ className="input-field"
+ />
+
+
+
+
+
+
+
+
+
服务说明
+
+ - · 免费上门量房服务
+ - · 专业设计师一对一沟通
+ - · 提供3套设计方案参考
+ - · 服务时长约1-2小时
+
+
+
+
+ 确认预约
+
+
+
+ )
+}
diff --git a/src/pages/user/mine/DataAssetsPage.tsx b/src/pages/user/mine/DataAssetsPage.tsx
new file mode 100644
index 0000000..6f8c031
--- /dev/null
+++ b/src/pages/user/mine/DataAssetsPage.tsx
@@ -0,0 +1,62 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, FileText, FileSignature, CheckCircle } from 'lucide-react'
+import { dataAssets } from '@/mock/mineData'
+
+export default function DataAssetsPage() {
+ const navigate = useNavigate()
+
+ const getIcon = (icon: string) => {
+ const icons: Record = {
+ 'palette': ,
+ 'file-text': ,
+ 'file-signature': ,
+ 'check-circle': ,
+ }
+ return icons[icon] ||
+ }
+
+ return (
+
+
+
+
数据资产
+
+
+
+
+ {dataAssets.map((asset) => (
+
+
+
+ {getIcon(asset.icon)}
+
+
+
{asset.title}
+
{asset.count} 个文件
+
+
+ {asset.count}
+
+
+
+ ))}
+
+
+
数据安全说明
+
+ - · 所有数据均加密存储
+ - · 支持随时下载备份
+ - · 数据永久保存
+
+
+
+
+ )
+}
diff --git a/src/pages/user/mine/DesignSchemesPage.tsx b/src/pages/user/mine/DesignSchemesPage.tsx
new file mode 100644
index 0000000..4a6eab2
--- /dev/null
+++ b/src/pages/user/mine/DesignSchemesPage.tsx
@@ -0,0 +1,102 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Eye, Edit, Share2 } from 'lucide-react'
+import { designSchemes } from '@/mock/mineData'
+
+export default function DesignSchemesPage() {
+ const navigate = useNavigate()
+
+ const getStatusColor = (status: string) => {
+ switch (status) {
+ case '已完成':
+ return 'bg-green-50 text-green-500'
+ case '进行中':
+ return 'bg-primary-50 text-primary-500'
+ case '待确认':
+ return 'bg-yellow-50 text-yellow-500'
+ default:
+ return 'bg-gray-50 text-gray-500'
+ }
+ }
+
+ return (
+
+
+
+
设计方案
+
+
+
+
+ {designSchemes.map((scheme) => (
+
+
+
+
+
{scheme.title}
+
+ {scheme.status}
+
+
+
+
+ 设计师:{scheme.designer}
+ |
+ {scheme.updateDate}
+
+
+
+
+ 完成进度
+ {scheme.progress}%
+
+
+
+
+
+
+
+
+
+ 预览
+
+
+
+ 编辑
+
+
+
+ 分享
+
+
+
+
+ ))}
+
+
+ )
+}
diff --git a/src/pages/user/mine/MinePage.tsx b/src/pages/user/mine/MinePage.tsx
new file mode 100644
index 0000000..bc0f463
--- /dev/null
+++ b/src/pages/user/mine/MinePage.tsx
@@ -0,0 +1,115 @@
+import { dataAssets, menuItems, userInfo } from '@/mock/mineData'
+import { motion } from 'framer-motion'
+import { ChevronRight, Clock, Heart, HelpCircle, MapPin, Settings, ShoppingBag } from 'lucide-react'
+import { useNavigate } from 'react-router-dom'
+
+export default function MinePage() {
+ const navigate = useNavigate()
+
+ const getIcon = (icon: string) => {
+ const icons: Record = {
+ 'clock': ,
+ 'shopping-bag': ,
+ 'heart': ,
+ 'map-pin': ,
+ 'help-circle': ,
+ 'settings': ,
+ }
+ return icons[icon] ||
+ }
+
+ return (
+
+
+
+

+
+
{userInfo.name}
+
{userInfo.phone}
+
+ {userInfo.level}
+
+
+
+
+
+
+
+
+
+
navigate('/user/mine/data')}
+ className="cursor-pointer"
+ whileTap={{ scale: 0.95 }}
+ >
+ {userInfo.points}
+ 积分
+
+
navigate('/user/mine/designs')}
+ className="cursor-pointer"
+ whileTap={{ scale: 0.95 }}
+ >
+ {userInfo.coupons}
+ 优惠券
+
+
navigate('/user/mine/partner')}
+ className="cursor-pointer"
+ whileTap={{ scale: 0.95 }}
+ >
+ 合伙人
+ 中心
+
+
+
+
+
+
+
数据资产
+
+
+
+ {dataAssets.map((asset) => (
+
navigate('/user/mine/data')}
+ className="text-center cursor-pointer"
+ whileTap={{ scale: 0.95 }}
+ >
+
+ {asset.count}
+
+ {asset.title}
+
+ ))}
+
+
+
+
+ {menuItems.map((item, index) => (
+
navigate(item.link)}
+ className={`w-full flex items-center justify-between p-4 ${index !== menuItems.length - 1 ? 'border-b' : ''}`}
+ style={{ borderColor: 'var(--border-color)' }}
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+ {getIcon(item.icon)}
+
+
{item.title}
+
+
+
+ ))}
+
+
+
+ )
+}
diff --git a/src/pages/user/mine/PartnerCenterPage.tsx b/src/pages/user/mine/PartnerCenterPage.tsx
new file mode 100644
index 0000000..d11116b
--- /dev/null
+++ b/src/pages/user/mine/PartnerCenterPage.tsx
@@ -0,0 +1,116 @@
+import { useNavigate } from 'react-router-dom'
+import { motion } from 'framer-motion'
+import { ChevronLeft, Users, TrendingUp, Gift, ChevronRight, Share2 } from 'lucide-react'
+import { partnerInfo, partnerProducts } from '@/mock/mineData'
+
+export default function PartnerCenterPage() {
+ const navigate = useNavigate()
+
+ return (
+
+
+
+
合伙人中心
+
+
+
+
+
+
+
+ {partnerInfo.level}
+
+
+
+
+
¥{partnerInfo.commission.toLocaleString()}
+
累计佣金
+
+
+
{partnerInfo.orders}
+
推广订单
+
+
+
{partnerInfo.teamSize}
+
团队人数
+
+
+
+
+
+
+
本月目标
+
+ ¥{partnerInfo.monthlyProgress.toLocaleString()} / ¥{partnerInfo.monthlyTarget.toLocaleString()}
+
+
+
+
+
+
+ 还差 ¥{(partnerInfo.monthlyTarget - partnerInfo.monthlyProgress).toLocaleString()} 达成本月目标
+
+
+
+
+
+
选品推广
+
+
+
+ {partnerProducts.map((product) => (
+
+
+
+
{product.title}
+
已售 {product.sales} 件
+
+ 佣金 ¥{product.commission.toLocaleString()}
+
+
+
+
+
+
+ ))}
+
+
+
+
+
+
+ 推广数据
+
+
+
+ 奖励中心
+
+
+
+
+ )
+}
diff --git a/src/pages/user/mine/SettingsPage.tsx b/src/pages/user/mine/SettingsPage.tsx
new file mode 100644
index 0000000..0e961d6
--- /dev/null
+++ b/src/pages/user/mine/SettingsPage.tsx
@@ -0,0 +1,148 @@
+import { useThemeStore, ThemeMode } from '@/store/themeStore'
+import { motion } from 'framer-motion'
+import { Check, Moon, Sun, Waves } from 'lucide-react'
+import { useNavigate } from 'react-router-dom'
+
+interface ThemeOption {
+ mode: ThemeMode
+ label: string
+ description: string
+ icon: React.ReactNode
+ previewBg: string
+ previewGradient: string
+}
+
+const themeOptions: ThemeOption[] = [
+ {
+ mode: 'light',
+ label: '亮色模式',
+ description: '明亮清新的界面风格',
+ icon: ,
+ previewBg: 'bg-white',
+ previewGradient: 'from-orange-400 to-orange-500',
+ },
+ {
+ mode: 'dark',
+ label: '暗色模式',
+ description: '护眼舒适的深色界面',
+ icon: ,
+ previewBg: 'bg-gray-800',
+ previewGradient: 'from-orange-400 to-orange-500',
+ },
+ {
+ mode: 'deepBlue',
+ label: '深蓝模式',
+ description: '沉稳专业的蓝色主题',
+ icon: ,
+ previewBg: 'bg-slate-900',
+ previewGradient: 'from-blue-500 to-blue-600',
+ },
+]
+
+export default function SettingsPage() {
+ const navigate = useNavigate()
+ const { theme, setTheme } = useThemeStore()
+
+ return (
+
+
+
+
+
+ 设置
+
+
+
+
+
+
+
+
+ 主题设置
+
+
+ 选择您喜欢的界面主题风格
+
+
+
+
+ {themeOptions.map((option) => {
+ const isActive = theme === option.mode
+ return (
+
setTheme(option.mode)}
+ className="w-full rounded-xl overflow-hidden transition-all duration-300"
+ style={{
+ backgroundColor: 'var(--bg-primary)',
+ boxShadow: isActive ? `0 4px 12px var(--shadow-color)` : '0 1px 3px var(--shadow-color)',
+ border: isActive ? '2px solid var(--gradient-start)' : '2px solid transparent',
+ }}
+ whileTap={{ scale: 0.98 }}
+ >
+
+
+
+
+
+
+
+ {option.icon}
+
+
+ {option.label}
+
+
+
+ {option.description}
+
+
+
+
+ {isActive && (
+
+
+
+ )}
+
+
+
+
+ )
+ })}
+
+
+
+
+ 当前主题
+
+
+ {themeOptions.find(t => t.mode === theme)?.label} - {themeOptions.find(t => t.mode === theme)?.description}
+
+
+
+
+ )
+}
diff --git a/src/pages/user/progress/ProgressPage.tsx b/src/pages/user/progress/ProgressPage.tsx
new file mode 100644
index 0000000..3c71b3d
--- /dev/null
+++ b/src/pages/user/progress/ProgressPage.tsx
@@ -0,0 +1,221 @@
+import { progressDetails, progressStages, projectInfo } from '@/mock/progressData'
+import { motion } from 'framer-motion'
+import { Check, ChevronRight, Clock, MessageCircle, Phone } from 'lucide-react'
+import { useState } from 'react'
+
+export default function ProgressPage() {
+ const [activeStage, setActiveStage] = useState('design')
+
+ const currentStageIndex = progressStages.findIndex((s) => s.id === activeStage)
+ const currentDetail = progressDetails[activeStage as keyof typeof progressDetails]
+
+ const getStatusColor = (status: string) => {
+ switch (status) {
+ case 'completed':
+ return 'text-green-500'
+ case 'in_progress':
+ return 'text-primary-500'
+ default:
+ return 'text-gray-400'
+ }
+ }
+
+ const getStatusBg = (status: string) => {
+ switch (status) {
+ case 'completed':
+ return 'bg-green-500'
+ case 'in_progress':
+ return 'bg-primary-500'
+ default:
+ return 'bg-gray-200'
+ }
+ }
+
+ return (
+
+
+
装修进度
+
+
+
+
+ 🏠
+
+
+
{projectInfo.name}
+
{projectInfo.area} · {projectInfo.style}
+
+
+
+
+
+ 整体进度
+ {projectInfo.progress}%
+
+
+
+
+
+
+
+
+
+
+
+

+
+
{projectInfo.designer.name}
+
设计师
+
+
+
+
+
+
+
施工阶段
+
+
+ {progressStages.map((stage, index) => {
+ const isActive = stage.id === activeStage
+ const isPast = index < currentStageIndex
+
+ return (
+
setActiveStage(stage.id)}
+ className="flex flex-col items-center"
+ whileTap={{ scale: 0.95 }}
+ >
+
+ {isPast ? : stage.icon}
+
+
+ {stage.name}
+
+
+ )
+ })}
+
+
+
+
+
{currentDetail.title}
+
+ {currentDetail.status === 'completed'
+ ? '已完成'
+ : currentDetail.status === 'in_progress'
+ ? '进行中'
+ : '待开始'}
+
+
+
+
+ {currentDetail.tasks.map((task) => (
+
+
+
+ {task.status === 'completed' ? (
+
+ ) : task.status === 'in_progress' ? (
+
+ ) : (
+
+ )}
+
+
+ {task.title}
+
+
+ {task.date && (
+
{task.date}
+ )}
+
+ ))}
+
+
+
+
+
+
+
+
+
项目信息
+
+
+
+
+
预算范围
+
{projectInfo.budget}
+
+
+
开工日期
+
{projectInfo.startDate}
+
+
+
预计完工
+
{projectInfo.expectedEndDate}
+
+
+
装修风格
+
{projectInfo.style}
+
+
+
+
+
+ )
+}
diff --git a/src/store/themeStore.ts b/src/store/themeStore.ts
new file mode 100644
index 0000000..8507fc9
--- /dev/null
+++ b/src/store/themeStore.ts
@@ -0,0 +1,21 @@
+import { create } from 'zustand'
+import { persist } from 'zustand/middleware'
+
+export type ThemeMode = 'light' | 'dark' | 'deepBlue'
+
+interface ThemeState {
+ theme: ThemeMode
+ setTheme: (theme: ThemeMode) => void
+}
+
+export const useThemeStore = create()(
+ persist(
+ (set) => ({
+ theme: 'light',
+ setTheme: (theme) => set({ theme }),
+ }),
+ {
+ name: 'theme-storage',
+ }
+ )
+)
diff --git a/tailwind.config.js b/tailwind.config.js
new file mode 100644
index 0000000..2200dad
--- /dev/null
+++ b/tailwind.config.js
@@ -0,0 +1,52 @@
+/** @type {import('tailwindcss').Config} */
+export default {
+ content: [
+ "./index.html",
+ "./src/**/*.{js,ts,jsx,tsx}",
+ ],
+ darkMode: 'class',
+ theme: {
+ extend: {
+ colors: {
+ primary: {
+ 50: '#fff7ed',
+ 100: '#ffedd5',
+ 200: '#fed7aa',
+ 300: '#fdba74',
+ 400: '#fb923c',
+ 500: '#f97316',
+ 600: '#ea580c',
+ 700: '#c2410c',
+ 800: '#9a3412',
+ 900: '#7c2d12',
+ },
+ secondary: {
+ 50: '#f0fdf4',
+ 100: '#dcfce7',
+ 200: '#bbf7d0',
+ 300: '#86efac',
+ 400: '#4ade80',
+ 500: '#22c55e',
+ 600: '#16a34a',
+ 700: '#15803d',
+ 800: '#166534',
+ 900: '#14532d',
+ },
+ },
+ fontFamily: {
+ sans: [
+ '-apple-system',
+ 'BlinkMacSystemFont',
+ 'Segoe UI',
+ 'Roboto',
+ 'PingFang SC',
+ 'Microsoft YaHei',
+ 'Helvetica Neue',
+ 'Arial',
+ 'sans-serif',
+ ],
+ },
+ },
+ },
+ plugins: [],
+}
diff --git a/tsconfig.app.json b/tsconfig.app.json
new file mode 100644
index 0000000..a9b5a59
--- /dev/null
+++ b/tsconfig.app.json
@@ -0,0 +1,28 @@
+{
+ "compilerOptions": {
+ "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
+ "target": "ES2022",
+ "useDefineForClassFields": true,
+ "lib": ["ES2022", "DOM", "DOM.Iterable"],
+ "module": "ESNext",
+ "types": ["vite/client"],
+ "skipLibCheck": true,
+
+ /* Bundler mode */
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "verbatimModuleSyntax": true,
+ "moduleDetection": "force",
+ "noEmit": true,
+ "jsx": "react-jsx",
+
+ /* Linting */
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "erasableSyntaxOnly": true,
+ "noFallthroughCasesInSwitch": true,
+ "noUncheckedSideEffectImports": true
+ },
+ "include": ["src"]
+}
diff --git a/tsconfig.json b/tsconfig.json
new file mode 100644
index 0000000..5413626
--- /dev/null
+++ b/tsconfig.json
@@ -0,0 +1,25 @@
+{
+ "compilerOptions": {
+ "target": "ES2020",
+ "useDefineForClassFields": true,
+ "lib": ["ES2020", "DOM", "DOM.Iterable"],
+ "module": "ESNext",
+ "skipLibCheck": true,
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "resolveJsonModule": true,
+ "isolatedModules": true,
+ "noEmit": true,
+ "jsx": "react-jsx",
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "noFallthroughCasesInSwitch": true,
+ "baseUrl": ".",
+ "paths": {
+ "@/*": ["src/*"]
+ }
+ },
+ "include": ["src"],
+ "references": [{ "path": "./tsconfig.node.json" }]
+}
diff --git a/tsconfig.node.json b/tsconfig.node.json
new file mode 100644
index 0000000..97ede7e
--- /dev/null
+++ b/tsconfig.node.json
@@ -0,0 +1,11 @@
+{
+ "compilerOptions": {
+ "composite": true,
+ "skipLibCheck": true,
+ "module": "ESNext",
+ "moduleResolution": "bundler",
+ "allowSyntheticDefaultImports": true,
+ "strict": true
+ },
+ "include": ["vite.config.ts"]
+}
diff --git a/vite.config.ts b/vite.config.ts
new file mode 100644
index 0000000..0d51f19
--- /dev/null
+++ b/vite.config.ts
@@ -0,0 +1,12 @@
+import { defineConfig } from 'vite'
+import react from '@vitejs/plugin-react'
+import path from 'path'
+
+export default defineConfig({
+ plugins: [react()],
+ resolve: {
+ alias: {
+ '@': path.resolve(__dirname, './src'),
+ },
+ },
+})