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.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 }} + > +
+
+ +
+ {item.title} +
+ +
+ ))} +
+ + + + 退出登录 + +
+
+ ) +} 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.title}

+

{task.createTime}

+
+
+ + {task.type} + + {task.amount > 0 && ( +

+ ¥{task.amount.toLocaleString()} +

+ )} +
+ +
+ ))} +
+
+ +
+
+ +

AI工作建议

+
+

+ 今日有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) => ( + +
+

{message.content}

+
+
+ ))} + +
+

快捷问题

+
+ {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.author.name}

+ {content.type === 'designer' && ( + 设计师 + )} + {content.type === 'video' && ( + 官方 + )} +
+

{content.author.title}

+
+ {content.time} +
+ +

{content.title}

+

{content.content}

+ + {content.type === 'video' && content.videoCover ? ( +
+ {content.title} +
+
+ +
+
+
+ ) : 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个) +

+
+ +
+

乘客信息

+
+
+ + setPassengerName(e.target.value)} + placeholder="请输入姓名" + className="input-field" + /> +
+
+ + setPassengerPhone(e.target.value)} + placeholder="请输入手机号" + className="input-field" + /> +
+
+
+
+ +
+ + 确认预约 + +
+
+ ) +} 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.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.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.name}

+
+ + {store.address} + | + {store.distance} +
+ +
+
+ + {store.rating} +
+
+ +
+ {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.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.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.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.name}

+

{designer.title}

+
+
+ + {designer.rating} +
+ | + {designer.projects}套作品 +
+
+
+
+
+ +
+
+
+
+

{designer.experience}

+

从业经验

+
+
+

{designer.style}

+

擅长风格

+
+
+

98%

+

好评率

+
+
+
+
+ +
+
+

个人简介

+

+ 拥有10年室内设计经验,专注于现代简约、北欧风格设计。曾获多项设计大奖, + 作品多次刊登于《家居廊》《Elle Decoration》等知名杂志。擅长将空间功能与美学完美结合, + 为每一位客户打造独一无二的理想之家。 +

+
+
+ +
+
+

设计作品

+
+ {works.map((work) => ( + + {`作品${work.id}`} + + ))} +
+
+
+ +
+
+

服务范围

+
+ + 广州市全区及周边城市 +
+
+
+ +
+
+ + + 咨询 + + + + 电话 + + 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].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.name}

+
+ ¥{pkg.groupPrice} + ¥{pkg.originalPrice} +
+
+ {pkg.tag} + 已售{pkg.sold} +
+
+
+ ))} +
+
+ +
+
+

精选案例

+ +
+ +
+ {featuredProjects.map((project) => ( + + {project.title} +
+

{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" + /> +
+
+ +