Merge remote-tracking branch 'origin/feat/wizard-plan-vista'
This commit is contained in:
43
bun.lock
43
bun.lock
@@ -5,6 +5,7 @@
|
|||||||
"name": "acad-ia-2",
|
"name": "acad-ia-2",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@radix-ui/react-avatar": "^1.1.11",
|
"@radix-ui/react-avatar": "^1.1.11",
|
||||||
|
"@radix-ui/react-checkbox": "^1.3.3",
|
||||||
"@radix-ui/react-dialog": "^1.1.15",
|
"@radix-ui/react-dialog": "^1.1.15",
|
||||||
"@radix-ui/react-label": "^2.1.8",
|
"@radix-ui/react-label": "^2.1.8",
|
||||||
"@radix-ui/react-popover": "^1.1.15",
|
"@radix-ui/react-popover": "^1.1.15",
|
||||||
@@ -25,10 +26,11 @@
|
|||||||
"class-variance-authority": "^0.7.1",
|
"class-variance-authority": "^0.7.1",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"cmdk": "^1.1.1",
|
"cmdk": "^1.1.1",
|
||||||
"lucide-react": "^0.561.0",
|
"lucide-react": "^0.562.0",
|
||||||
|
"motion": "^12.24.7",
|
||||||
"react": "^19.2.0",
|
"react": "^19.2.0",
|
||||||
"react-dom": "^19.2.0",
|
"react-dom": "^19.2.0",
|
||||||
"tailwind-merge": "^3.0.2",
|
"tailwind-merge": "^3.4.0",
|
||||||
"tailwindcss": "^4.0.6",
|
"tailwindcss": "^4.0.6",
|
||||||
"tw-animate-css": "^1.3.6",
|
"tw-animate-css": "^1.3.6",
|
||||||
},
|
},
|
||||||
@@ -45,6 +47,7 @@
|
|||||||
"eslint-import-resolver-typescript": "^4.4.4",
|
"eslint-import-resolver-typescript": "^4.4.4",
|
||||||
"eslint-plugin-import": "^2.32.0",
|
"eslint-plugin-import": "^2.32.0",
|
||||||
"eslint-plugin-jsx-a11y": "^6.10.2",
|
"eslint-plugin-jsx-a11y": "^6.10.2",
|
||||||
|
"eslint-plugin-react-hooks": "^7.0.1",
|
||||||
"eslint-plugin-unused-imports": "^4.3.0",
|
"eslint-plugin-unused-imports": "^4.3.0",
|
||||||
"jsdom": "^27.0.0",
|
"jsdom": "^27.0.0",
|
||||||
"prettier": "^3.5.3",
|
"prettier": "^3.5.3",
|
||||||
@@ -239,6 +242,8 @@
|
|||||||
|
|
||||||
"@radix-ui/react-avatar": ["@radix-ui/react-avatar@1.1.11", "", { "dependencies": { "@radix-ui/react-context": "1.1.3", "@radix-ui/react-primitive": "2.1.4", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-is-hydrated": "0.1.0", "@radix-ui/react-use-layout-effect": "1.1.1" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-0Qk603AHGV28BOBO34p7IgD5m+V5Sg/YovfayABkoDDBM5d3NCx0Mp4gGrjzLGes1jV5eNOE1r3itqOR33VC6Q=="],
|
"@radix-ui/react-avatar": ["@radix-ui/react-avatar@1.1.11", "", { "dependencies": { "@radix-ui/react-context": "1.1.3", "@radix-ui/react-primitive": "2.1.4", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-is-hydrated": "0.1.0", "@radix-ui/react-use-layout-effect": "1.1.1" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-0Qk603AHGV28BOBO34p7IgD5m+V5Sg/YovfayABkoDDBM5d3NCx0Mp4gGrjzLGes1jV5eNOE1r3itqOR33VC6Q=="],
|
||||||
|
|
||||||
|
"@radix-ui/react-checkbox": ["@radix-ui/react-checkbox@1.3.3", "", { "dependencies": { "@radix-ui/primitive": "1.1.3", "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-presence": "1.1.5", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-use-controllable-state": "1.2.2", "@radix-ui/react-use-previous": "1.1.1", "@radix-ui/react-use-size": "1.1.1" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-wBbpv+NQftHDdG86Qc0pIyXk5IR3tM8Vd0nWLKDcX8nNn4nXFOFwsKuqw2okA/1D/mpaAkmuyndrPJTYDNZtFw=="],
|
||||||
|
|
||||||
"@radix-ui/react-collection": ["@radix-ui/react-collection@1.1.7", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-Fh9rGN0MoI4ZFUNyfFVNU4y9LUz93u9/0K+yLgA2bwRojxM8JU1DyvvMBabnZPBgMWREAJvU2jjVzq+LrFUglw=="],
|
"@radix-ui/react-collection": ["@radix-ui/react-collection@1.1.7", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-Fh9rGN0MoI4ZFUNyfFVNU4y9LUz93u9/0K+yLgA2bwRojxM8JU1DyvvMBabnZPBgMWREAJvU2jjVzq+LrFUglw=="],
|
||||||
|
|
||||||
"@radix-ui/react-compose-refs": ["@radix-ui/react-compose-refs@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg=="],
|
"@radix-ui/react-compose-refs": ["@radix-ui/react-compose-refs@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-z4eqJvfiNnFMHIIvXP3CY57y2WJs5g2v3X0zm9mEJkrkNv4rDxu+sg9Jh8EkXyeqBkB7SOcboo9dMVqhyrACIg=="],
|
||||||
@@ -271,8 +276,6 @@
|
|||||||
|
|
||||||
"@radix-ui/react-roving-focus": ["@radix-ui/react-roving-focus@1.1.11", "", { "dependencies": { "@radix-ui/primitive": "1.1.3", "@radix-ui/react-collection": "1.1.7", "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-direction": "1.1.1", "@radix-ui/react-id": "1.1.1", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-controllable-state": "1.2.2" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-7A6S9jSgm/S+7MdtNDSb+IU859vQqJ/QAtcYQcfFC6W8RS4IxIZDldLR0xqCFZ6DCyrQLjLPsxtTNch5jVA4lA=="],
|
"@radix-ui/react-roving-focus": ["@radix-ui/react-roving-focus@1.1.11", "", { "dependencies": { "@radix-ui/primitive": "1.1.3", "@radix-ui/react-collection": "1.1.7", "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-direction": "1.1.1", "@radix-ui/react-id": "1.1.1", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-controllable-state": "1.2.2" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-7A6S9jSgm/S+7MdtNDSb+IU859vQqJ/QAtcYQcfFC6W8RS4IxIZDldLR0xqCFZ6DCyrQLjLPsxtTNch5jVA4lA=="],
|
||||||
|
|
||||||
"@radix-ui/react-scroll-area": ["@radix-ui/react-scroll-area@1.2.10", "", { "dependencies": { "@radix-ui/number": "1.1.1", "@radix-ui/primitive": "1.1.3", "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-direction": "1.1.1", "@radix-ui/react-presence": "1.1.5", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-layout-effect": "1.1.1" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-tAXIa1g3sM5CGpVT0uIbUx/U3Gs5N8T52IICuCtObaos1S8fzsrPXG5WObkQN3S6NVl6wKgPhAIiBGbWnvc97A=="],
|
|
||||||
|
|
||||||
"@radix-ui/react-select": ["@radix-ui/react-select@2.2.6", "", { "dependencies": { "@radix-ui/number": "1.1.1", "@radix-ui/primitive": "1.1.3", "@radix-ui/react-collection": "1.1.7", "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-direction": "1.1.1", "@radix-ui/react-dismissable-layer": "1.1.11", "@radix-ui/react-focus-guards": "1.1.3", "@radix-ui/react-focus-scope": "1.1.7", "@radix-ui/react-id": "1.1.1", "@radix-ui/react-popper": "1.2.8", "@radix-ui/react-portal": "1.1.9", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-slot": "1.2.3", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-controllable-state": "1.2.2", "@radix-ui/react-use-layout-effect": "1.1.1", "@radix-ui/react-use-previous": "1.1.1", "@radix-ui/react-visually-hidden": "1.2.3", "aria-hidden": "^1.2.4", "react-remove-scroll": "^2.6.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-I30RydO+bnn2PQztvo25tswPH+wFBjehVGtmagkU78yMdwTwVf12wnAOF+AeP8S2N8xD+5UPbGhkUfPyvT+mwQ=="],
|
"@radix-ui/react-select": ["@radix-ui/react-select@2.2.6", "", { "dependencies": { "@radix-ui/number": "1.1.1", "@radix-ui/primitive": "1.1.3", "@radix-ui/react-collection": "1.1.7", "@radix-ui/react-compose-refs": "1.1.2", "@radix-ui/react-context": "1.1.2", "@radix-ui/react-direction": "1.1.1", "@radix-ui/react-dismissable-layer": "1.1.11", "@radix-ui/react-focus-guards": "1.1.3", "@radix-ui/react-focus-scope": "1.1.7", "@radix-ui/react-id": "1.1.1", "@radix-ui/react-popper": "1.2.8", "@radix-ui/react-portal": "1.1.9", "@radix-ui/react-primitive": "2.1.3", "@radix-ui/react-slot": "1.2.3", "@radix-ui/react-use-callback-ref": "1.1.1", "@radix-ui/react-use-controllable-state": "1.2.2", "@radix-ui/react-use-layout-effect": "1.1.1", "@radix-ui/react-use-previous": "1.1.1", "@radix-ui/react-visually-hidden": "1.2.3", "aria-hidden": "^1.2.4", "react-remove-scroll": "^2.6.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-I30RydO+bnn2PQztvo25tswPH+wFBjehVGtmagkU78yMdwTwVf12wnAOF+AeP8S2N8xD+5UPbGhkUfPyvT+mwQ=="],
|
||||||
|
|
||||||
"@radix-ui/react-separator": ["@radix-ui/react-separator@1.1.8", "", { "dependencies": { "@radix-ui/react-primitive": "2.1.4" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-sDvqVY4itsKwwSMEe0jtKgfTh+72Sy3gPmQpjqcQneqQ4PFmr/1I0YA+2/puilhggCe2gJcx5EBAYFkWkdpa5g=="],
|
"@radix-ui/react-separator": ["@radix-ui/react-separator@1.1.8", "", { "dependencies": { "@radix-ui/react-primitive": "2.1.4" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-sDvqVY4itsKwwSMEe0jtKgfTh+72Sy3gPmQpjqcQneqQ4PFmr/1I0YA+2/puilhggCe2gJcx5EBAYFkWkdpa5g=="],
|
||||||
@@ -759,6 +762,8 @@
|
|||||||
|
|
||||||
"eslint-plugin-n": ["eslint-plugin-n@17.23.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.5.0", "enhanced-resolve": "^5.17.1", "eslint-plugin-es-x": "^7.8.0", "get-tsconfig": "^4.8.1", "globals": "^15.11.0", "globrex": "^0.1.2", "ignore": "^5.3.2", "semver": "^7.6.3", "ts-declaration-location": "^1.0.6" }, "peerDependencies": { "eslint": ">=8.23.0" } }, "sha512-68PealUpYoHOBh332JLLD9Sj7OQUDkFpmcfqt8R9sySfFSeuGJjMTJQvCRRB96zO3A/PELRLkPrzsHmzEFQQ5A=="],
|
"eslint-plugin-n": ["eslint-plugin-n@17.23.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.5.0", "enhanced-resolve": "^5.17.1", "eslint-plugin-es-x": "^7.8.0", "get-tsconfig": "^4.8.1", "globals": "^15.11.0", "globrex": "^0.1.2", "ignore": "^5.3.2", "semver": "^7.6.3", "ts-declaration-location": "^1.0.6" }, "peerDependencies": { "eslint": ">=8.23.0" } }, "sha512-68PealUpYoHOBh332JLLD9Sj7OQUDkFpmcfqt8R9sySfFSeuGJjMTJQvCRRB96zO3A/PELRLkPrzsHmzEFQQ5A=="],
|
||||||
|
|
||||||
|
"eslint-plugin-react-hooks": ["eslint-plugin-react-hooks@7.0.1", "", { "dependencies": { "@babel/core": "^7.24.4", "@babel/parser": "^7.24.4", "hermes-parser": "^0.25.1", "zod": "^3.25.0 || ^4.0.0", "zod-validation-error": "^3.5.0 || ^4.0.0" }, "peerDependencies": { "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0" } }, "sha512-O0d0m04evaNzEPoSW+59Mezf8Qt0InfgGIBJnpC0h3NH/WjUAR7BIKUfysC6todmtiZ/A0oUVS8Gce0WhBrHsA=="],
|
||||||
|
|
||||||
"eslint-plugin-unused-imports": ["eslint-plugin-unused-imports@4.3.0", "", { "peerDependencies": { "@typescript-eslint/eslint-plugin": "^8.0.0-0 || ^7.0.0 || ^6.0.0 || ^5.0.0", "eslint": "^9.0.0 || ^8.0.0" }, "optionalPeers": ["@typescript-eslint/eslint-plugin"] }, "sha512-ZFBmXMGBYfHttdRtOG9nFFpmUvMtbHSjsKrS20vdWdbfiVYsO3yA2SGYy9i9XmZJDfMGBflZGBCm70SEnFQtOA=="],
|
"eslint-plugin-unused-imports": ["eslint-plugin-unused-imports@4.3.0", "", { "peerDependencies": { "@typescript-eslint/eslint-plugin": "^8.0.0-0 || ^7.0.0 || ^6.0.0 || ^5.0.0", "eslint": "^9.0.0 || ^8.0.0" }, "optionalPeers": ["@typescript-eslint/eslint-plugin"] }, "sha512-ZFBmXMGBYfHttdRtOG9nFFpmUvMtbHSjsKrS20vdWdbfiVYsO3yA2SGYy9i9XmZJDfMGBflZGBCm70SEnFQtOA=="],
|
||||||
|
|
||||||
"eslint-scope": ["eslint-scope@8.4.0", "", { "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg=="],
|
"eslint-scope": ["eslint-scope@8.4.0", "", { "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg=="],
|
||||||
@@ -801,6 +806,8 @@
|
|||||||
|
|
||||||
"for-each": ["for-each@0.3.5", "", { "dependencies": { "is-callable": "^1.2.7" } }, "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg=="],
|
"for-each": ["for-each@0.3.5", "", { "dependencies": { "is-callable": "^1.2.7" } }, "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg=="],
|
||||||
|
|
||||||
|
"framer-motion": ["framer-motion@12.24.7", "", { "dependencies": { "motion-dom": "^12.24.3", "motion-utils": "^12.23.28", "tslib": "^2.4.0" }, "peerDependencies": { "@emotion/is-prop-valid": "*", "react": "^18.0.0 || ^19.0.0", "react-dom": "^18.0.0 || ^19.0.0" }, "optionalPeers": ["@emotion/is-prop-valid", "react", "react-dom"] }, "sha512-EolFLm7NdEMhWO/VTMZ0LlR4fLHGDiJItTx3i8dlyQooOOBoYAaysK4paGD4PrwqnoDdeDOS+TxnSBIAnNHs3w=="],
|
||||||
|
|
||||||
"fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="],
|
"fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="],
|
||||||
|
|
||||||
"function-bind": ["function-bind@1.1.2", "", {}, "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA=="],
|
"function-bind": ["function-bind@1.1.2", "", {}, "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA=="],
|
||||||
@@ -851,7 +858,11 @@
|
|||||||
|
|
||||||
"hasown": ["hasown@2.0.2", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ=="],
|
"hasown": ["hasown@2.0.2", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ=="],
|
||||||
|
|
||||||
"html-encoding-sniffer": ["html-encoding-sniffer@6.0.0", "", { "dependencies": { "@exodus/bytes": "^1.6.0" } }, "sha512-CV9TW3Y3f8/wT0BRFc1/KAVQ3TUHiXmaAb6VW9vtiMFf7SLoMd1PdAc4W3KFOFETBJUb90KatHqlsZMWV+R9Gg=="],
|
"hermes-estree": ["hermes-estree@0.25.1", "", {}, "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw=="],
|
||||||
|
|
||||||
|
"hermes-parser": ["hermes-parser@0.25.1", "", { "dependencies": { "hermes-estree": "0.25.1" } }, "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA=="],
|
||||||
|
|
||||||
|
"html-encoding-sniffer": ["html-encoding-sniffer@4.0.0", "", { "dependencies": { "whatwg-encoding": "^3.1.1" } }, "sha512-Y22oTqIU4uuPgEemfz7NDJz6OeKf12Lsu+QC+s3BVpda64lTiMYCyGwg5ki4vFxkMwQdeZDl2adZoqUgdFuTgQ=="],
|
||||||
|
|
||||||
"http-proxy-agent": ["http-proxy-agent@7.0.2", "", { "dependencies": { "agent-base": "^7.1.0", "debug": "^4.3.4" } }, "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig=="],
|
"http-proxy-agent": ["http-proxy-agent@7.0.2", "", { "dependencies": { "agent-base": "^7.1.0", "debug": "^4.3.4" } }, "sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig=="],
|
||||||
|
|
||||||
@@ -989,7 +1000,7 @@
|
|||||||
|
|
||||||
"lru-cache": ["lru-cache@11.2.4", "", {}, "sha512-B5Y16Jr9LB9dHVkh6ZevG+vAbOsNOYCX+sXvFWFu7B3Iz5mijW3zdbMyhsh8ANd2mSWBYdJgnqi+mL7/LrOPYg=="],
|
"lru-cache": ["lru-cache@11.2.4", "", {}, "sha512-B5Y16Jr9LB9dHVkh6ZevG+vAbOsNOYCX+sXvFWFu7B3Iz5mijW3zdbMyhsh8ANd2mSWBYdJgnqi+mL7/LrOPYg=="],
|
||||||
|
|
||||||
"lucide-react": ["lucide-react@0.561.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Y59gMY38tl4/i0qewcqohPdEbieBy7SovpBL9IFebhc2mDd8x4PZSOsiFRkpPcOq6bj1r/mjH/Rk73gSlIJP2A=="],
|
"lucide-react": ["lucide-react@0.562.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-82hOAu7y0dbVuFfmO4bYF1XEwYk/mEbM5E+b1jgci/udUBEE/R7LF5Ip0CCEmXe8AybRM8L+04eP+LGZeDvkiw=="],
|
||||||
|
|
||||||
"lz-string": ["lz-string@1.5.0", "", { "bin": { "lz-string": "bin/bin.js" } }, "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ=="],
|
"lz-string": ["lz-string@1.5.0", "", { "bin": { "lz-string": "bin/bin.js" } }, "sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ=="],
|
||||||
|
|
||||||
@@ -1003,6 +1014,12 @@
|
|||||||
|
|
||||||
"minimist": ["minimist@1.2.8", "", {}, "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA=="],
|
"minimist": ["minimist@1.2.8", "", {}, "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA=="],
|
||||||
|
|
||||||
|
"motion": ["motion@12.24.7", "", { "dependencies": { "framer-motion": "^12.24.7", "tslib": "^2.4.0" }, "peerDependencies": { "@emotion/is-prop-valid": "*", "react": "^18.0.0 || ^19.0.0", "react-dom": "^18.0.0 || ^19.0.0" }, "optionalPeers": ["@emotion/is-prop-valid", "react", "react-dom"] }, "sha512-0jOoqFlQ7JBvAcRhRv28pwUgZ1xw9WS4+tCU6aqYjxgiNVZCVi34ED2cihW3EgjIIWPBoZJis5og1mx/LmQWVQ=="],
|
||||||
|
|
||||||
|
"motion-dom": ["motion-dom@12.24.3", "", { "dependencies": { "motion-utils": "^12.23.28" } }, "sha512-ZjMZCwhTglim0LM64kC1iFdm4o+2P9IKk3rl/Nb4RKsb5p4O9HJ1C2LWZXOFdsRtp6twpqWRXaFKOduF30ntow=="],
|
||||||
|
|
||||||
|
"motion-utils": ["motion-utils@12.23.28", "", {}, "sha512-0W6cWd5Okoyf8jmessVK3spOmbyE0yTdNKujHctHH9XdAE4QDuZ1/LjSXC68rrhsJU+TkzXURC5OdSWh9ibOwQ=="],
|
||||||
|
|
||||||
"ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="],
|
"ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="],
|
||||||
|
|
||||||
"nanoid": ["nanoid@3.3.11", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w=="],
|
"nanoid": ["nanoid@3.3.11", "", { "bin": { "nanoid": "bin/nanoid.cjs" } }, "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w=="],
|
||||||
@@ -1293,6 +1310,8 @@
|
|||||||
|
|
||||||
"zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="],
|
"zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="],
|
||||||
|
|
||||||
|
"zod-validation-error": ["zod-validation-error@4.0.2", "", { "peerDependencies": { "zod": "^3.25.0 || ^4.0.0" } }, "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ=="],
|
||||||
|
|
||||||
"@babel/helper-compilation-targets/lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="],
|
"@babel/helper-compilation-targets/lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="],
|
||||||
|
|
||||||
"@eslint-community/eslint-utils/eslint-visitor-keys": ["eslint-visitor-keys@3.4.3", "", {}, "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag=="],
|
"@eslint-community/eslint-utils/eslint-visitor-keys": ["eslint-visitor-keys@3.4.3", "", {}, "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag=="],
|
||||||
@@ -1301,6 +1320,10 @@
|
|||||||
|
|
||||||
"@radix-ui/react-arrow/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
"@radix-ui/react-arrow/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
||||||
|
|
||||||
|
"@radix-ui/react-checkbox/@radix-ui/react-context": ["@radix-ui/react-context@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA=="],
|
||||||
|
|
||||||
|
"@radix-ui/react-checkbox/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
||||||
|
|
||||||
"@radix-ui/react-collection/@radix-ui/react-context": ["@radix-ui/react-context@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA=="],
|
"@radix-ui/react-collection/@radix-ui/react-context": ["@radix-ui/react-context@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA=="],
|
||||||
|
|
||||||
"@radix-ui/react-collection/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
"@radix-ui/react-collection/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
||||||
@@ -1333,10 +1356,6 @@
|
|||||||
|
|
||||||
"@radix-ui/react-roving-focus/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
"@radix-ui/react-roving-focus/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
||||||
|
|
||||||
"@radix-ui/react-scroll-area/@radix-ui/react-context": ["@radix-ui/react-context@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA=="],
|
|
||||||
|
|
||||||
"@radix-ui/react-scroll-area/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
|
||||||
|
|
||||||
"@radix-ui/react-select/@radix-ui/react-context": ["@radix-ui/react-context@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA=="],
|
"@radix-ui/react-select/@radix-ui/react-context": ["@radix-ui/react-context@1.1.2", "", { "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-jCi/QKUM2r1Ju5a3J64TH2A5SpKAgh0LpknyqdQ4m6DCV0xJ2HG1xARRwNGPQfi1SLdLWZ1OJz6F4OMBBNiGJA=="],
|
||||||
|
|
||||||
"@radix-ui/react-select/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
"@radix-ui/react-select/@radix-ui/react-primitive": ["@radix-ui/react-primitive@2.1.3", "", { "dependencies": { "@radix-ui/react-slot": "1.2.3" }, "peerDependencies": { "@types/react": "*", "@types/react-dom": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react", "@types/react-dom"] }, "sha512-m9gTwRkhy2lvCPe6QJp4d3G1TYEUHn/FzJUtq9MjH46an1wJU+GdoGC5VLof8RX8Ft/DlpshApkhswDLZzHIcQ=="],
|
||||||
@@ -1417,6 +1436,8 @@
|
|||||||
|
|
||||||
"@radix-ui/react-arrow/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
"@radix-ui/react-arrow/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
|
|
||||||
|
"@radix-ui/react-checkbox/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
|
|
||||||
"@radix-ui/react-dismissable-layer/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
"@radix-ui/react-dismissable-layer/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
|
|
||||||
"@radix-ui/react-focus-scope/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
"@radix-ui/react-focus-scope/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
@@ -1427,8 +1448,6 @@
|
|||||||
|
|
||||||
"@radix-ui/react-roving-focus/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
"@radix-ui/react-roving-focus/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
|
|
||||||
"@radix-ui/react-scroll-area/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
|
||||||
|
|
||||||
"@radix-ui/react-tabs/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
"@radix-ui/react-tabs/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
|
|
||||||
"@radix-ui/react-visually-hidden/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
"@radix-ui/react-visually-hidden/@radix-ui/react-primitive/@radix-ui/react-slot": ["@radix-ui/react-slot@1.2.3", "", { "dependencies": { "@radix-ui/react-compose-refs": "1.1.2" }, "peerDependencies": { "@types/react": "*", "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-aeNmHnBxbi2St0au6VBVC7JXFlhLlOnvIIlePNniyUNAClzmtAUEY8/pBiK3iHjufOlwA+c20/8jngo7xcrg8A=="],
|
||||||
|
|||||||
@@ -17,5 +17,11 @@
|
|||||||
"lib": "@/lib",
|
"lib": "@/lib",
|
||||||
"hooks": "@/hooks"
|
"hooks": "@/hooks"
|
||||||
},
|
},
|
||||||
"iconLibrary": "lucide"
|
"iconLibrary": "lucide",
|
||||||
}
|
"registries": {
|
||||||
|
"@shadcn-studio": "https://shadcnstudio.com/r/{name}.json",
|
||||||
|
"@ss-components": "https://shadcnstudio.com/r/components/{name}.json",
|
||||||
|
"@ss-blocks": "https://shadcnstudio.com/r/blocks/{name}.json",
|
||||||
|
"@ss-themes": "https://shadcnstudio.com/r/themes/{name}.json"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
import { tanstackConfig } from '@tanstack/eslint-config'
|
import { tanstackConfig } from '@tanstack/eslint-config'
|
||||||
import eslintConfigPrettier from 'eslint-config-prettier'
|
import eslintConfigPrettier from 'eslint-config-prettier'
|
||||||
import jsxA11y from 'eslint-plugin-jsx-a11y'
|
import jsxA11y from 'eslint-plugin-jsx-a11y'
|
||||||
|
import reactHooks from 'eslint-plugin-react-hooks'
|
||||||
import unusedImports from 'eslint-plugin-unused-imports'
|
import unusedImports from 'eslint-plugin-unused-imports'
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
@@ -24,9 +25,12 @@ export default [
|
|||||||
|
|
||||||
// 3. TUS REGLAS Y CONFIGURACIÓN "PRO"
|
// 3. TUS REGLAS Y CONFIGURACIÓN "PRO"
|
||||||
{
|
{
|
||||||
|
// Opcional: Puedes ser explícito sobre dónde aplicar esto
|
||||||
|
files: ['**/*.{ts,tsx,js,jsx}'],
|
||||||
plugins: {
|
plugins: {
|
||||||
'jsx-a11y': jsxA11y,
|
'jsx-a11y': jsxA11y,
|
||||||
'unused-imports': unusedImports,
|
'unused-imports': unusedImports,
|
||||||
|
'react-hooks': reactHooks,
|
||||||
},
|
},
|
||||||
// Configuración robusta del Resolver (La versión de Copilot)
|
// Configuración robusta del Resolver (La versión de Copilot)
|
||||||
settings: {
|
settings: {
|
||||||
@@ -44,7 +48,8 @@ export default [
|
|||||||
// --- REGLAS DE ACCESIBILIDAD (A11Y) ---
|
// --- REGLAS DE ACCESIBILIDAD (A11Y) ---
|
||||||
// Activamos las recomendadas manualmente
|
// Activamos las recomendadas manualmente
|
||||||
...jsxA11y.configs.recommended.rules,
|
...jsxA11y.configs.recommended.rules,
|
||||||
|
'react-hooks/rules-of-hooks': 'error',
|
||||||
|
'react-hooks/exhaustive-deps': 'warn',
|
||||||
// --- ORDEN DE IMPORTS ---
|
// --- ORDEN DE IMPORTS ---
|
||||||
'sort-imports': 'off', // Apagamos el nativo
|
'sort-imports': 'off', // Apagamos el nativo
|
||||||
'import/order': [
|
'import/order': [
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
"@radix-ui/react-alert-dialog": "^1.1.15",
|
"@radix-ui/react-alert-dialog": "^1.1.15",
|
||||||
"@radix-ui/react-avatar": "^1.1.11",
|
"@radix-ui/react-avatar": "^1.1.11",
|
||||||
"@radix-ui/react-collapsible": "^1.1.12",
|
"@radix-ui/react-collapsible": "^1.1.12",
|
||||||
|
"@radix-ui/react-checkbox": "^1.3.3",
|
||||||
"@radix-ui/react-dialog": "^1.1.15",
|
"@radix-ui/react-dialog": "^1.1.15",
|
||||||
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
||||||
"@radix-ui/react-label": "^2.1.8",
|
"@radix-ui/react-label": "^2.1.8",
|
||||||
@@ -43,10 +44,11 @@
|
|||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"cmdk": "^1.1.1",
|
"cmdk": "^1.1.1",
|
||||||
"date-fns": "^4.1.0",
|
"date-fns": "^4.1.0",
|
||||||
"lucide-react": "^0.561.0",
|
"lucide-react": "^0.562.0",
|
||||||
|
"motion": "^12.24.7",
|
||||||
"react": "^19.2.0",
|
"react": "^19.2.0",
|
||||||
"react-dom": "^19.2.0",
|
"react-dom": "^19.2.0",
|
||||||
"tailwind-merge": "^3.0.2",
|
"tailwind-merge": "^3.4.0",
|
||||||
"tailwindcss": "^4.0.6",
|
"tailwindcss": "^4.0.6",
|
||||||
"tw-animate-css": "^1.3.6"
|
"tw-animate-css": "^1.3.6"
|
||||||
},
|
},
|
||||||
@@ -63,6 +65,7 @@
|
|||||||
"eslint-import-resolver-typescript": "^4.4.4",
|
"eslint-import-resolver-typescript": "^4.4.4",
|
||||||
"eslint-plugin-import": "^2.32.0",
|
"eslint-plugin-import": "^2.32.0",
|
||||||
"eslint-plugin-jsx-a11y": "^6.10.2",
|
"eslint-plugin-jsx-a11y": "^6.10.2",
|
||||||
|
"eslint-plugin-react-hooks": "^7.0.1",
|
||||||
"eslint-plugin-unused-imports": "^4.3.0",
|
"eslint-plugin-unused-imports": "^4.3.0",
|
||||||
"jsdom": "^27.0.0",
|
"jsdom": "^27.0.0",
|
||||||
"prettier": "^3.5.3",
|
"prettier": "^3.5.3",
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import type {
|
import type {
|
||||||
NewSubjectWizardState,
|
NewSubjectWizardState,
|
||||||
TipoAsignatura,
|
TipoAsignatura,
|
||||||
} from '@/features/asignaturas/new/types'
|
} from '@/features/asignaturas/nueva/types'
|
||||||
|
|
||||||
import { Input } from '@/components/ui/input'
|
import { Input } from '@/components/ui/input'
|
||||||
import { Label } from '@/components/ui/label'
|
import { Label } from '@/components/ui/label'
|
||||||
@@ -15,7 +15,7 @@ import {
|
|||||||
import {
|
import {
|
||||||
ESTRUCTURAS_SEP,
|
ESTRUCTURAS_SEP,
|
||||||
TIPOS_MATERIA,
|
TIPOS_MATERIA,
|
||||||
} from '@/features/asignaturas/new/catalogs'
|
} from '@/features/asignaturas/nueva/catalogs'
|
||||||
|
|
||||||
export function PasoBasicosForm({
|
export function PasoBasicosForm({
|
||||||
wizard,
|
wizard,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import * as Icons from 'lucide-react'
|
import * as Icons from 'lucide-react'
|
||||||
|
|
||||||
import type { NewSubjectWizardState } from '@/features/asignaturas/new/types'
|
import type { NewSubjectWizardState } from '@/features/asignaturas/nueva/types'
|
||||||
|
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
import {
|
import {
|
||||||
@@ -25,7 +25,7 @@ import {
|
|||||||
FACULTADES,
|
FACULTADES,
|
||||||
MATERIAS_MOCK,
|
MATERIAS_MOCK,
|
||||||
PLANES_MOCK,
|
PLANES_MOCK,
|
||||||
} from '@/features/asignaturas/new/catalogs'
|
} from '@/features/asignaturas/nueva/catalogs'
|
||||||
|
|
||||||
export function PasoConfiguracionPanel({
|
export function PasoConfiguracionPanel({
|
||||||
wizard,
|
wizard,
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import type {
|
|||||||
ModoCreacion,
|
ModoCreacion,
|
||||||
NewSubjectWizardState,
|
NewSubjectWizardState,
|
||||||
SubModoClonado,
|
SubModoClonado,
|
||||||
} from '@/features/asignaturas/new/types'
|
} from '@/features/asignaturas/nueva/types'
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Card,
|
Card,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import * as Icons from 'lucide-react'
|
import * as Icons from 'lucide-react'
|
||||||
|
|
||||||
import type { NewSubjectWizardState } from '@/features/asignaturas/new/types'
|
import type { NewSubjectWizardState } from '@/features/asignaturas/nueva/types'
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Card,
|
Card,
|
||||||
@@ -9,7 +9,7 @@ import {
|
|||||||
CardHeader,
|
CardHeader,
|
||||||
CardTitle,
|
CardTitle,
|
||||||
} from '@/components/ui/card'
|
} from '@/components/ui/card'
|
||||||
import { ESTRUCTURAS_SEP } from '@/features/asignaturas/new/catalogs'
|
import { ESTRUCTURAS_SEP } from '@/features/asignaturas/nueva/catalogs'
|
||||||
|
|
||||||
export function PasoResumenCard({ wizard }: { wizard: NewSubjectWizardState }) {
|
export function PasoResumenCard({ wizard }: { wizard: NewSubjectWizardState }) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { NewSubjectWizardState } from '@/features/asignaturas/new/types'
|
import type { NewSubjectWizardState } from '@/features/asignaturas/nueva/types'
|
||||||
|
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
|
|
||||||
|
|||||||
@@ -1,181 +0,0 @@
|
|||||||
import type { CARRERAS } from '@/features/planes/new/catalogs'
|
|
||||||
import type { NewPlanWizardState } from '@/features/planes/new/types'
|
|
||||||
|
|
||||||
import { Input } from '@/components/ui/input'
|
|
||||||
import { Label } from '@/components/ui/label'
|
|
||||||
import {
|
|
||||||
Select,
|
|
||||||
SelectContent,
|
|
||||||
SelectItem,
|
|
||||||
SelectTrigger,
|
|
||||||
SelectValue,
|
|
||||||
} from '@/components/ui/select'
|
|
||||||
import {
|
|
||||||
FACULTADES,
|
|
||||||
NIVELES,
|
|
||||||
TIPOS_CICLO,
|
|
||||||
} from '@/features/planes/new/catalogs'
|
|
||||||
|
|
||||||
export function PasoBasicosForm({
|
|
||||||
wizard,
|
|
||||||
onChange,
|
|
||||||
carrerasFiltradas,
|
|
||||||
}: {
|
|
||||||
wizard: NewPlanWizardState
|
|
||||||
onChange: React.Dispatch<React.SetStateAction<NewPlanWizardState>>
|
|
||||||
carrerasFiltradas: typeof CARRERAS
|
|
||||||
}) {
|
|
||||||
return (
|
|
||||||
<div className="grid gap-4 sm:grid-cols-2">
|
|
||||||
<div className="grid gap-1 sm:col-span-2">
|
|
||||||
<Label htmlFor="nombrePlan">Nombre del plan</Label>
|
|
||||||
<Input
|
|
||||||
id="nombrePlan"
|
|
||||||
placeholder="Ej. Ingeniería en Sistemas 2026"
|
|
||||||
value={wizard.datosBasicos.nombrePlan}
|
|
||||||
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
datosBasicos: { ...w.datosBasicos, nombrePlan: e.target.value },
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="grid gap-1">
|
|
||||||
<Label htmlFor="facultad">Facultad</Label>
|
|
||||||
<Select
|
|
||||||
value={wizard.datosBasicos.facultadId}
|
|
||||||
onValueChange={(value) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
datosBasicos: {
|
|
||||||
...w.datosBasicos,
|
|
||||||
facultadId: value,
|
|
||||||
carreraId: '',
|
|
||||||
},
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<SelectTrigger
|
|
||||||
id="facultad"
|
|
||||||
className="w-full min-w-0 [&>span]:block! [&>span]:truncate!"
|
|
||||||
>
|
|
||||||
<SelectValue placeholder="Selecciona facultad…" />
|
|
||||||
</SelectTrigger>
|
|
||||||
<SelectContent>
|
|
||||||
{FACULTADES.map((f) => (
|
|
||||||
<SelectItem key={f.id} value={f.id}>
|
|
||||||
{f.nombre}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</SelectContent>
|
|
||||||
</Select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="grid gap-1">
|
|
||||||
<Label htmlFor="carrera">Carrera</Label>
|
|
||||||
<Select
|
|
||||||
value={wizard.datosBasicos.carreraId}
|
|
||||||
onValueChange={(value) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
datosBasicos: { ...w.datosBasicos, carreraId: value },
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
disabled={!wizard.datosBasicos.facultadId}
|
|
||||||
>
|
|
||||||
<SelectTrigger
|
|
||||||
id="carrera"
|
|
||||||
className="w-full min-w-0 [&>span]:block! [&>span]:truncate!"
|
|
||||||
>
|
|
||||||
<SelectValue placeholder="Selecciona carrera…" />
|
|
||||||
</SelectTrigger>
|
|
||||||
<SelectContent>
|
|
||||||
{carrerasFiltradas.map((c) => (
|
|
||||||
<SelectItem key={c.id} value={c.id}>
|
|
||||||
{c.nombre}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</SelectContent>
|
|
||||||
</Select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="grid gap-1">
|
|
||||||
<Label htmlFor="nivel">Nivel</Label>
|
|
||||||
<Select
|
|
||||||
value={wizard.datosBasicos.nivel}
|
|
||||||
onValueChange={(value) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
datosBasicos: { ...w.datosBasicos, nivel: value },
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<SelectTrigger
|
|
||||||
id="nivel"
|
|
||||||
className="w-full min-w-0 [&>span]:block! [&>span]:truncate!"
|
|
||||||
>
|
|
||||||
<SelectValue placeholder="Selecciona nivel…" />
|
|
||||||
</SelectTrigger>
|
|
||||||
<SelectContent>
|
|
||||||
{NIVELES.map((n) => (
|
|
||||||
<SelectItem key={n} value={n}>
|
|
||||||
{n}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</SelectContent>
|
|
||||||
</Select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="grid gap-1">
|
|
||||||
<Label htmlFor="tipoCiclo">Tipo de ciclo</Label>
|
|
||||||
<Select
|
|
||||||
value={wizard.datosBasicos.tipoCiclo}
|
|
||||||
onValueChange={(value) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
datosBasicos: {
|
|
||||||
...w.datosBasicos,
|
|
||||||
tipoCiclo: value as any,
|
|
||||||
},
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<SelectTrigger
|
|
||||||
id="tipoCiclo"
|
|
||||||
className="w-full min-w-0 [&>span]:block! [&>span]:truncate!"
|
|
||||||
>
|
|
||||||
<SelectValue />
|
|
||||||
</SelectTrigger>
|
|
||||||
<SelectContent>
|
|
||||||
{TIPOS_CICLO.map((t) => (
|
|
||||||
<SelectItem key={t.value} value={t.value}>
|
|
||||||
{t.label}
|
|
||||||
</SelectItem>
|
|
||||||
))}
|
|
||||||
</SelectContent>
|
|
||||||
</Select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="grid gap-1">
|
|
||||||
<Label htmlFor="numCiclos">Número de ciclos</Label>
|
|
||||||
<Input
|
|
||||||
id="numCiclos"
|
|
||||||
type="number"
|
|
||||||
min={1}
|
|
||||||
value={wizard.datosBasicos.numCiclos}
|
|
||||||
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
datosBasicos: {
|
|
||||||
...w.datosBasicos,
|
|
||||||
numCiclos: Number(e.target.value || 1),
|
|
||||||
},
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
253
src/components/planes/wizard/PasoBasicosForm/PasoBasicosForm.tsx
Normal file
253
src/components/planes/wizard/PasoBasicosForm/PasoBasicosForm.tsx
Normal file
@@ -0,0 +1,253 @@
|
|||||||
|
import { TemplateSelectorCard } from './TemplateSelectorCard'
|
||||||
|
|
||||||
|
import type { CARRERAS } from '@/features/planes/nuevo/catalogs'
|
||||||
|
import type { NewPlanWizardState } from '@/features/planes/nuevo/types'
|
||||||
|
|
||||||
|
import { Input } from '@/components/ui/input'
|
||||||
|
import { Label } from '@/components/ui/label'
|
||||||
|
import {
|
||||||
|
Select,
|
||||||
|
SelectContent,
|
||||||
|
SelectItem,
|
||||||
|
SelectTrigger,
|
||||||
|
SelectValue,
|
||||||
|
} from '@/components/ui/select'
|
||||||
|
import { Separator } from '@/components/ui/separator'
|
||||||
|
import {
|
||||||
|
FACULTADES,
|
||||||
|
NIVELES,
|
||||||
|
TIPOS_CICLO,
|
||||||
|
PLANTILLAS_ANEXO_1,
|
||||||
|
PLANTILLAS_ANEXO_2,
|
||||||
|
} from '@/features/planes/nuevo/catalogs'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
export function PasoBasicosForm({
|
||||||
|
wizard,
|
||||||
|
onChange,
|
||||||
|
carrerasFiltradas,
|
||||||
|
}: {
|
||||||
|
wizard: NewPlanWizardState
|
||||||
|
onChange: React.Dispatch<React.SetStateAction<NewPlanWizardState>>
|
||||||
|
carrerasFiltradas: typeof CARRERAS
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
<div className="grid gap-4 sm:grid-cols-2">
|
||||||
|
<div className="grid gap-1 sm:col-span-2">
|
||||||
|
<Label htmlFor="nombrePlan">
|
||||||
|
Nombre del plan <span className="text-destructive">*</span>
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="nombrePlan"
|
||||||
|
placeholder="Ej. Ingeniería en Sistemas 2026"
|
||||||
|
value={wizard.datosBasicos.nombrePlan}
|
||||||
|
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: { ...w.datosBasicos, nombrePlan: e.target.value },
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
className="placeholder:text-muted-foreground/70 font-medium not-italic placeholder:font-normal placeholder:italic"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid gap-1">
|
||||||
|
<Label htmlFor="facultad">Facultad</Label>
|
||||||
|
<Select
|
||||||
|
value={wizard.datosBasicos.facultadId}
|
||||||
|
onValueChange={(value) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: {
|
||||||
|
...w.datosBasicos,
|
||||||
|
facultadId: value,
|
||||||
|
carreraId: '',
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<SelectTrigger
|
||||||
|
id="facultad"
|
||||||
|
className={cn(
|
||||||
|
'w-full min-w-0 [&>span]:block! [&>span]:truncate!',
|
||||||
|
!wizard.datosBasicos.facultadId
|
||||||
|
? 'text-muted-foreground font-normal italic opacity-70' // Es Placeholder
|
||||||
|
: 'font-medium not-italic', // Tiene Valor (Medium)
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<SelectValue placeholder="Ej. Facultad de Ingeniería" />
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
{FACULTADES.map((f) => (
|
||||||
|
<SelectItem key={f.id} value={f.id}>
|
||||||
|
{f.nombre}
|
||||||
|
</SelectItem>
|
||||||
|
))}
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid gap-1">
|
||||||
|
<Label htmlFor="carrera">Carrera</Label>
|
||||||
|
<Select
|
||||||
|
value={wizard.datosBasicos.carreraId}
|
||||||
|
onValueChange={(value) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: { ...w.datosBasicos, carreraId: value },
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
disabled={!wizard.datosBasicos.facultadId}
|
||||||
|
>
|
||||||
|
<SelectTrigger
|
||||||
|
id="carrera"
|
||||||
|
className={cn(
|
||||||
|
'w-full min-w-0 [&>span]:block! [&>span]:truncate!',
|
||||||
|
!wizard.datosBasicos.carreraId
|
||||||
|
? 'text-muted-foreground font-normal italic opacity-70' // Es Placeholder
|
||||||
|
: 'font-medium not-italic', // Tiene Valor (Medium)
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<SelectValue placeholder="Ej. Ingeniería en Cibernética y Sistemas Computacionales" />
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
{carrerasFiltradas.map((c) => (
|
||||||
|
<SelectItem key={c.id} value={c.id}>
|
||||||
|
{c.nombre}
|
||||||
|
</SelectItem>
|
||||||
|
))}
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid gap-1">
|
||||||
|
<Label htmlFor="nivel">Nivel</Label>
|
||||||
|
<Select
|
||||||
|
value={wizard.datosBasicos.nivel}
|
||||||
|
onValueChange={(value) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: { ...w.datosBasicos, nivel: value },
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<SelectTrigger
|
||||||
|
id="nivel"
|
||||||
|
className={cn(
|
||||||
|
'w-full min-w-0 [&>span]:block! [&>span]:truncate!',
|
||||||
|
!wizard.datosBasicos.nivel
|
||||||
|
? 'text-muted-foreground font-normal italic opacity-70' // Es Placeholder
|
||||||
|
: 'font-medium not-italic', // Tiene Valor (Medium)
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<SelectValue placeholder="Ej. Licenciatura" />
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
{NIVELES.map((n) => (
|
||||||
|
<SelectItem key={n} value={n}>
|
||||||
|
{n}
|
||||||
|
</SelectItem>
|
||||||
|
))}
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid gap-1">
|
||||||
|
<Label htmlFor="tipoCiclo">Tipo de ciclo</Label>
|
||||||
|
<Select
|
||||||
|
value={wizard.datosBasicos.tipoCiclo}
|
||||||
|
onValueChange={(value) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: {
|
||||||
|
...w.datosBasicos,
|
||||||
|
tipoCiclo: value as any,
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<SelectTrigger
|
||||||
|
id="tipoCiclo"
|
||||||
|
className={cn(
|
||||||
|
'w-full min-w-0 [&>span]:block! [&>span]:truncate!',
|
||||||
|
!wizard.datosBasicos.tipoCiclo
|
||||||
|
? 'text-muted-foreground font-normal italic opacity-70' // Es Placeholder
|
||||||
|
: 'font-medium not-italic', // Tiene Valor (Medium)
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<SelectValue placeholder="Ej. Semestre" />
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
{TIPOS_CICLO.map((t) => (
|
||||||
|
<SelectItem key={t.value} value={t.value}>
|
||||||
|
{t.label}
|
||||||
|
</SelectItem>
|
||||||
|
))}
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid gap-1">
|
||||||
|
<Label htmlFor="numCiclos">Número de ciclos</Label>
|
||||||
|
<Input
|
||||||
|
id="numCiclos"
|
||||||
|
type="number"
|
||||||
|
min={1}
|
||||||
|
value={wizard.datosBasicos.numCiclos ?? ''}
|
||||||
|
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: {
|
||||||
|
...w.datosBasicos,
|
||||||
|
// Keep undefined when the input is empty so the field stays optional
|
||||||
|
numCiclos:
|
||||||
|
e.target.value === '' ? undefined : Number(e.target.value),
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
className="placeholder:text-muted-foreground/70 font-medium not-italic placeholder:font-normal placeholder:italic"
|
||||||
|
placeholder="Ej. 8"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Separator className="my-3" />
|
||||||
|
<div className="grid gap-4 sm:grid-cols-2">
|
||||||
|
<TemplateSelectorCard
|
||||||
|
cardTitle="Plantilla de plan de estudios"
|
||||||
|
cardDescription="Selecciona el Word para tu nuevo plan."
|
||||||
|
templatesData={PLANTILLAS_ANEXO_1}
|
||||||
|
selectedTemplateId={wizard.datosBasicos.plantillaPlanId || ''}
|
||||||
|
selectedVersion={wizard.datosBasicos.plantillaPlanVersion || ''}
|
||||||
|
onChange={({ templateId, version }) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: {
|
||||||
|
...w.datosBasicos,
|
||||||
|
plantillaPlanId: templateId,
|
||||||
|
plantillaPlanVersion: version,
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<TemplateSelectorCard
|
||||||
|
cardTitle="Plantilla de mapa curricular"
|
||||||
|
cardDescription="Selecciona el Excel para tu mapa curricular."
|
||||||
|
templatesData={PLANTILLAS_ANEXO_2}
|
||||||
|
selectedTemplateId={wizard.datosBasicos.plantillaMapaId || ''}
|
||||||
|
selectedVersion={wizard.datosBasicos.plantillaMapaVersion || ''}
|
||||||
|
onChange={({ templateId, version }) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
datosBasicos: {
|
||||||
|
...w.datosBasicos,
|
||||||
|
plantillaMapaId: templateId,
|
||||||
|
plantillaMapaVersion: version,
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -0,0 +1,189 @@
|
|||||||
|
import { useMemo, useState } from 'react'
|
||||||
|
|
||||||
|
import {
|
||||||
|
Card,
|
||||||
|
CardContent,
|
||||||
|
CardHeader,
|
||||||
|
CardTitle,
|
||||||
|
CardDescription,
|
||||||
|
} from '@/components/ui/card'
|
||||||
|
import { Label } from '@/components/ui/label'
|
||||||
|
import {
|
||||||
|
Select,
|
||||||
|
SelectContent,
|
||||||
|
SelectItem,
|
||||||
|
SelectTrigger,
|
||||||
|
SelectValue,
|
||||||
|
} from '@/components/ui/select'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
type TemplateData = {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
versions: Array<string>
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default data (kept for backward compatibility if caller doesn't pass templates)
|
||||||
|
const DEFAULT_TEMPLATES_DATA: Array<TemplateData> = [
|
||||||
|
{
|
||||||
|
id: 'sep-2025',
|
||||||
|
name: 'Licenciatura RVOE SEP',
|
||||||
|
versions: ['v2025.2 (Vigente)', 'v2025.1', 'v2024.Final'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'interno-mix',
|
||||||
|
name: 'Estándar Institucional Mixto',
|
||||||
|
versions: ['v2.0', 'v1.5', 'v1.0-beta'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'conacyt',
|
||||||
|
name: 'Formato Posgrado CONAHCYT',
|
||||||
|
versions: ['v3.0 (2025)', 'v2.8'],
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
cardTitle?: string
|
||||||
|
cardDescription?: string
|
||||||
|
templatesData?: Array<TemplateData>
|
||||||
|
// Controlled selection (optional). If not provided, component manages its own state
|
||||||
|
selectedTemplateId?: string
|
||||||
|
selectedVersion?: string
|
||||||
|
onChange?: (sel: { templateId: string; version: string }) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export function TemplateSelectorCard({
|
||||||
|
cardTitle = 'Configuración del Documento',
|
||||||
|
cardDescription = 'Selecciona la base para tu nuevo plan.',
|
||||||
|
templatesData = DEFAULT_TEMPLATES_DATA,
|
||||||
|
selectedTemplateId,
|
||||||
|
selectedVersion,
|
||||||
|
onChange,
|
||||||
|
}: Props) {
|
||||||
|
const [internalTemplate, setInternalTemplate] = useState<string>('')
|
||||||
|
const [internalVersion, setInternalVersion] = useState<string>('')
|
||||||
|
|
||||||
|
const selectedTemplate = selectedTemplateId ?? internalTemplate
|
||||||
|
const version = selectedVersion ?? internalVersion
|
||||||
|
|
||||||
|
// Buscamos las versiones de la plantilla seleccionada
|
||||||
|
const currentTemplateData = useMemo(
|
||||||
|
() => templatesData.find((t) => t.id === selectedTemplate),
|
||||||
|
[templatesData, selectedTemplate],
|
||||||
|
)
|
||||||
|
const availableVersions = currentTemplateData?.versions || []
|
||||||
|
|
||||||
|
const handleTemplateChange = (value: string) => {
|
||||||
|
const template = templatesData.find((t) => t.id === value)
|
||||||
|
const firstVersion = template?.versions?.[0] ?? ''
|
||||||
|
if (onChange) {
|
||||||
|
onChange({ templateId: value, version: firstVersion })
|
||||||
|
} else {
|
||||||
|
setInternalTemplate(value)
|
||||||
|
setInternalVersion(firstVersion)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleVersionChange = (value: string) => {
|
||||||
|
if (onChange) {
|
||||||
|
onChange({ templateId: selectedTemplate, version: value })
|
||||||
|
} else {
|
||||||
|
setInternalVersion(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card className="w-full max-w-lg gap-2 overflow-hidden">
|
||||||
|
<CardHeader className="px-4 pb-2 sm:px-6 sm:pb-4">
|
||||||
|
<CardTitle className="text-lg">{cardTitle}</CardTitle>
|
||||||
|
<CardDescription>{cardDescription}</CardDescription>
|
||||||
|
</CardHeader>
|
||||||
|
|
||||||
|
<CardContent className="flex flex-col gap-4">
|
||||||
|
{/* SELECT 1: PRIMARIO (Llamativo) */}
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label
|
||||||
|
htmlFor="template-select"
|
||||||
|
className="text-foreground text-base font-semibold"
|
||||||
|
>
|
||||||
|
Plantilla
|
||||||
|
</Label>
|
||||||
|
<Select value={selectedTemplate} onValueChange={handleTemplateChange}>
|
||||||
|
<SelectTrigger
|
||||||
|
id="template-select"
|
||||||
|
className="bg-background border-primary/40 focus:ring-primary/20 focus:border-primary flex h-11 w-full min-w-0 items-center justify-between gap-2 text-base shadow-sm [&>span]:block! [&>span]:truncate! [&>span]:text-left"
|
||||||
|
>
|
||||||
|
<SelectValue placeholder="Selecciona una plantilla..." />
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
{templatesData.map((t) => (
|
||||||
|
<SelectItem key={t.id} value={t.id} className="font-medium">
|
||||||
|
{t.name}
|
||||||
|
</SelectItem>
|
||||||
|
))}
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* SELECT 2: SECUNDARIO (Sutil) */}
|
||||||
|
<div className="space-y-1.5">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<Label
|
||||||
|
htmlFor="version-select"
|
||||||
|
className={cn(
|
||||||
|
'text-xs tracking-wider uppercase transition-colors',
|
||||||
|
!selectedTemplate
|
||||||
|
? 'text-muted-foreground/50'
|
||||||
|
: 'text-muted-foreground',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
Versión
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Select
|
||||||
|
value={version}
|
||||||
|
onValueChange={handleVersionChange}
|
||||||
|
disabled={!selectedTemplate}
|
||||||
|
>
|
||||||
|
<SelectTrigger
|
||||||
|
id="version-select"
|
||||||
|
className={cn(
|
||||||
|
'flex h-9 min-w-0 items-center justify-between gap-2 text-sm transition-all duration-300',
|
||||||
|
/* AQUÍ ESTÁ EL CAMBIO DE ANCHO: */
|
||||||
|
'w-full max-w-full sm:w-55',
|
||||||
|
|
||||||
|
/* Las correcciones vitales para truncado que ya teníamos: */
|
||||||
|
'min-w-0 [&>span]:block! [&>span]:truncate! [&>span]:text-left',
|
||||||
|
'[&>span]:block [&>span]:min-w-0 [&>span]:truncate [&>span]:text-left',
|
||||||
|
|
||||||
|
!selectedTemplate
|
||||||
|
? 'bg-muted/50 cursor-not-allowed border-transparent opacity-50'
|
||||||
|
: 'bg-muted/20 border-border hover:bg-background hover:border-primary/30',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<SelectValue
|
||||||
|
placeholder={
|
||||||
|
!selectedTemplate
|
||||||
|
? '— Esperando plantilla —'
|
||||||
|
: 'Selecciona versión'
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</SelectTrigger>
|
||||||
|
<SelectContent>
|
||||||
|
{availableVersions.map((v) => (
|
||||||
|
<SelectItem
|
||||||
|
key={v}
|
||||||
|
value={v}
|
||||||
|
className="text-muted-foreground focus:text-foreground text-sm"
|
||||||
|
>
|
||||||
|
{v}
|
||||||
|
</SelectItem>
|
||||||
|
))}
|
||||||
|
</SelectContent>
|
||||||
|
</Select>
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
)
|
||||||
|
}
|
||||||
205
src/components/planes/wizard/PasoDetallesPanel/FileDropZone.tsx
Normal file
205
src/components/planes/wizard/PasoDetallesPanel/FileDropZone.tsx
Normal file
@@ -0,0 +1,205 @@
|
|||||||
|
import { Upload, File, X, FileText } from 'lucide-react'
|
||||||
|
import { useState, useCallback, useEffect, useRef } from 'react'
|
||||||
|
|
||||||
|
import { Button } from '@/components/ui/button'
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
interface UploadedFile {
|
||||||
|
id: string
|
||||||
|
name: string
|
||||||
|
size: string
|
||||||
|
type: string
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileDropzoneProps {
|
||||||
|
onFilesChange?: (files: Array<UploadedFile>) => void
|
||||||
|
acceptedTypes?: string
|
||||||
|
maxFiles?: number
|
||||||
|
title?: string
|
||||||
|
description?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FileDropzone({
|
||||||
|
onFilesChange,
|
||||||
|
acceptedTypes = '.doc,.docx,.pdf',
|
||||||
|
maxFiles = 5,
|
||||||
|
title = 'Arrastra archivos aquí',
|
||||||
|
description = 'o haz clic para seleccionar',
|
||||||
|
}: FileDropzoneProps) {
|
||||||
|
const [isDragging, setIsDragging] = useState(false)
|
||||||
|
const [files, setFiles] = useState<Array<UploadedFile>>([])
|
||||||
|
const onFilesChangeRef = useRef<typeof onFilesChange>(onFilesChange)
|
||||||
|
|
||||||
|
const addFiles = useCallback(
|
||||||
|
(newFiles: Array<File>) => {
|
||||||
|
const toUpload: Array<UploadedFile> = newFiles.map((file) => ({
|
||||||
|
id:
|
||||||
|
typeof crypto !== 'undefined' && 'randomUUID' in crypto
|
||||||
|
? (crypto as any).randomUUID()
|
||||||
|
: `file-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,
|
||||||
|
name: file.name,
|
||||||
|
size: formatFileSize(file.size),
|
||||||
|
type: file.name.split('.').pop() || 'file',
|
||||||
|
}))
|
||||||
|
setFiles((prev) => {
|
||||||
|
const room = Math.max(0, maxFiles - prev.length)
|
||||||
|
const next = [...prev, ...toUpload.slice(0, room)].slice(0, maxFiles)
|
||||||
|
return next
|
||||||
|
})
|
||||||
|
},
|
||||||
|
[maxFiles],
|
||||||
|
)
|
||||||
|
|
||||||
|
const handleDragOver = useCallback((e: React.DragEvent) => {
|
||||||
|
e.preventDefault()
|
||||||
|
setIsDragging(true)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const handleDragLeave = useCallback((e: React.DragEvent) => {
|
||||||
|
e.preventDefault()
|
||||||
|
setIsDragging(false)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const handleDrop = useCallback(
|
||||||
|
(e: React.DragEvent) => {
|
||||||
|
e.preventDefault()
|
||||||
|
setIsDragging(false)
|
||||||
|
const droppedFiles = Array.from(e.dataTransfer.files)
|
||||||
|
addFiles(droppedFiles)
|
||||||
|
},
|
||||||
|
[addFiles],
|
||||||
|
)
|
||||||
|
|
||||||
|
const handleFileInput = useCallback(
|
||||||
|
(e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
if (e.target.files) {
|
||||||
|
const selectedFiles = Array.from(e.target.files)
|
||||||
|
addFiles(selectedFiles)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[addFiles],
|
||||||
|
)
|
||||||
|
|
||||||
|
const removeFile = useCallback((fileId: string) => {
|
||||||
|
setFiles((prev) => {
|
||||||
|
const next = prev.filter((f) => f.id !== fileId)
|
||||||
|
return next
|
||||||
|
})
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
// Keep latest callback in a ref to avoid retriggering effect on identity change
|
||||||
|
useEffect(() => {
|
||||||
|
onFilesChangeRef.current = onFilesChange
|
||||||
|
}, [onFilesChange])
|
||||||
|
|
||||||
|
// Only emit when files actually change to avoid parent update loops
|
||||||
|
useEffect(() => {
|
||||||
|
if (onFilesChangeRef.current) onFilesChangeRef.current(files)
|
||||||
|
}, [files])
|
||||||
|
|
||||||
|
const formatFileSize = (bytes: number): string => {
|
||||||
|
if (bytes < 1024) return bytes + ' B'
|
||||||
|
if (bytes < 1024 * 1024) return (bytes / 1024).toFixed(1) + ' KB'
|
||||||
|
return (bytes / (1024 * 1024)).toFixed(1) + ' MB'
|
||||||
|
}
|
||||||
|
|
||||||
|
const getFileIcon = (type: string) => {
|
||||||
|
switch (type.toLowerCase()) {
|
||||||
|
case 'pdf':
|
||||||
|
return <FileText className="text-destructive h-4 w-4" />
|
||||||
|
case 'doc':
|
||||||
|
case 'docx':
|
||||||
|
return <FileText className="text-info h-4 w-4" />
|
||||||
|
default:
|
||||||
|
return <File className="text-muted-foreground h-4 w-4" />
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="space-y-3">
|
||||||
|
<div
|
||||||
|
onDragOver={handleDragOver}
|
||||||
|
onDragLeave={handleDragLeave}
|
||||||
|
onDrop={handleDrop}
|
||||||
|
className={cn(
|
||||||
|
'border-border hover:border-primary/50 cursor-pointer rounded-xl border-2 border-dashed p-8 text-center transition-all duration-300',
|
||||||
|
isDragging && 'active',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
type="file"
|
||||||
|
accept={acceptedTypes}
|
||||||
|
multiple
|
||||||
|
onChange={handleFileInput}
|
||||||
|
className="hidden"
|
||||||
|
id="file-upload"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
htmlFor="file-upload"
|
||||||
|
className="cursor-pointer"
|
||||||
|
aria-label="Seleccionar archivos"
|
||||||
|
>
|
||||||
|
<div className="flex flex-col items-center gap-3">
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
'flex h-12 w-12 items-center justify-center rounded-xl transition-colors',
|
||||||
|
isDragging
|
||||||
|
? 'bg-primary text-primary-foreground'
|
||||||
|
: 'bg-accent text-accent-foreground',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<Upload className="h-6 w-6" />
|
||||||
|
</div>
|
||||||
|
<div className="text-center">
|
||||||
|
<p className="text-foreground text-sm font-medium">{title}</p>
|
||||||
|
<p className="text-muted-foreground mt-1 text-xs">
|
||||||
|
{description}
|
||||||
|
</p>
|
||||||
|
<p className="text-muted-foreground mt-1 text-xs">
|
||||||
|
Formatos:{' '}
|
||||||
|
{acceptedTypes
|
||||||
|
.replace(/\./g, '')
|
||||||
|
.toUpperCase()
|
||||||
|
.replace(/,/g, ', ')}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Uploaded files list */}
|
||||||
|
{files.length > 0 && (
|
||||||
|
<div className="space-y-2">
|
||||||
|
{files.map((file) => (
|
||||||
|
<div
|
||||||
|
key={file.id}
|
||||||
|
className="bg-accent/50 border-border fade-in flex items-center gap-3 rounded-lg border p-3"
|
||||||
|
>
|
||||||
|
{getFileIcon(file.type)}
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<p className="text-foreground truncate text-sm font-medium">
|
||||||
|
{file.name}
|
||||||
|
</p>
|
||||||
|
<p className="text-muted-foreground text-xs">{file.size}</p>
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
className="text-muted-foreground hover:text-destructive h-8 w-8"
|
||||||
|
onClick={() => removeFile(file.id)}
|
||||||
|
>
|
||||||
|
<X className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{files.length >= maxFiles && (
|
||||||
|
<p className="text-warning text-center text-xs">
|
||||||
|
Máximo de {maxFiles} archivos alcanzado
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,4 +1,7 @@
|
|||||||
import type { NewPlanWizardState } from '@/features/planes/new/types'
|
import { FileDropzone } from './FileDropZone'
|
||||||
|
import ReferenciasParaIA from './ReferenciasParaIA'
|
||||||
|
|
||||||
|
import type { NewPlanWizardState } from '@/features/planes/nuevo/types'
|
||||||
|
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
import {
|
import {
|
||||||
@@ -14,7 +17,7 @@ import {
|
|||||||
CARRERAS,
|
CARRERAS,
|
||||||
FACULTADES,
|
FACULTADES,
|
||||||
PLANES_EXISTENTES,
|
PLANES_EXISTENTES,
|
||||||
} from '@/features/planes/new/catalogs'
|
} from '@/features/planes/nuevo/catalogs'
|
||||||
|
|
||||||
export function PasoDetallesPanel({
|
export function PasoDetallesPanel({
|
||||||
wizard,
|
wizard,
|
||||||
@@ -42,8 +45,8 @@ export function PasoDetallesPanel({
|
|||||||
|
|
||||||
if (wizard.modoCreacion === 'IA') {
|
if (wizard.modoCreacion === 'IA') {
|
||||||
return (
|
return (
|
||||||
<div className="grid gap-4">
|
<div className="flex flex-col gap-4">
|
||||||
<div>
|
<div className="flex flex-col gap-1">
|
||||||
<Label htmlFor="desc">Descripción del enfoque</Label>
|
<Label htmlFor="desc">Descripción del enfoque</Label>
|
||||||
<textarea
|
<textarea
|
||||||
id="desc"
|
id="desc"
|
||||||
@@ -61,24 +64,8 @@ export function PasoDetallesPanel({
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
|
||||||
<Label htmlFor="poblacion">Población objetivo</Label>
|
<div className="flex flex-col gap-1">
|
||||||
<Input
|
|
||||||
id="poblacion"
|
|
||||||
placeholder="Ej. Egresados de bachillerato con perfil STEM"
|
|
||||||
value={wizard.iaConfig?.poblacionObjetivo || ''}
|
|
||||||
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
|
||||||
onChange((w) => ({
|
|
||||||
...w,
|
|
||||||
iaConfig: {
|
|
||||||
...(w.iaConfig || ({} as any)),
|
|
||||||
poblacionObjetivo: e.target.value,
|
|
||||||
},
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<Label htmlFor="notas">Notas adicionales</Label>
|
<Label htmlFor="notas">Notas adicionales</Label>
|
||||||
<textarea
|
<textarea
|
||||||
id="notas"
|
id="notas"
|
||||||
@@ -96,6 +83,49 @@ export function PasoDetallesPanel({
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
<ReferenciasParaIA
|
||||||
|
selectedArchivoIds={wizard.iaConfig?.archivosReferencia || []}
|
||||||
|
selectedRepositorioIds={wizard.iaConfig?.repositoriosReferencia || []}
|
||||||
|
onToggleArchivo={(id, checked) =>
|
||||||
|
onChange((w) => {
|
||||||
|
const prev = w.iaConfig?.archivosReferencia || []
|
||||||
|
const next = checked
|
||||||
|
? [...prev, id]
|
||||||
|
: prev.filter((x) => x !== id)
|
||||||
|
return {
|
||||||
|
...w,
|
||||||
|
iaConfig: {
|
||||||
|
...(w.iaConfig || ({} as any)),
|
||||||
|
archivosReferencia: next,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
onToggleRepositorio={(id, checked) =>
|
||||||
|
onChange((w) => {
|
||||||
|
const prev = w.iaConfig?.repositoriosReferencia || []
|
||||||
|
const next = checked
|
||||||
|
? [...prev, id]
|
||||||
|
: prev.filter((x) => x !== id)
|
||||||
|
return {
|
||||||
|
...w,
|
||||||
|
iaConfig: {
|
||||||
|
...(w.iaConfig || ({} as any)),
|
||||||
|
repositoriosReferencia: next,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
onFilesChange={(files) =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
iaConfig: {
|
||||||
|
...(w.iaConfig || ({} as any)),
|
||||||
|
archivosAdjuntos: files,
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
/>
|
||||||
<div className="flex items-center justify-between">
|
<div className="flex items-center justify-between">
|
||||||
<div className="text-muted-foreground text-sm">
|
<div className="text-muted-foreground text-sm">
|
||||||
Opcional: se pueden adjuntar recursos IA más adelante.
|
Opcional: se pueden adjuntar recursos IA más adelante.
|
||||||
@@ -144,6 +174,7 @@ export function PasoDetallesPanel({
|
|||||||
<select
|
<select
|
||||||
id="clonFacultad"
|
id="clonFacultad"
|
||||||
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring h-10 w-full rounded-md border px-3 text-sm shadow-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring h-10 w-full rounded-md border px-3 text-sm shadow-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
||||||
|
aria-label="Facultad"
|
||||||
value={wizard.datosBasicos.facultadId}
|
value={wizard.datosBasicos.facultadId}
|
||||||
onChange={(e: React.ChangeEvent<HTMLSelectElement>) =>
|
onChange={(e: React.ChangeEvent<HTMLSelectElement>) =>
|
||||||
onChange((w) => ({
|
onChange((w) => ({
|
||||||
@@ -168,6 +199,7 @@ export function PasoDetallesPanel({
|
|||||||
<select
|
<select
|
||||||
id="clonCarrera"
|
id="clonCarrera"
|
||||||
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring h-10 w-full rounded-md border px-3 text-sm shadow-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring h-10 w-full rounded-md border px-3 text-sm shadow-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
||||||
|
aria-label="Carrera"
|
||||||
value={wizard.datosBasicos.carreraId}
|
value={wizard.datosBasicos.carreraId}
|
||||||
onChange={(e: React.ChangeEvent<HTMLSelectElement>) =>
|
onChange={(e: React.ChangeEvent<HTMLSelectElement>) =>
|
||||||
onChange((w) => ({
|
onChange((w) => ({
|
||||||
@@ -242,10 +274,10 @@ export function PasoDetallesPanel({
|
|||||||
wizard.subModoClonado === 'TRADICIONAL'
|
wizard.subModoClonado === 'TRADICIONAL'
|
||||||
) {
|
) {
|
||||||
return (
|
return (
|
||||||
<div className="grid gap-4">
|
<div className="flex flex-col gap-4">
|
||||||
<div>
|
<div className="flex flex-col gap-1">
|
||||||
<Label htmlFor="word">Word del plan (obligatorio)</Label>
|
<Label htmlFor="word">Word del plan (obligatorio)</Label>
|
||||||
<input
|
{/* <input
|
||||||
id="word"
|
id="word"
|
||||||
type="file"
|
type="file"
|
||||||
accept=".doc,.docx"
|
accept=".doc,.docx"
|
||||||
@@ -261,6 +293,21 @@ export function PasoDetallesPanel({
|
|||||||
},
|
},
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
/> */}
|
||||||
|
|
||||||
|
<FileDropzone
|
||||||
|
acceptedTypes=".doc,.docx"
|
||||||
|
maxFiles={1}
|
||||||
|
onFilesChange={(files) => {
|
||||||
|
const f = files[0] || null
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
clonTradicional: {
|
||||||
|
...(w.clonTradicional || ({} as any)),
|
||||||
|
archivoWordPlanId: f,
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
@@ -269,17 +316,32 @@ export function PasoDetallesPanel({
|
|||||||
id="mapa"
|
id="mapa"
|
||||||
type="file"
|
type="file"
|
||||||
accept=".xls,.xlsx"
|
accept=".xls,.xlsx"
|
||||||
|
title="Subir mapa curricular"
|
||||||
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring file:bg-secondary block w-full rounded-md border px-3 py-2 text-sm shadow-sm file:mr-4 file:rounded-md file:border-0 file:px-3 file:py-1.5 file:text-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring file:bg-secondary block w-full rounded-md border px-3 py-2 text-sm shadow-sm file:mr-4 file:rounded-md file:border-0 file:px-3 file:py-1.5 file:text-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
||||||
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
onChange((w) => ({
|
onChange((w) => {
|
||||||
...w,
|
const file = e.target.files?.[0] || null
|
||||||
clonTradicional: {
|
const next = file
|
||||||
...(w.clonTradicional || ({} as any)),
|
? {
|
||||||
archivoMapaExcelId: e.target.files?.[0]
|
id:
|
||||||
? `file_${e.target.files[0].name}`
|
typeof crypto !== 'undefined' && 'randomUUID' in crypto
|
||||||
: null,
|
? (crypto as any).randomUUID()
|
||||||
},
|
: `file-${Date.now()}-${Math.random()
|
||||||
}))
|
.toString(36)
|
||||||
|
.substr(2, 9)}`,
|
||||||
|
name: file.name,
|
||||||
|
size: formatFileSize(file.size),
|
||||||
|
type: file.name.split('.').pop() || 'file',
|
||||||
|
}
|
||||||
|
: null
|
||||||
|
return {
|
||||||
|
...w,
|
||||||
|
clonTradicional: {
|
||||||
|
...(w.clonTradicional || ({} as any)),
|
||||||
|
archivoMapaExcelId: next,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -289,17 +351,32 @@ export function PasoDetallesPanel({
|
|||||||
id="asignaturas"
|
id="asignaturas"
|
||||||
type="file"
|
type="file"
|
||||||
accept=".xls,.xlsx,.csv"
|
accept=".xls,.xlsx,.csv"
|
||||||
|
title="Subir listado de asignaturas"
|
||||||
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring file:bg-secondary block w-full rounded-md border px-3 py-2 text-sm shadow-sm file:mr-4 file:rounded-md file:border-0 file:px-3 file:py-1.5 file:text-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
className="bg-background text-foreground ring-offset-background focus-visible:ring-ring file:bg-secondary block w-full rounded-md border px-3 py-2 text-sm shadow-sm file:mr-4 file:rounded-md file:border-0 file:px-3 file:py-1.5 file:text-sm focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none"
|
||||||
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
onChange={(e: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
onChange((w) => ({
|
onChange((w) => {
|
||||||
...w,
|
const file = e.target.files?.[0] || null
|
||||||
clonTradicional: {
|
const next = file
|
||||||
...(w.clonTradicional || ({} as any)),
|
? {
|
||||||
archivoAsignaturasExcelId: e.target.files?.[0]
|
id:
|
||||||
? `file_${e.target.files[0].name}`
|
typeof crypto !== 'undefined' && 'randomUUID' in crypto
|
||||||
: null,
|
? (crypto as any).randomUUID()
|
||||||
},
|
: `file-${Date.now()}-${Math.random()
|
||||||
}))
|
.toString(36)
|
||||||
|
.substr(2, 9)}`,
|
||||||
|
name: file.name,
|
||||||
|
size: formatFileSize(file.size),
|
||||||
|
type: file.name.split('.').pop() || 'file',
|
||||||
|
}
|
||||||
|
: null
|
||||||
|
return {
|
||||||
|
...w,
|
||||||
|
clonTradicional: {
|
||||||
|
...(w.clonTradicional || ({} as any)),
|
||||||
|
archivoAsignaturasExcelId: next,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -321,3 +398,9 @@ export function PasoDetallesPanel({
|
|||||||
</Card>
|
</Card>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function formatFileSize(bytes: number): string {
|
||||||
|
if (bytes < 1024) return bytes + ' B'
|
||||||
|
if (bytes < 1024 * 1024) return (bytes / 1024).toFixed(1) + ' KB'
|
||||||
|
return (bytes / (1024 * 1024)).toFixed(1) + ' MB'
|
||||||
|
}
|
||||||
@@ -0,0 +1,210 @@
|
|||||||
|
import { FileText, FolderOpen, Upload } from 'lucide-react'
|
||||||
|
import { useMemo, useState } from 'react'
|
||||||
|
|
||||||
|
import BarraBusqueda from '../../BarraBusqueda'
|
||||||
|
|
||||||
|
import { FileDropzone } from './FileDropZone'
|
||||||
|
|
||||||
|
import { Checkbox } from '@/components/ui/checkbox'
|
||||||
|
import { Label } from '@/components/ui/label'
|
||||||
|
import {
|
||||||
|
Tabs,
|
||||||
|
TabsContent,
|
||||||
|
TabsList,
|
||||||
|
TabsTrigger,
|
||||||
|
TabsContents,
|
||||||
|
} from '@/components/ui/motion-tabs'
|
||||||
|
import { ARCHIVOS, REPOSITORIOS } from '@/features/planes/nuevo/catalogs'
|
||||||
|
|
||||||
|
const ReferenciasParaIA = ({
|
||||||
|
selectedArchivoIds = [],
|
||||||
|
selectedRepositorioIds = [],
|
||||||
|
onToggleArchivo,
|
||||||
|
onToggleRepositorio,
|
||||||
|
onFilesChange,
|
||||||
|
}: {
|
||||||
|
selectedArchivoIds?: Array<string>
|
||||||
|
selectedRepositorioIds?: Array<string>
|
||||||
|
onToggleArchivo?: (id: string, checked: boolean) => void
|
||||||
|
onToggleRepositorio?: (id: string, checked: boolean) => void
|
||||||
|
onFilesChange?: (
|
||||||
|
files: Array<{ id: string; name: string; size: string; type: string }>,
|
||||||
|
) => void
|
||||||
|
}) => {
|
||||||
|
const [busquedaArchivos, setBusquedaArchivos] = useState('')
|
||||||
|
const [busquedaRepositorios, setBusquedaRepositorios] = useState('')
|
||||||
|
|
||||||
|
const cleanText = (text: string) => {
|
||||||
|
return text
|
||||||
|
.normalize('NFD') // Descompone "á" en "a" + "´"
|
||||||
|
.replace(/[\u0300-\u036f]/g, '') // Elimina los símbolos diacríticos
|
||||||
|
.toLowerCase() // Convierte a minúsculas
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filtrado de archivos y de repositorios
|
||||||
|
const archivosFiltrados = useMemo(() => {
|
||||||
|
// Función helper para limpiar texto (quita acentos y hace minúsculas)
|
||||||
|
|
||||||
|
const term = cleanText(busquedaArchivos)
|
||||||
|
return ARCHIVOS.filter((archivo) =>
|
||||||
|
cleanText(archivo.nombre).includes(term),
|
||||||
|
)
|
||||||
|
}, [busquedaArchivos])
|
||||||
|
|
||||||
|
const repositoriosFiltrados = useMemo(() => {
|
||||||
|
const term = cleanText(busquedaRepositorios)
|
||||||
|
return REPOSITORIOS.filter((repositorio) =>
|
||||||
|
cleanText(repositorio.nombre).includes(term),
|
||||||
|
)
|
||||||
|
}, [busquedaRepositorios])
|
||||||
|
|
||||||
|
const tabs = [
|
||||||
|
{
|
||||||
|
name: 'Archivos existentes',
|
||||||
|
|
||||||
|
value: 'archivos-existentes',
|
||||||
|
|
||||||
|
icon: FileText,
|
||||||
|
|
||||||
|
content: (
|
||||||
|
<div className="flex flex-col">
|
||||||
|
<BarraBusqueda
|
||||||
|
value={busquedaArchivos}
|
||||||
|
onChange={setBusquedaArchivos}
|
||||||
|
placeholder="Buscar archivo existente..."
|
||||||
|
className="m-1 mb-1.5"
|
||||||
|
/>
|
||||||
|
<div className="flex h-72 flex-col gap-0.5 overflow-y-auto">
|
||||||
|
{archivosFiltrados.map((archivo) => (
|
||||||
|
<Label
|
||||||
|
key={archivo.id}
|
||||||
|
className="border-border hover:border-primary/30 hover:bg-accent/50 m-0.5 flex cursor-pointer items-center gap-3 rounded-lg border p-3 transition-colors has-aria-checked:border-blue-600 has-aria-checked:bg-blue-50 dark:has-aria-checked:border-blue-900 dark:has-aria-checked:bg-blue-950"
|
||||||
|
>
|
||||||
|
<Checkbox
|
||||||
|
checked={selectedArchivoIds.includes(archivo.id)}
|
||||||
|
onCheckedChange={(checked) =>
|
||||||
|
onToggleArchivo?.(archivo.id, !!checked)
|
||||||
|
}
|
||||||
|
className="peer border-primary ring-offset-background data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground focus-visible:ring-ring h-5 w-5 shrink-0 rounded-sm border focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none disabled:cursor-not-allowed disabled:opacity-50"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<FileText className="text-muted-foreground h-4 w-4" />
|
||||||
|
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<p className="text-foreground truncate text-sm font-medium">
|
||||||
|
{archivo.nombre}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p className="text-muted-foreground text-xs">
|
||||||
|
{archivo.tamaño}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</Label>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
name: 'Repositorios',
|
||||||
|
|
||||||
|
value: 'repositorios',
|
||||||
|
|
||||||
|
icon: FolderOpen,
|
||||||
|
|
||||||
|
content: (
|
||||||
|
<div className="flex flex-col">
|
||||||
|
<BarraBusqueda
|
||||||
|
value={busquedaRepositorios}
|
||||||
|
onChange={setBusquedaRepositorios}
|
||||||
|
placeholder="Buscar repositorio..."
|
||||||
|
className="m-1 mb-1.5"
|
||||||
|
/>
|
||||||
|
<div className="flex h-72 flex-col gap-0.5 overflow-y-auto">
|
||||||
|
{repositoriosFiltrados.map((repositorio) => (
|
||||||
|
<Label
|
||||||
|
key={repositorio.id}
|
||||||
|
className="border-border hover:border-primary/30 hover:bg-accent/50 m-0.5 flex cursor-pointer items-center gap-3 rounded-lg border p-3 transition-colors has-aria-checked:border-blue-600 has-aria-checked:bg-blue-50 dark:has-aria-checked:border-blue-900 dark:has-aria-checked:bg-blue-950"
|
||||||
|
>
|
||||||
|
<Checkbox
|
||||||
|
checked={selectedRepositorioIds.includes(repositorio.id)}
|
||||||
|
onCheckedChange={(checked) =>
|
||||||
|
onToggleRepositorio?.(repositorio.id, !!checked)
|
||||||
|
}
|
||||||
|
className="peer border-primary ring-offset-background data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground focus-visible:ring-ring h-5 w-5 shrink-0 rounded-sm border focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none disabled:cursor-not-allowed disabled:opacity-50"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<FolderOpen className="text-muted-foreground h-4 w-4" />
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<p className="text-foreground text-sm font-medium">
|
||||||
|
{repositorio.nombre}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p className="text-muted-foreground text-xs">
|
||||||
|
{repositorio.descripcion} · {repositorio.cantidadArchivos}{' '}
|
||||||
|
archivos
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</Label>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
name: 'Subir archivos',
|
||||||
|
|
||||||
|
value: 'subir-archivos',
|
||||||
|
|
||||||
|
icon: Upload,
|
||||||
|
|
||||||
|
content: (
|
||||||
|
<div>
|
||||||
|
<FileDropzone
|
||||||
|
onFilesChange={onFilesChange}
|
||||||
|
title="Sube archivos de referencia"
|
||||||
|
description="Documentos que serán usados como contexto para la generación"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex w-full flex-col gap-1">
|
||||||
|
<Label>Referencias para la IA</Label>
|
||||||
|
|
||||||
|
<Tabs defaultValue="archivos-existentes" className="gap-4">
|
||||||
|
<TabsList className="w-full">
|
||||||
|
{tabs.map(({ icon: Icon, name, value }) => (
|
||||||
|
<TabsTrigger
|
||||||
|
key={value}
|
||||||
|
value={value}
|
||||||
|
className="flex items-center gap-1 px-2.5 sm:px-3"
|
||||||
|
>
|
||||||
|
<Icon />
|
||||||
|
|
||||||
|
<span className="hidden sm:inline">{name}</span>
|
||||||
|
</TabsTrigger>
|
||||||
|
))}
|
||||||
|
</TabsList>
|
||||||
|
|
||||||
|
<TabsContents className="bg-background mx-1 -mt-2 mb-1 h-full rounded-sm">
|
||||||
|
{tabs.map((tab) => (
|
||||||
|
<TabsContent
|
||||||
|
key={tab.value}
|
||||||
|
value={tab.value}
|
||||||
|
className="animate-in fade-in duration-300 ease-out"
|
||||||
|
>
|
||||||
|
{tab.content}
|
||||||
|
</TabsContent>
|
||||||
|
))}
|
||||||
|
</TabsContents>
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ReferenciasParaIA
|
||||||
@@ -4,7 +4,7 @@ import type {
|
|||||||
NewPlanWizardState,
|
NewPlanWizardState,
|
||||||
ModoCreacion,
|
ModoCreacion,
|
||||||
SubModoClonado,
|
SubModoClonado,
|
||||||
} from '@/features/planes/new/types'
|
} from '@/features/planes/nuevo/types'
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Card,
|
Card,
|
||||||
@@ -23,6 +23,19 @@ export function PasoModoCardGroup({
|
|||||||
}) {
|
}) {
|
||||||
const isSelected = (m: ModoCreacion) => wizard.modoCreacion === m
|
const isSelected = (m: ModoCreacion) => wizard.modoCreacion === m
|
||||||
const isSubSelected = (s: SubModoClonado) => wizard.subModoClonado === s
|
const isSubSelected = (s: SubModoClonado) => wizard.subModoClonado === s
|
||||||
|
const handleKeyActivate = (e: React.KeyboardEvent, cb: () => void) => {
|
||||||
|
const key = e.key
|
||||||
|
if (
|
||||||
|
key === 'Enter' ||
|
||||||
|
key === ' ' ||
|
||||||
|
key === 'Spacebar' ||
|
||||||
|
key === 'Space'
|
||||||
|
) {
|
||||||
|
e.preventDefault()
|
||||||
|
e.stopPropagation()
|
||||||
|
cb()
|
||||||
|
}
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<div className="grid gap-4 sm:grid-cols-3">
|
<div className="grid gap-4 sm:grid-cols-3">
|
||||||
<Card
|
<Card
|
||||||
@@ -34,6 +47,15 @@ export function PasoModoCardGroup({
|
|||||||
subModoClonado: undefined,
|
subModoClonado: undefined,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
onKeyDown={(e: React.KeyboardEvent) =>
|
||||||
|
handleKeyActivate(e, () =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
modoCreacion: 'MANUAL',
|
||||||
|
subModoClonado: undefined,
|
||||||
|
})),
|
||||||
|
)
|
||||||
|
}
|
||||||
role="button"
|
role="button"
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
>
|
>
|
||||||
@@ -54,6 +76,15 @@ export function PasoModoCardGroup({
|
|||||||
subModoClonado: undefined,
|
subModoClonado: undefined,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
onKeyDown={(e: React.KeyboardEvent) =>
|
||||||
|
handleKeyActivate(e, () =>
|
||||||
|
onChange((w) => ({
|
||||||
|
...w,
|
||||||
|
modoCreacion: 'IA',
|
||||||
|
subModoClonado: undefined,
|
||||||
|
})),
|
||||||
|
)
|
||||||
|
}
|
||||||
role="button"
|
role="button"
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
>
|
>
|
||||||
@@ -70,6 +101,11 @@ export function PasoModoCardGroup({
|
|||||||
<Card
|
<Card
|
||||||
className={isSelected('CLONADO') ? 'ring-ring ring-2' : ''}
|
className={isSelected('CLONADO') ? 'ring-ring ring-2' : ''}
|
||||||
onClick={() => onChange((w) => ({ ...w, modoCreacion: 'CLONADO' }))}
|
onClick={() => onChange((w) => ({ ...w, modoCreacion: 'CLONADO' }))}
|
||||||
|
onKeyDown={(e: React.KeyboardEvent) =>
|
||||||
|
handleKeyActivate(e, () =>
|
||||||
|
onChange((w) => ({ ...w, modoCreacion: 'CLONADO' })),
|
||||||
|
)
|
||||||
|
}
|
||||||
role="button"
|
role="button"
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
>
|
>
|
||||||
@@ -88,6 +124,11 @@ export function PasoModoCardGroup({
|
|||||||
e.stopPropagation()
|
e.stopPropagation()
|
||||||
onChange((w) => ({ ...w, subModoClonado: 'INTERNO' }))
|
onChange((w) => ({ ...w, subModoClonado: 'INTERNO' }))
|
||||||
}}
|
}}
|
||||||
|
onKeyDown={(e: React.KeyboardEvent) =>
|
||||||
|
handleKeyActivate(e, () =>
|
||||||
|
onChange((w) => ({ ...w, subModoClonado: 'INTERNO' })),
|
||||||
|
)
|
||||||
|
}
|
||||||
className={`hover:border-primary/50 hover:bg-accent flex cursor-pointer flex-row items-center justify-center gap-2 rounded-lg border p-4 text-center transition-all sm:flex-col ${
|
className={`hover:border-primary/50 hover:bg-accent flex cursor-pointer flex-row items-center justify-center gap-2 rounded-lg border p-4 text-center transition-all sm:flex-col ${
|
||||||
isSubSelected('INTERNO')
|
isSubSelected('INTERNO')
|
||||||
? 'border-primary bg-primary/5 ring-primary text-primary ring-1'
|
? 'border-primary bg-primary/5 ring-primary text-primary ring-1'
|
||||||
@@ -105,6 +146,11 @@ export function PasoModoCardGroup({
|
|||||||
e.stopPropagation()
|
e.stopPropagation()
|
||||||
onChange((w) => ({ ...w, subModoClonado: 'TRADICIONAL' }))
|
onChange((w) => ({ ...w, subModoClonado: 'TRADICIONAL' }))
|
||||||
}}
|
}}
|
||||||
|
onKeyDown={(e: React.KeyboardEvent) =>
|
||||||
|
handleKeyActivate(e, () =>
|
||||||
|
onChange((w) => ({ ...w, subModoClonado: 'TRADICIONAL' })),
|
||||||
|
)
|
||||||
|
}
|
||||||
className={`hover:border-primary/50 hover:bg-accent flex cursor-pointer flex-row items-center justify-center gap-2 rounded-lg border p-4 text-center transition-all sm:flex-col ${
|
className={`hover:border-primary/50 hover:bg-accent flex cursor-pointer flex-row items-center justify-center gap-2 rounded-lg border p-4 text-center transition-all sm:flex-col ${
|
||||||
isSubSelected('TRADICIONAL')
|
isSubSelected('TRADICIONAL')
|
||||||
? 'border-primary bg-primary/5 ring-primary text-primary ring-1'
|
? 'border-primary bg-primary/5 ring-primary text-primary ring-1'
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { NewPlanWizardState } from '@/features/planes/new/types'
|
import type { NewPlanWizardState } from '@/features/planes/nuevo/types'
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Card,
|
Card,
|
||||||
@@ -7,10 +7,15 @@ import {
|
|||||||
CardHeader,
|
CardHeader,
|
||||||
CardTitle,
|
CardTitle,
|
||||||
} from '@/components/ui/card'
|
} from '@/components/ui/card'
|
||||||
|
import {
|
||||||
|
PLANTILLAS_ANEXO_1,
|
||||||
|
PLANTILLAS_ANEXO_2,
|
||||||
|
PLANES_EXISTENTES,
|
||||||
|
ARCHIVOS,
|
||||||
|
REPOSITORIOS,
|
||||||
|
} from '@/features/planes/nuevo/catalogs'
|
||||||
|
|
||||||
export function PasoResumenCard({ wizard }: { wizard: NewPlanWizardState }) {
|
export function PasoResumenCard({ wizard }: { wizard: NewPlanWizardState }) {
|
||||||
const modo = wizard.modoCreacion
|
|
||||||
const sub = wizard.subModoClonado
|
|
||||||
return (
|
return (
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeader>
|
<CardHeader>
|
||||||
@@ -21,53 +26,210 @@ export function PasoResumenCard({ wizard }: { wizard: NewPlanWizardState }) {
|
|||||||
</CardHeader>
|
</CardHeader>
|
||||||
<CardContent>
|
<CardContent>
|
||||||
<div className="grid gap-2 text-sm">
|
<div className="grid gap-2 text-sm">
|
||||||
<div>
|
{(() => {
|
||||||
<span className="text-muted-foreground">Nombre: </span>
|
// Precompute common derived values to avoid unnecessary optional chaining warnings
|
||||||
<span className="font-medium">
|
const archivosRef = wizard.iaConfig?.archivosReferencia ?? []
|
||||||
{wizard.datosBasicos.nombrePlan || '—'}
|
const repositoriosRef =
|
||||||
</span>
|
wizard.iaConfig?.repositoriosReferencia ?? []
|
||||||
</div>
|
const adjuntos = wizard.iaConfig?.archivosAdjuntos ?? []
|
||||||
<div>
|
const plantillaPlan = PLANTILLAS_ANEXO_1.find(
|
||||||
<span className="text-muted-foreground">Facultad/Carrera: </span>
|
(x) => x.id === wizard.datosBasicos.plantillaPlanId,
|
||||||
<span className="font-medium">
|
)
|
||||||
{wizard.datosBasicos.facultadId || '—'} /{' '}
|
const plantillaMapa = PLANTILLAS_ANEXO_2.find(
|
||||||
{wizard.datosBasicos.carreraId || '—'}
|
(x) => x.id === wizard.datosBasicos.plantillaMapaId,
|
||||||
</span>
|
)
|
||||||
</div>
|
const contenido = (
|
||||||
<div>
|
<>
|
||||||
<span className="text-muted-foreground">Nivel: </span>
|
<div>
|
||||||
<span className="font-medium">
|
<span className="text-muted-foreground">Nombre: </span>
|
||||||
{wizard.datosBasicos.nivel || '—'}
|
<span className="font-medium">
|
||||||
</span>
|
{wizard.datosBasicos.nombrePlan || '—'}
|
||||||
</div>
|
</span>
|
||||||
<div>
|
</div>
|
||||||
<span className="text-muted-foreground">Ciclos: </span>
|
<div>
|
||||||
<span className="font-medium">
|
<span className="text-muted-foreground">
|
||||||
{wizard.datosBasicos.numCiclos} ({wizard.datosBasicos.tipoCiclo})
|
Facultad/Carrera:{' '}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
<span className="font-medium">
|
||||||
<div className="mt-2">
|
{wizard.datosBasicos.facultadId || '—'} /{' '}
|
||||||
<span className="text-muted-foreground">Modo: </span>
|
{wizard.datosBasicos.carreraId || '—'}
|
||||||
<span className="font-medium">
|
</span>
|
||||||
{modo === 'MANUAL' && 'Manual'}
|
</div>
|
||||||
{modo === 'IA' && 'Generado con IA'}
|
<div>
|
||||||
{modo === 'CLONADO' &&
|
<span className="text-muted-foreground">Nivel: </span>
|
||||||
sub === 'INTERNO' &&
|
<span className="font-medium">
|
||||||
'Clonado desde plan del sistema'}
|
{wizard.datosBasicos.nivel || '—'}
|
||||||
{modo === 'CLONADO' &&
|
</span>
|
||||||
sub === 'TRADICIONAL' &&
|
</div>
|
||||||
'Importado desde documentos tradicionales'}
|
<div>
|
||||||
</span>
|
<span className="text-muted-foreground">Ciclos: </span>
|
||||||
</div>
|
<span className="font-medium">
|
||||||
{wizard.resumen.previewPlan && (
|
{wizard.datosBasicos.numCiclos} (
|
||||||
<div className="bg-muted mt-2 rounded-md p-3">
|
{wizard.datosBasicos.tipoCiclo})
|
||||||
<div className="font-medium">Preview IA</div>
|
</span>
|
||||||
<div className="text-muted-foreground">
|
</div>
|
||||||
Asignaturas aprox.:{' '}
|
<div className="mt-2">
|
||||||
{wizard.resumen.previewPlan.numAsignaturasAprox}
|
<span className="text-muted-foreground">
|
||||||
</div>
|
Plantilla plan:{' '}
|
||||||
</div>
|
</span>
|
||||||
)}
|
<span className="font-medium">
|
||||||
|
{(plantillaPlan?.name ||
|
||||||
|
wizard.datosBasicos.plantillaPlanId ||
|
||||||
|
'—') +
|
||||||
|
' · ' +
|
||||||
|
(wizard.datosBasicos.plantillaPlanVersion || '—')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span className="text-muted-foreground">
|
||||||
|
Mapa curricular:{' '}
|
||||||
|
</span>
|
||||||
|
<span className="font-medium">
|
||||||
|
{(plantillaMapa?.name ||
|
||||||
|
wizard.datosBasicos.plantillaMapaId ||
|
||||||
|
'—') +
|
||||||
|
' · ' +
|
||||||
|
(wizard.datosBasicos.plantillaMapaVersion || '—')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="mt-2">
|
||||||
|
<span className="text-muted-foreground">Modo: </span>
|
||||||
|
<span className="font-medium">
|
||||||
|
{wizard.modoCreacion === 'MANUAL' && 'Manual'}
|
||||||
|
{wizard.modoCreacion === 'IA' && 'Generado con IA'}
|
||||||
|
{wizard.modoCreacion === 'CLONADO' &&
|
||||||
|
wizard.subModoClonado === 'INTERNO' &&
|
||||||
|
'Clonado desde plan del sistema'}
|
||||||
|
{wizard.modoCreacion === 'CLONADO' &&
|
||||||
|
wizard.subModoClonado === 'TRADICIONAL' &&
|
||||||
|
'Importado desde documentos tradicionales'}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{wizard.modoCreacion === 'CLONADO' &&
|
||||||
|
wizard.subModoClonado === 'INTERNO' && (
|
||||||
|
<div className="mt-2">
|
||||||
|
<span className="text-muted-foreground">
|
||||||
|
Plan origen:{' '}
|
||||||
|
</span>
|
||||||
|
<span className="font-medium">
|
||||||
|
{(() => {
|
||||||
|
const p = PLANES_EXISTENTES.find(
|
||||||
|
(x) => x.id === wizard.clonInterno?.planOrigenId,
|
||||||
|
)
|
||||||
|
return (
|
||||||
|
p?.nombre || wizard.clonInterno?.planOrigenId || '—'
|
||||||
|
)
|
||||||
|
})()}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{wizard.modoCreacion === 'CLONADO' &&
|
||||||
|
wizard.subModoClonado === 'TRADICIONAL' && (
|
||||||
|
<div className="mt-2">
|
||||||
|
<div className="font-medium">Documentos adjuntos</div>
|
||||||
|
<ul className="text-muted-foreground list-disc pl-5 text-xs">
|
||||||
|
<li>
|
||||||
|
<span className="text-foreground">
|
||||||
|
Word del plan:
|
||||||
|
</span>{' '}
|
||||||
|
{wizard.clonTradicional?.archivoWordPlanId?.name ||
|
||||||
|
'—'}
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<span className="text-foreground">
|
||||||
|
Mapa curricular:
|
||||||
|
</span>{' '}
|
||||||
|
{wizard.clonTradicional?.archivoMapaExcelId?.name ||
|
||||||
|
'—'}
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<span className="text-foreground">Asignaturas:</span>{' '}
|
||||||
|
{wizard.clonTradicional?.archivoAsignaturasExcelId
|
||||||
|
?.name || '—'}
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{wizard.modoCreacion === 'IA' && (
|
||||||
|
<div className="bg-muted/50 mt-2 rounded-md p-3">
|
||||||
|
<div>
|
||||||
|
<span className="text-muted-foreground">Enfoque: </span>
|
||||||
|
<span className="font-medium">
|
||||||
|
{wizard.iaConfig?.descripcionEnfoque || '—'}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span className="text-muted-foreground">Notas: </span>
|
||||||
|
<span className="font-medium">
|
||||||
|
{wizard.iaConfig?.notasAdicionales || '—'}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{archivosRef.length > 0 && (
|
||||||
|
<div className="mt-2">
|
||||||
|
<div className="font-medium">Archivos existentes</div>
|
||||||
|
<ul className="text-muted-foreground list-disc pl-5 text-xs">
|
||||||
|
{archivosRef.map((id) => {
|
||||||
|
const a = ARCHIVOS.find((x) => x.id === id)
|
||||||
|
return (
|
||||||
|
<li key={id}>
|
||||||
|
<span className="text-foreground">
|
||||||
|
{a?.nombre || id}
|
||||||
|
</span>{' '}
|
||||||
|
{a?.tamaño ? <span>· {a.tamaño}</span> : null}
|
||||||
|
</li>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{repositoriosRef.length > 0 && (
|
||||||
|
<div className="mt-2">
|
||||||
|
<div className="font-medium">Repositorios</div>
|
||||||
|
<ul className="text-muted-foreground list-disc pl-5 text-xs">
|
||||||
|
{repositoriosRef.map((id) => {
|
||||||
|
const r = REPOSITORIOS.find((x) => x.id === id)
|
||||||
|
return (
|
||||||
|
<li key={id}>
|
||||||
|
<span className="text-foreground">
|
||||||
|
{r?.nombre || id}
|
||||||
|
</span>{' '}
|
||||||
|
{r?.cantidadArchivos ? (
|
||||||
|
<span>· {r.cantidadArchivos} archivos</span>
|
||||||
|
) : null}
|
||||||
|
</li>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{adjuntos.length > 0 && (
|
||||||
|
<div className="mt-2">
|
||||||
|
<div className="font-medium">Adjuntos</div>
|
||||||
|
<ul className="text-muted-foreground list-disc pl-5 text-xs">
|
||||||
|
{adjuntos.map((f) => (
|
||||||
|
<li key={f.id}>
|
||||||
|
<span className="text-foreground">{f.name}</span>{' '}
|
||||||
|
<span>· {f.size}</span>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{wizard.resumen.previewPlan && (
|
||||||
|
<div className="bg-muted mt-2 rounded-md p-3">
|
||||||
|
<div className="font-medium">Preview IA</div>
|
||||||
|
<div className="text-muted-foreground">
|
||||||
|
Asignaturas aprox.:{' '}
|
||||||
|
{wizard.resumen.previewPlan.numAsignaturasAprox}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
return contenido
|
||||||
|
})()}
|
||||||
</div>
|
</div>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
|
|||||||
32
src/components/shadcn-studio/checkbox/checkbox-13.tsx
Normal file
32
src/components/shadcn-studio/checkbox/checkbox-13.tsx
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import { Checkbox } from '@/components/ui/checkbox'
|
||||||
|
import { Label } from '@/components/ui/label'
|
||||||
|
|
||||||
|
const CheckboxCardDemo = () => {
|
||||||
|
return (
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label className="border-border hover:border-primary/30 hover:bg-accent/50 flex cursor-pointer items-center items-start gap-2 gap-3 rounded-lg border p-3 transition-colors has-[[aria-checked=true]]:border-blue-600 has-[[aria-checked=true]]:bg-blue-50 dark:has-[[aria-checked=true]]:border-blue-900 dark:has-[[aria-checked=true]]:bg-blue-950">
|
||||||
|
<Checkbox
|
||||||
|
defaultChecked
|
||||||
|
className="peer border-primary ring-offset-background data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground focus-visible:ring-ring h-5 w-5 shrink-0 rounded-sm border focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none disabled:cursor-not-allowed disabled:opacity-50"
|
||||||
|
/>
|
||||||
|
<div className="grid gap-1.5 font-normal">
|
||||||
|
<p className="text-sm leading-none font-medium">Auto Start</p>
|
||||||
|
<p className="text-muted-foreground text-sm">
|
||||||
|
Starting with your OS.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</Label>
|
||||||
|
<Label className="hover:bg-accent/50 flex items-start gap-2 rounded-lg border p-3 has-[[aria-checked=true]]:border-blue-600 has-[[aria-checked=true]]:bg-blue-50 dark:has-[[aria-checked=true]]:border-blue-900 dark:has-[[aria-checked=true]]:bg-blue-950">
|
||||||
|
<Checkbox className="peer border-primary ring-offset-background data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground focus-visible:ring-ring h-5 w-5 shrink-0 rounded-sm border focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none disabled:cursor-not-allowed disabled:opacity-50" />
|
||||||
|
<div className="grid gap-1.5 font-normal">
|
||||||
|
<p className="text-sm leading-none font-medium">Auto update</p>
|
||||||
|
<p className="text-muted-foreground text-sm">
|
||||||
|
Download and install new version
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default CheckboxCardDemo
|
||||||
76
src/components/shadcn-studio/tabs/tabs-03.tsx
Normal file
76
src/components/shadcn-studio/tabs/tabs-03.tsx
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
import { BookIcon, GiftIcon, HeartIcon } from 'lucide-react'
|
||||||
|
|
||||||
|
import CheckboxCardDemo from '../checkbox/checkbox-13'
|
||||||
|
|
||||||
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'
|
||||||
|
|
||||||
|
const tabs = [
|
||||||
|
{
|
||||||
|
name: 'Explore',
|
||||||
|
value: 'explore',
|
||||||
|
icon: BookIcon,
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
<CheckboxCardDemo />
|
||||||
|
</>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Favorites',
|
||||||
|
value: 'favorites',
|
||||||
|
icon: HeartIcon,
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
All your{' '}
|
||||||
|
<span className="text-foreground font-semibold">favorites</span> are
|
||||||
|
saved here. Revisit articles, collections, and moments you love, any
|
||||||
|
time you want a little inspiration.
|
||||||
|
</>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Surprise',
|
||||||
|
value: 'surprise',
|
||||||
|
icon: GiftIcon,
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
<span className="text-foreground font-semibold">Surprise!</span>{' '}
|
||||||
|
Here's something unexpected—a fun fact, a quirky tip, or a daily
|
||||||
|
challenge. Come back for a new surprise every day!
|
||||||
|
</>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
const TabsWithIconDemo = () => {
|
||||||
|
return (
|
||||||
|
<div className="w-full">
|
||||||
|
<Tabs defaultValue="explore" className="gap-4">
|
||||||
|
<TabsList className="w-full">
|
||||||
|
{tabs.map(({ icon: Icon, name, value }) => (
|
||||||
|
<TabsTrigger
|
||||||
|
key={value}
|
||||||
|
value={value}
|
||||||
|
className="flex items-center gap-1 px-2.5 sm:px-3"
|
||||||
|
>
|
||||||
|
<Icon />
|
||||||
|
{name}
|
||||||
|
</TabsTrigger>
|
||||||
|
))}
|
||||||
|
</TabsList>
|
||||||
|
|
||||||
|
{tabs.map((tab) => (
|
||||||
|
<TabsContent
|
||||||
|
key={tab.value}
|
||||||
|
value={tab.value}
|
||||||
|
className="animate-in fade-in duration-300 ease-out"
|
||||||
|
>
|
||||||
|
<p className="text-muted-foreground text-sm">{tab.content}</p>
|
||||||
|
</TabsContent>
|
||||||
|
))}
|
||||||
|
</Tabs>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default TabsWithIconDemo
|
||||||
72
src/components/shadcn-studio/tabs/tabs-27.tsx
Normal file
72
src/components/shadcn-studio/tabs/tabs-27.tsx
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
import { Tabs, TabsContent, TabsContents, TabsList, TabsTrigger } from '@/components/ui/motion-tabs'
|
||||||
|
|
||||||
|
const tabs = [
|
||||||
|
{
|
||||||
|
name: 'Explore',
|
||||||
|
value: 'explore',
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
Discover <span className='text-foreground font-semibold'>fresh ideas</span>, trending topics, and hidden gems
|
||||||
|
curated just for you. Start exploring and let your curiosity lead the way!
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Favorites',
|
||||||
|
value: 'favorites',
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
All your <span className='text-foreground font-semibold'>favorites</span> are saved here. Revisit articles,
|
||||||
|
collections, and moments you love, any time you want a little inspiration.
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Surprise Me',
|
||||||
|
value: 'surprise',
|
||||||
|
content: (
|
||||||
|
<>
|
||||||
|
<span className='text-foreground font-semibold'>Surprise!</span> Here's something unexpected—a fun fact, a
|
||||||
|
quirky tip, or a daily challenge. Come back for a new surprise every day!
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
const AnimatedTabsDemo = () => {
|
||||||
|
return (
|
||||||
|
<div className='w-full max-w-md'>
|
||||||
|
<Tabs defaultValue='explore' className='gap-4'>
|
||||||
|
<TabsList>
|
||||||
|
{tabs.map(tab => (
|
||||||
|
<TabsTrigger key={tab.value} value={tab.value}>
|
||||||
|
{tab.name}
|
||||||
|
</TabsTrigger>
|
||||||
|
))}
|
||||||
|
</TabsList>
|
||||||
|
|
||||||
|
<TabsContents className='bg-background mx-1 -mt-2 mb-1 h-full rounded-sm'>
|
||||||
|
{tabs.map(tab => (
|
||||||
|
<TabsContent key={tab.value} value={tab.value}>
|
||||||
|
<p className='text-muted-foreground text-sm'>{tab.content}</p>
|
||||||
|
</TabsContent>
|
||||||
|
))}
|
||||||
|
</TabsContents>
|
||||||
|
</Tabs>
|
||||||
|
|
||||||
|
<p className='text-muted-foreground mt-4 text-center text-xs'>
|
||||||
|
Inspired by{' '}
|
||||||
|
<a
|
||||||
|
className='hover:text-foreground underline'
|
||||||
|
href='https://animate-ui.com/docs/components/tabs'
|
||||||
|
target='_blank'
|
||||||
|
rel='noopener noreferrer'
|
||||||
|
>
|
||||||
|
Animate UI
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default AnimatedTabsDemo
|
||||||
30
src/components/ui/checkbox.tsx
Normal file
30
src/components/ui/checkbox.tsx
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import * as CheckboxPrimitive from "@radix-ui/react-checkbox"
|
||||||
|
import { CheckIcon } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
function Checkbox({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CheckboxPrimitive.Root>) {
|
||||||
|
return (
|
||||||
|
<CheckboxPrimitive.Root
|
||||||
|
data-slot="checkbox"
|
||||||
|
className={cn(
|
||||||
|
"peer border-input dark:bg-input/30 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground dark:data-[state=checked]:bg-primary data-[state=checked]:border-primary focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive size-4 shrink-0 rounded-[4px] border shadow-xs transition-shadow outline-none focus-visible:ring-[3px] disabled:cursor-not-allowed disabled:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
<CheckboxPrimitive.Indicator
|
||||||
|
data-slot="checkbox-indicator"
|
||||||
|
className="grid place-content-center text-current transition-none"
|
||||||
|
>
|
||||||
|
<CheckIcon className="size-3.5" />
|
||||||
|
</CheckboxPrimitive.Indicator>
|
||||||
|
</CheckboxPrimitive.Root>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Checkbox }
|
||||||
549
src/components/ui/motion-highlight.tsx
Normal file
549
src/components/ui/motion-highlight.tsx
Normal file
@@ -0,0 +1,549 @@
|
|||||||
|
import * as React from 'react'
|
||||||
|
|
||||||
|
import type { Transition } from 'motion/react'
|
||||||
|
import { AnimatePresence, motion } from 'motion/react'
|
||||||
|
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
|
||||||
|
type MotionHighlightMode = 'children' | 'parent'
|
||||||
|
|
||||||
|
type Bounds = {
|
||||||
|
top: number
|
||||||
|
left: number
|
||||||
|
width: number
|
||||||
|
height: number
|
||||||
|
}
|
||||||
|
|
||||||
|
type MotionHighlightContextType<T extends string> = {
|
||||||
|
mode: MotionHighlightMode
|
||||||
|
activeValue: T | null
|
||||||
|
setActiveValue: (value: T | null) => void
|
||||||
|
setBounds: (bounds: DOMRect) => void
|
||||||
|
clearBounds: () => void
|
||||||
|
id: string
|
||||||
|
hover: boolean
|
||||||
|
className?: string
|
||||||
|
activeClassName?: string
|
||||||
|
setActiveClassName: (className: string) => void
|
||||||
|
transition?: Transition
|
||||||
|
disabled?: boolean
|
||||||
|
enabled?: boolean
|
||||||
|
exitDelay?: number
|
||||||
|
forceUpdateBounds?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const MotionHighlightContext = React.createContext<
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
MotionHighlightContextType<any> | undefined
|
||||||
|
>(undefined)
|
||||||
|
|
||||||
|
function useMotionHighlight<T extends string>(): MotionHighlightContextType<T> {
|
||||||
|
const context = React.useContext(MotionHighlightContext)
|
||||||
|
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('useMotionHighlight must be used within a MotionHighlightProvider')
|
||||||
|
}
|
||||||
|
|
||||||
|
return context as unknown as MotionHighlightContextType<T>
|
||||||
|
}
|
||||||
|
|
||||||
|
type BaseMotionHighlightProps<T extends string> = {
|
||||||
|
mode?: MotionHighlightMode
|
||||||
|
value?: T | null
|
||||||
|
defaultValue?: T | null
|
||||||
|
onValueChange?: (value: T | null) => void
|
||||||
|
className?: string
|
||||||
|
transition?: Transition
|
||||||
|
hover?: boolean
|
||||||
|
disabled?: boolean
|
||||||
|
enabled?: boolean
|
||||||
|
exitDelay?: number
|
||||||
|
}
|
||||||
|
|
||||||
|
type ParentModeMotionHighlightProps = {
|
||||||
|
boundsOffset?: Partial<Bounds>
|
||||||
|
containerClassName?: string
|
||||||
|
forceUpdateBounds?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
type ControlledParentModeMotionHighlightProps<T extends string> = BaseMotionHighlightProps<T> &
|
||||||
|
ParentModeMotionHighlightProps & {
|
||||||
|
mode: 'parent'
|
||||||
|
controlledItems: true
|
||||||
|
children: React.ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
type ControlledChildrenModeMotionHighlightProps<T extends string> = BaseMotionHighlightProps<T> & {
|
||||||
|
mode?: 'children' | undefined
|
||||||
|
controlledItems: true
|
||||||
|
children: React.ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
type UncontrolledParentModeMotionHighlightProps<T extends string> = BaseMotionHighlightProps<T> &
|
||||||
|
ParentModeMotionHighlightProps & {
|
||||||
|
mode: 'parent'
|
||||||
|
controlledItems?: false
|
||||||
|
itemsClassName?: string
|
||||||
|
children: React.ReactElement | React.ReactElement[]
|
||||||
|
}
|
||||||
|
|
||||||
|
type UncontrolledChildrenModeMotionHighlightProps<T extends string> = BaseMotionHighlightProps<T> & {
|
||||||
|
mode?: 'children'
|
||||||
|
controlledItems?: false
|
||||||
|
itemsClassName?: string
|
||||||
|
children: React.ReactElement | React.ReactElement[]
|
||||||
|
}
|
||||||
|
|
||||||
|
type MotionHighlightProps<T extends string> = React.ComponentProps<'div'> &
|
||||||
|
(
|
||||||
|
| ControlledParentModeMotionHighlightProps<T>
|
||||||
|
| ControlledChildrenModeMotionHighlightProps<T>
|
||||||
|
| UncontrolledParentModeMotionHighlightProps<T>
|
||||||
|
| UncontrolledChildrenModeMotionHighlightProps<T>
|
||||||
|
)
|
||||||
|
|
||||||
|
function MotionHighlight<T extends string>({ ref, ...props }: MotionHighlightProps<T>) {
|
||||||
|
const {
|
||||||
|
children,
|
||||||
|
value,
|
||||||
|
defaultValue,
|
||||||
|
onValueChange,
|
||||||
|
className,
|
||||||
|
transition = { type: 'spring', stiffness: 350, damping: 35 },
|
||||||
|
hover = false,
|
||||||
|
enabled = true,
|
||||||
|
controlledItems,
|
||||||
|
disabled = false,
|
||||||
|
exitDelay = 0.2,
|
||||||
|
mode = 'children'
|
||||||
|
} = props
|
||||||
|
|
||||||
|
const localRef = React.useRef<HTMLDivElement>(null)
|
||||||
|
|
||||||
|
React.useImperativeHandle(ref, () => localRef.current as HTMLDivElement)
|
||||||
|
|
||||||
|
const [activeValue, setActiveValue] = React.useState<T | null>(value ?? defaultValue ?? null)
|
||||||
|
const [boundsState, setBoundsState] = React.useState<Bounds | null>(null)
|
||||||
|
const [activeClassNameState, setActiveClassNameState] = React.useState<string>('')
|
||||||
|
|
||||||
|
const safeSetActiveValue = React.useCallback(
|
||||||
|
(id: T | null) => {
|
||||||
|
setActiveValue(prev => (prev === id ? prev : id))
|
||||||
|
if (id !== activeValue) onValueChange?.(id as T)
|
||||||
|
},
|
||||||
|
[activeValue, onValueChange]
|
||||||
|
)
|
||||||
|
|
||||||
|
const safeSetBounds = React.useCallback(
|
||||||
|
(bounds: DOMRect) => {
|
||||||
|
if (!localRef.current) return
|
||||||
|
|
||||||
|
const boundsOffset = (props as ParentModeMotionHighlightProps)?.boundsOffset ?? {
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: 0,
|
||||||
|
height: 0
|
||||||
|
}
|
||||||
|
|
||||||
|
const containerRect = localRef.current.getBoundingClientRect()
|
||||||
|
|
||||||
|
const newBounds: Bounds = {
|
||||||
|
top: bounds.top - containerRect.top + (boundsOffset.top ?? 0),
|
||||||
|
left: bounds.left - containerRect.left + (boundsOffset.left ?? 0),
|
||||||
|
width: bounds.width + (boundsOffset.width ?? 0),
|
||||||
|
height: bounds.height + (boundsOffset.height ?? 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
setBoundsState(prev => {
|
||||||
|
if (
|
||||||
|
prev &&
|
||||||
|
prev.top === newBounds.top &&
|
||||||
|
prev.left === newBounds.left &&
|
||||||
|
prev.width === newBounds.width &&
|
||||||
|
prev.height === newBounds.height
|
||||||
|
) {
|
||||||
|
return prev
|
||||||
|
}
|
||||||
|
|
||||||
|
return newBounds
|
||||||
|
})
|
||||||
|
},
|
||||||
|
[props]
|
||||||
|
)
|
||||||
|
|
||||||
|
const clearBounds = React.useCallback(() => {
|
||||||
|
setBoundsState(prev => (prev === null ? prev : null))
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (value !== undefined) setActiveValue(value)
|
||||||
|
else if (defaultValue !== undefined) setActiveValue(defaultValue)
|
||||||
|
}, [value, defaultValue])
|
||||||
|
|
||||||
|
const id = React.useId()
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (mode !== 'parent') return
|
||||||
|
const container = localRef.current
|
||||||
|
|
||||||
|
if (!container) return
|
||||||
|
|
||||||
|
const onScroll = () => {
|
||||||
|
if (!activeValue) return
|
||||||
|
const activeEl = container.querySelector<HTMLElement>(`[data-value="${activeValue}"][data-highlight="true"]`)
|
||||||
|
|
||||||
|
if (activeEl) safeSetBounds(activeEl.getBoundingClientRect())
|
||||||
|
}
|
||||||
|
|
||||||
|
container.addEventListener('scroll', onScroll, { passive: true })
|
||||||
|
|
||||||
|
return () => container.removeEventListener('scroll', onScroll)
|
||||||
|
}, [mode, activeValue, safeSetBounds])
|
||||||
|
|
||||||
|
const render = React.useCallback(
|
||||||
|
(children: React.ReactNode) => {
|
||||||
|
if (mode === 'parent') {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
ref={localRef}
|
||||||
|
data-slot='motion-highlight-container'
|
||||||
|
className={cn('relative', (props as ParentModeMotionHighlightProps)?.containerClassName)}
|
||||||
|
>
|
||||||
|
<AnimatePresence initial={false}>
|
||||||
|
{boundsState && (
|
||||||
|
<motion.div
|
||||||
|
data-slot='motion-highlight'
|
||||||
|
animate={{
|
||||||
|
top: boundsState.top,
|
||||||
|
left: boundsState.left,
|
||||||
|
width: boundsState.width,
|
||||||
|
height: boundsState.height,
|
||||||
|
opacity: 1
|
||||||
|
}}
|
||||||
|
initial={{
|
||||||
|
top: boundsState.top,
|
||||||
|
left: boundsState.left,
|
||||||
|
width: boundsState.width,
|
||||||
|
height: boundsState.height,
|
||||||
|
opacity: 0
|
||||||
|
}}
|
||||||
|
exit={{
|
||||||
|
opacity: 0,
|
||||||
|
transition: {
|
||||||
|
...transition,
|
||||||
|
delay: (transition?.delay ?? 0) + (exitDelay ?? 0)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
transition={transition}
|
||||||
|
className={cn('bg-muted absolute z-0', className, activeClassNameState)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</AnimatePresence>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return children
|
||||||
|
},
|
||||||
|
[mode, props, boundsState, transition, exitDelay, className, activeClassNameState]
|
||||||
|
)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MotionHighlightContext.Provider
|
||||||
|
value={{
|
||||||
|
mode,
|
||||||
|
activeValue,
|
||||||
|
setActiveValue: safeSetActiveValue,
|
||||||
|
id,
|
||||||
|
hover,
|
||||||
|
className,
|
||||||
|
transition,
|
||||||
|
disabled,
|
||||||
|
enabled,
|
||||||
|
exitDelay,
|
||||||
|
setBounds: safeSetBounds,
|
||||||
|
clearBounds,
|
||||||
|
activeClassName: activeClassNameState,
|
||||||
|
setActiveClassName: setActiveClassNameState,
|
||||||
|
forceUpdateBounds: (props as ParentModeMotionHighlightProps)?.forceUpdateBounds
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{enabled
|
||||||
|
? controlledItems
|
||||||
|
? render(children)
|
||||||
|
: render(
|
||||||
|
React.Children.map(children, (child, index) => (
|
||||||
|
<MotionHighlightItem key={index} className={props?.itemsClassName}>
|
||||||
|
{child}
|
||||||
|
</MotionHighlightItem>
|
||||||
|
))
|
||||||
|
)
|
||||||
|
: children}
|
||||||
|
</MotionHighlightContext.Provider>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNonOverridingDataAttributes(
|
||||||
|
element: React.ReactElement,
|
||||||
|
dataAttributes: Record<string, unknown>
|
||||||
|
): Record<string, unknown> {
|
||||||
|
return Object.keys(dataAttributes).reduce<Record<string, unknown>>((acc, key) => {
|
||||||
|
if ((element.props as Record<string, unknown>)[key] === undefined) {
|
||||||
|
acc[key] = dataAttributes[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
return acc
|
||||||
|
}, {})
|
||||||
|
}
|
||||||
|
|
||||||
|
type ExtendedChildProps = React.ComponentProps<'div'> & {
|
||||||
|
id?: string
|
||||||
|
ref?: React.Ref<HTMLElement>
|
||||||
|
'data-active'?: string
|
||||||
|
'data-value'?: string
|
||||||
|
'data-disabled'?: boolean
|
||||||
|
'data-highlight'?: boolean
|
||||||
|
'data-slot'?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type MotionHighlightItemProps = React.ComponentProps<'div'> & {
|
||||||
|
children: React.ReactElement
|
||||||
|
id?: string
|
||||||
|
value?: string
|
||||||
|
className?: string
|
||||||
|
transition?: Transition
|
||||||
|
activeClassName?: string
|
||||||
|
disabled?: boolean
|
||||||
|
exitDelay?: number
|
||||||
|
asChild?: boolean
|
||||||
|
forceUpdateBounds?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
function MotionHighlightItem({
|
||||||
|
ref,
|
||||||
|
children,
|
||||||
|
id,
|
||||||
|
value,
|
||||||
|
className,
|
||||||
|
transition,
|
||||||
|
disabled = false,
|
||||||
|
activeClassName,
|
||||||
|
exitDelay,
|
||||||
|
asChild = false,
|
||||||
|
forceUpdateBounds,
|
||||||
|
...props
|
||||||
|
}: MotionHighlightItemProps) {
|
||||||
|
const itemId = React.useId()
|
||||||
|
|
||||||
|
const {
|
||||||
|
activeValue,
|
||||||
|
setActiveValue,
|
||||||
|
mode,
|
||||||
|
setBounds,
|
||||||
|
clearBounds,
|
||||||
|
hover,
|
||||||
|
enabled,
|
||||||
|
className: contextClassName,
|
||||||
|
transition: contextTransition,
|
||||||
|
id: contextId,
|
||||||
|
disabled: contextDisabled,
|
||||||
|
exitDelay: contextExitDelay,
|
||||||
|
forceUpdateBounds: contextForceUpdateBounds,
|
||||||
|
setActiveClassName
|
||||||
|
} = useMotionHighlight()
|
||||||
|
|
||||||
|
const element = children as React.ReactElement<ExtendedChildProps>
|
||||||
|
const childValue = id ?? value ?? element.props?.['data-value'] ?? element.props?.id ?? itemId
|
||||||
|
const isActive = activeValue === childValue
|
||||||
|
const isDisabled = disabled === undefined ? contextDisabled : disabled
|
||||||
|
const itemTransition = transition ?? contextTransition
|
||||||
|
|
||||||
|
const localRef = React.useRef<HTMLDivElement>(null)
|
||||||
|
|
||||||
|
React.useImperativeHandle(ref, () => localRef.current as HTMLDivElement)
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (mode !== 'parent') return
|
||||||
|
let rafId: number
|
||||||
|
let previousBounds: Bounds | null = null
|
||||||
|
const shouldUpdateBounds = forceUpdateBounds === true || (contextForceUpdateBounds && forceUpdateBounds !== false)
|
||||||
|
|
||||||
|
const updateBounds = () => {
|
||||||
|
if (!localRef.current) return
|
||||||
|
|
||||||
|
const bounds = localRef.current.getBoundingClientRect()
|
||||||
|
|
||||||
|
if (shouldUpdateBounds) {
|
||||||
|
if (
|
||||||
|
previousBounds &&
|
||||||
|
previousBounds.top === bounds.top &&
|
||||||
|
previousBounds.left === bounds.left &&
|
||||||
|
previousBounds.width === bounds.width &&
|
||||||
|
previousBounds.height === bounds.height
|
||||||
|
) {
|
||||||
|
rafId = requestAnimationFrame(updateBounds)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
previousBounds = bounds
|
||||||
|
rafId = requestAnimationFrame(updateBounds)
|
||||||
|
}
|
||||||
|
|
||||||
|
setBounds(bounds)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isActive) {
|
||||||
|
updateBounds()
|
||||||
|
setActiveClassName(activeClassName ?? '')
|
||||||
|
} else if (!activeValue) clearBounds()
|
||||||
|
|
||||||
|
if (shouldUpdateBounds) return () => cancelAnimationFrame(rafId)
|
||||||
|
}, [
|
||||||
|
mode,
|
||||||
|
isActive,
|
||||||
|
activeValue,
|
||||||
|
setBounds,
|
||||||
|
clearBounds,
|
||||||
|
activeClassName,
|
||||||
|
setActiveClassName,
|
||||||
|
forceUpdateBounds,
|
||||||
|
contextForceUpdateBounds
|
||||||
|
])
|
||||||
|
|
||||||
|
if (!React.isValidElement(children)) return children
|
||||||
|
|
||||||
|
const dataAttributes = {
|
||||||
|
'data-active': isActive ? 'true' : 'false',
|
||||||
|
'aria-selected': isActive,
|
||||||
|
'data-disabled': isDisabled,
|
||||||
|
'data-value': childValue,
|
||||||
|
'data-highlight': true
|
||||||
|
}
|
||||||
|
|
||||||
|
const commonHandlers = hover
|
||||||
|
? {
|
||||||
|
onMouseEnter: (e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
|
setActiveValue(childValue)
|
||||||
|
element.props.onMouseEnter?.(e)
|
||||||
|
},
|
||||||
|
onMouseLeave: (e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
|
setActiveValue(null)
|
||||||
|
element.props.onMouseLeave?.(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
: {
|
||||||
|
onClick: (e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
|
setActiveValue(childValue)
|
||||||
|
element.props.onClick?.(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (asChild) {
|
||||||
|
if (mode === 'children') {
|
||||||
|
return React.cloneElement(
|
||||||
|
element,
|
||||||
|
{
|
||||||
|
key: childValue,
|
||||||
|
ref: localRef,
|
||||||
|
className: cn('relative', element.props.className),
|
||||||
|
...getNonOverridingDataAttributes(element, {
|
||||||
|
...dataAttributes,
|
||||||
|
'data-slot': 'motion-highlight-item-container'
|
||||||
|
}),
|
||||||
|
...commonHandlers,
|
||||||
|
...props
|
||||||
|
},
|
||||||
|
<>
|
||||||
|
<AnimatePresence initial={false}>
|
||||||
|
{isActive && !isDisabled && (
|
||||||
|
<motion.div
|
||||||
|
layoutId={`transition-background-${contextId}`}
|
||||||
|
data-slot='motion-highlight'
|
||||||
|
className={cn('bg-muted absolute inset-0 z-0', contextClassName, activeClassName)}
|
||||||
|
transition={itemTransition}
|
||||||
|
initial={{ opacity: 0 }}
|
||||||
|
animate={{ opacity: 1 }}
|
||||||
|
exit={{
|
||||||
|
opacity: 0,
|
||||||
|
transition: {
|
||||||
|
...itemTransition,
|
||||||
|
delay: (itemTransition?.delay ?? 0) + (exitDelay ?? contextExitDelay ?? 0)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
{...dataAttributes}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</AnimatePresence>
|
||||||
|
|
||||||
|
<div data-slot='motion-highlight-item' className={cn('relative z-[1]', className)} {...dataAttributes}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return React.cloneElement(element, {
|
||||||
|
ref: localRef,
|
||||||
|
...getNonOverridingDataAttributes(element, {
|
||||||
|
...dataAttributes,
|
||||||
|
'data-slot': 'motion-highlight-item'
|
||||||
|
}),
|
||||||
|
...commonHandlers
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return enabled ? (
|
||||||
|
<div
|
||||||
|
key={childValue}
|
||||||
|
ref={localRef}
|
||||||
|
data-slot='motion-highlight-item-container'
|
||||||
|
className={cn(mode === 'children' && 'relative', className)}
|
||||||
|
{...dataAttributes}
|
||||||
|
{...props}
|
||||||
|
{...commonHandlers}
|
||||||
|
>
|
||||||
|
{mode === 'children' && (
|
||||||
|
<AnimatePresence initial={false}>
|
||||||
|
{isActive && !isDisabled && (
|
||||||
|
<motion.div
|
||||||
|
layoutId={`transition-background-${contextId}`}
|
||||||
|
data-slot='motion-highlight'
|
||||||
|
className={cn('bg-muted absolute inset-0 z-0', contextClassName, activeClassName)}
|
||||||
|
transition={itemTransition}
|
||||||
|
initial={{ opacity: 0 }}
|
||||||
|
animate={{ opacity: 1 }}
|
||||||
|
exit={{
|
||||||
|
opacity: 0,
|
||||||
|
transition: {
|
||||||
|
...itemTransition,
|
||||||
|
delay: (itemTransition?.delay ?? 0) + (exitDelay ?? contextExitDelay ?? 0)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
{...dataAttributes}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</AnimatePresence>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{React.cloneElement(element, {
|
||||||
|
className: cn('relative z-[1]', element.props.className),
|
||||||
|
...getNonOverridingDataAttributes(element, {
|
||||||
|
...dataAttributes,
|
||||||
|
'data-slot': 'motion-highlight-item'
|
||||||
|
})
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
children
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
MotionHighlight,
|
||||||
|
MotionHighlightItem,
|
||||||
|
useMotionHighlight,
|
||||||
|
type MotionHighlightProps,
|
||||||
|
type MotionHighlightItemProps
|
||||||
|
}
|
||||||
261
src/components/ui/motion-tabs.tsx
Normal file
261
src/components/ui/motion-tabs.tsx
Normal file
@@ -0,0 +1,261 @@
|
|||||||
|
'use client'
|
||||||
|
|
||||||
|
import * as React from 'react'
|
||||||
|
|
||||||
|
import { motion, type Transition, type HTMLMotionProps } from 'motion/react'
|
||||||
|
|
||||||
|
import { cn } from '@/lib/utils'
|
||||||
|
import { MotionHighlight, MotionHighlightItem } from '@/components/ui/motion-highlight'
|
||||||
|
|
||||||
|
type TabsContextType<T extends string> = {
|
||||||
|
activeValue: T
|
||||||
|
handleValueChange: (value: T) => void
|
||||||
|
registerTrigger: (value: T, node: HTMLElement | null) => void
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
const TabsContext = React.createContext<TabsContextType<any> | undefined>(undefined)
|
||||||
|
|
||||||
|
function useTabs<T extends string = string>(): TabsContextType<T> {
|
||||||
|
const context = React.useContext(TabsContext)
|
||||||
|
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('useTabs must be used within a TabsProvider')
|
||||||
|
}
|
||||||
|
|
||||||
|
return context
|
||||||
|
}
|
||||||
|
|
||||||
|
type BaseTabsProps = React.ComponentProps<'div'> & {
|
||||||
|
children: React.ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
type UnControlledTabsProps<T extends string = string> = BaseTabsProps & {
|
||||||
|
defaultValue?: T
|
||||||
|
value?: never
|
||||||
|
onValueChange?: never
|
||||||
|
}
|
||||||
|
|
||||||
|
type ControlledTabsProps<T extends string = string> = BaseTabsProps & {
|
||||||
|
value: T
|
||||||
|
onValueChange?: (value: T) => void
|
||||||
|
defaultValue?: never
|
||||||
|
}
|
||||||
|
|
||||||
|
type TabsProps<T extends string = string> = UnControlledTabsProps<T> | ControlledTabsProps<T>
|
||||||
|
|
||||||
|
function Tabs<T extends string = string>({
|
||||||
|
defaultValue,
|
||||||
|
value,
|
||||||
|
onValueChange,
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: TabsProps<T>) {
|
||||||
|
const [activeValue, setActiveValue] = React.useState<T | undefined>(defaultValue ?? undefined)
|
||||||
|
const triggersRef = React.useRef(new Map<string, HTMLElement>())
|
||||||
|
const initialSet = React.useRef(false)
|
||||||
|
const isControlled = value !== undefined
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!isControlled && activeValue === undefined && triggersRef.current.size > 0 && !initialSet.current) {
|
||||||
|
const firstTab = Array.from(triggersRef.current.keys())[0]
|
||||||
|
|
||||||
|
setActiveValue(firstTab as T)
|
||||||
|
initialSet.current = true
|
||||||
|
}
|
||||||
|
}, [activeValue, isControlled])
|
||||||
|
|
||||||
|
const registerTrigger = (value: string, node: HTMLElement | null) => {
|
||||||
|
if (node) {
|
||||||
|
triggersRef.current.set(value, node)
|
||||||
|
|
||||||
|
if (!isControlled && activeValue === undefined && !initialSet.current) {
|
||||||
|
setActiveValue(value as T)
|
||||||
|
initialSet.current = true
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
triggersRef.current.delete(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleValueChange = (val: T) => {
|
||||||
|
if (!isControlled) setActiveValue(val)
|
||||||
|
else onValueChange?.(val)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<TabsContext.Provider
|
||||||
|
value={{
|
||||||
|
activeValue: (value ?? activeValue)!,
|
||||||
|
handleValueChange,
|
||||||
|
registerTrigger
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div data-slot='tabs' className={cn('flex flex-col gap-2', className)} {...props}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</TabsContext.Provider>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type TabsListProps = React.ComponentProps<'div'> & {
|
||||||
|
children: React.ReactNode
|
||||||
|
activeClassName?: string
|
||||||
|
transition?: Transition
|
||||||
|
}
|
||||||
|
|
||||||
|
function TabsList({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
activeClassName,
|
||||||
|
transition = {
|
||||||
|
type: 'spring',
|
||||||
|
stiffness: 200,
|
||||||
|
damping: 25
|
||||||
|
},
|
||||||
|
...props
|
||||||
|
}: TabsListProps) {
|
||||||
|
const { activeValue } = useTabs()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MotionHighlight
|
||||||
|
controlledItems
|
||||||
|
className={cn('bg-background rounded-sm shadow-sm', activeClassName)}
|
||||||
|
value={activeValue}
|
||||||
|
transition={transition}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
role='tablist'
|
||||||
|
data-slot='tabs-list'
|
||||||
|
className={cn(
|
||||||
|
'bg-muted text-muted-foreground inline-flex h-10 w-fit items-center justify-center rounded-lg p-[4px]',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</MotionHighlight>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type TabsTriggerProps = HTMLMotionProps<'button'> & {
|
||||||
|
value: string
|
||||||
|
children: React.ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
function TabsTrigger({ ref, value, children, className, ...props }: TabsTriggerProps) {
|
||||||
|
const { activeValue, handleValueChange, registerTrigger } = useTabs()
|
||||||
|
|
||||||
|
const localRef = React.useRef<HTMLButtonElement | null>(null)
|
||||||
|
|
||||||
|
React.useImperativeHandle(ref, () => localRef.current as HTMLButtonElement)
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
registerTrigger(value, localRef.current)
|
||||||
|
|
||||||
|
return () => registerTrigger(value, null)
|
||||||
|
}, [value, registerTrigger])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MotionHighlightItem value={value} className='size-full'>
|
||||||
|
<motion.button
|
||||||
|
ref={localRef}
|
||||||
|
data-slot='tabs-trigger'
|
||||||
|
role='tab'
|
||||||
|
onClick={() => handleValueChange(value)}
|
||||||
|
data-state={activeValue === value ? 'active' : 'inactive'}
|
||||||
|
className={cn(
|
||||||
|
'ring-offset-background focus-visible:ring-ring data-[state=active]:text-foreground z-[1] inline-flex size-full cursor-pointer items-center justify-center rounded-sm px-2 py-1 text-sm font-medium whitespace-nowrap transition-transform focus-visible:ring-2 focus-visible:ring-offset-2 focus-visible:outline-none disabled:pointer-events-none disabled:opacity-50',
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</motion.button>
|
||||||
|
</MotionHighlightItem>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type TabsContentsProps = React.ComponentProps<'div'> & {
|
||||||
|
children: React.ReactNode
|
||||||
|
transition?: Transition
|
||||||
|
}
|
||||||
|
|
||||||
|
function TabsContents({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
transition = {
|
||||||
|
type: 'spring',
|
||||||
|
stiffness: 300,
|
||||||
|
damping: 30,
|
||||||
|
bounce: 0,
|
||||||
|
restDelta: 0.01
|
||||||
|
},
|
||||||
|
...props
|
||||||
|
}: TabsContentsProps) {
|
||||||
|
const { activeValue } = useTabs()
|
||||||
|
const childrenArray = React.Children.toArray(children)
|
||||||
|
|
||||||
|
const activeIndex = childrenArray.findIndex(
|
||||||
|
(child): child is React.ReactElement<{ value: string }> =>
|
||||||
|
React.isValidElement(child) &&
|
||||||
|
typeof child.props === 'object' &&
|
||||||
|
child.props !== null &&
|
||||||
|
'value' in child.props &&
|
||||||
|
child.props.value === activeValue
|
||||||
|
)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div data-slot='tabs-contents' className={cn('overflow-hidden', className)} {...props}>
|
||||||
|
<motion.div className='-mx-2 flex' animate={{ x: activeIndex * -100 + '%' }} transition={transition}>
|
||||||
|
{childrenArray.map((child, index) => (
|
||||||
|
<div key={index} className='w-full shrink-0 px-2'>
|
||||||
|
{child}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</motion.div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type TabsContentProps = HTMLMotionProps<'div'> & {
|
||||||
|
value: string
|
||||||
|
children: React.ReactNode
|
||||||
|
}
|
||||||
|
|
||||||
|
function TabsContent({ children, value, className, ...props }: TabsContentProps) {
|
||||||
|
const { activeValue } = useTabs()
|
||||||
|
const isActive = activeValue === value
|
||||||
|
|
||||||
|
return (
|
||||||
|
<motion.div
|
||||||
|
role='tabpanel'
|
||||||
|
data-slot='tabs-content'
|
||||||
|
className={cn('overflow-hidden', className)}
|
||||||
|
initial={{ filter: 'blur(0px)' }}
|
||||||
|
animate={{ filter: isActive ? 'blur(0px)' : 'blur(2px)' }}
|
||||||
|
exit={{ filter: 'blur(0px)' }}
|
||||||
|
transition={{ type: 'spring', stiffness: 300, damping: 20 }}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</motion.div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
Tabs,
|
||||||
|
TabsList,
|
||||||
|
TabsTrigger,
|
||||||
|
TabsContents,
|
||||||
|
TabsContent,
|
||||||
|
useTabs,
|
||||||
|
type TabsContextType,
|
||||||
|
type TabsProps,
|
||||||
|
type TabsListProps,
|
||||||
|
type TabsTriggerProps,
|
||||||
|
type TabsContentsProps,
|
||||||
|
type TabsContentProps
|
||||||
|
}
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
import type { TipoCiclo } from "./types";
|
|
||||||
|
|
||||||
export const FACULTADES = [
|
|
||||||
{ id: "ing", nombre: "Facultad de Ingeniería" },
|
|
||||||
{
|
|
||||||
id: "med",
|
|
||||||
nombre: "Facultad de Medicina en medicina en medicina en medicina",
|
|
||||||
},
|
|
||||||
{ id: "neg", nombre: "Facultad de Negocios" },
|
|
||||||
];
|
|
||||||
|
|
||||||
export const CARRERAS = [
|
|
||||||
{ id: "sis", nombre: "Ing. en Sistemas", facultadId: "ing" },
|
|
||||||
{ id: "ind", nombre: "Ing. Industrial", facultadId: "ing" },
|
|
||||||
{ id: "medico", nombre: "Médico Cirujano", facultadId: "med" },
|
|
||||||
{ id: "act", nombre: "Actuaría", facultadId: "neg" },
|
|
||||||
];
|
|
||||||
|
|
||||||
export const NIVELES = [
|
|
||||||
"Licenciatura",
|
|
||||||
"Especialidad",
|
|
||||||
"Maestría",
|
|
||||||
"Doctorado",
|
|
||||||
];
|
|
||||||
export const TIPOS_CICLO: Array<{ value: TipoCiclo; label: string }> = [
|
|
||||||
{ value: "SEMESTRE", label: "Semestre" },
|
|
||||||
{ value: "CUATRIMESTRE", label: "Cuatrimestre" },
|
|
||||||
{ value: "TRIMESTRE", label: "Trimestre" },
|
|
||||||
];
|
|
||||||
|
|
||||||
export const PLANES_EXISTENTES = [
|
|
||||||
{
|
|
||||||
id: "plan-2021-sis",
|
|
||||||
nombre: "ISC 2021",
|
|
||||||
estado: "Aprobado",
|
|
||||||
anio: 2021,
|
|
||||||
facultadId: "ing",
|
|
||||||
carreraId: "sis",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: "plan-2020-ind",
|
|
||||||
nombre: "I. Industrial 2020",
|
|
||||||
estado: "Aprobado",
|
|
||||||
anio: 2020,
|
|
||||||
facultadId: "ing",
|
|
||||||
carreraId: "ind",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: "plan-2019-med",
|
|
||||||
nombre: "Medicina 2019",
|
|
||||||
estado: "Vigente",
|
|
||||||
anio: 2019,
|
|
||||||
facultadId: "med",
|
|
||||||
carreraId: "medico",
|
|
||||||
},
|
|
||||||
];
|
|
||||||
@@ -3,8 +3,8 @@ import * as Icons from 'lucide-react'
|
|||||||
|
|
||||||
import { useNuevoPlanWizard } from './hooks/useNuevoPlanWizard'
|
import { useNuevoPlanWizard } from './hooks/useNuevoPlanWizard'
|
||||||
|
|
||||||
import { PasoBasicosForm } from '@/components/planes/wizard/PasoBasicosForm'
|
import { PasoBasicosForm } from '@/components/planes/wizard/PasoBasicosForm/PasoBasicosForm'
|
||||||
import { PasoDetallesPanel } from '@/components/planes/wizard/PasoDetallesPanel'
|
import { PasoDetallesPanel } from '@/components/planes/wizard/PasoDetallesPanel/PasoDetallesPanel'
|
||||||
import { PasoModoCardGroup } from '@/components/planes/wizard/PasoModoCardGroup'
|
import { PasoModoCardGroup } from '@/components/planes/wizard/PasoModoCardGroup'
|
||||||
import { PasoResumenCard } from '@/components/planes/wizard/PasoResumenCard'
|
import { PasoResumenCard } from '@/components/planes/wizard/PasoResumenCard'
|
||||||
import { WizardControls } from '@/components/planes/wizard/WizardControls'
|
import { WizardControls } from '@/components/planes/wizard/WizardControls'
|
||||||
152
src/features/planes/nuevo/catalogs.ts
Normal file
152
src/features/planes/nuevo/catalogs.ts
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
import type { TipoCiclo } from "./types";
|
||||||
|
|
||||||
|
export const FACULTADES = [
|
||||||
|
{ id: "ing", nombre: "Facultad de Ingeniería" },
|
||||||
|
{
|
||||||
|
id: "med",
|
||||||
|
nombre: "Facultad de Medicina en medicina en medicina en medicina",
|
||||||
|
},
|
||||||
|
{ id: "neg", nombre: "Facultad de Negocios" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const CARRERAS = [
|
||||||
|
{ id: "sis", nombre: "Ing. en Sistemas", facultadId: "ing" },
|
||||||
|
{ id: "ind", nombre: "Ing. Industrial", facultadId: "ing" },
|
||||||
|
{ id: "medico", nombre: "Médico Cirujano", facultadId: "med" },
|
||||||
|
{ id: "act", nombre: "Actuaría", facultadId: "neg" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const NIVELES = [
|
||||||
|
"Licenciatura",
|
||||||
|
"Especialidad",
|
||||||
|
"Maestría",
|
||||||
|
"Doctorado",
|
||||||
|
];
|
||||||
|
export const TIPOS_CICLO: Array<{ value: TipoCiclo; label: string }> = [
|
||||||
|
{ value: "SEMESTRE", label: "Semestre" },
|
||||||
|
{ value: "CUATRIMESTRE", label: "Cuatrimestre" },
|
||||||
|
{ value: "TRIMESTRE", label: "Trimestre" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const PLANES_EXISTENTES = [
|
||||||
|
{
|
||||||
|
id: "plan-2021-sis",
|
||||||
|
nombre: "ISC 2021",
|
||||||
|
estado: "Aprobado",
|
||||||
|
anio: 2021,
|
||||||
|
facultadId: "ing",
|
||||||
|
carreraId: "sis",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "plan-2020-ind",
|
||||||
|
nombre: "I. Industrial 2020",
|
||||||
|
estado: "Aprobado",
|
||||||
|
anio: 2020,
|
||||||
|
facultadId: "ing",
|
||||||
|
carreraId: "ind",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "plan-2019-med",
|
||||||
|
nombre: "Medicina 2019",
|
||||||
|
estado: "Vigente",
|
||||||
|
anio: 2019,
|
||||||
|
facultadId: "med",
|
||||||
|
carreraId: "medico",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const ARCHIVOS = [
|
||||||
|
{
|
||||||
|
id: "file-1",
|
||||||
|
nombre: "Sílabo POO 2023.docx",
|
||||||
|
tipo: "docx",
|
||||||
|
tamaño: "245 KB",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "file-2",
|
||||||
|
nombre: "Guía de prácticas BD.pdf",
|
||||||
|
tipo: "pdf",
|
||||||
|
tamaño: "1.2 MB",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "file-3",
|
||||||
|
nombre: "Rúbrica evaluación proyectos.xlsx",
|
||||||
|
tipo: "xlsx",
|
||||||
|
tamaño: "89 KB",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "file-4",
|
||||||
|
nombre: "Banco de reactivos IA.docx",
|
||||||
|
tipo: "docx",
|
||||||
|
tamaño: "567 KB",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "file-5",
|
||||||
|
nombre: "Material didáctico Web.pdf",
|
||||||
|
tipo: "pdf",
|
||||||
|
tamaño: "3.4 MB",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const REPOSITORIOS = [
|
||||||
|
{
|
||||||
|
id: "repo-1",
|
||||||
|
nombre: "Materiales ISC 2024",
|
||||||
|
descripcion: "Documentos de referencia para Ingeniería en Sistemas",
|
||||||
|
cantidadArchivos: 45,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "repo-2",
|
||||||
|
nombre: "Lineamientos SEP",
|
||||||
|
descripcion: "Documentos oficiales y normativas SEP actualizadas",
|
||||||
|
cantidadArchivos: 12,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "repo-3",
|
||||||
|
nombre: "Bibliografía Digital",
|
||||||
|
descripcion: "Recursos bibliográficos digitalizados",
|
||||||
|
cantidadArchivos: 128,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "repo-4",
|
||||||
|
nombre: "Plantillas Institucionales",
|
||||||
|
descripcion: "Formatos y plantillas oficiales ULSA",
|
||||||
|
cantidadArchivos: 23,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const PLANTILLAS_ANEXO_1 = [
|
||||||
|
{
|
||||||
|
id: "sep-2025",
|
||||||
|
name: "Licenciatura RVOE SEP.docx",
|
||||||
|
versions: ["v2025.2 (Vigente)", "v2025.1", "v2024.Final"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "interno-mix",
|
||||||
|
name: "Estándar Institucional Mixto.docx",
|
||||||
|
versions: ["v2.0", "v1.5", "v1.0-beta"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "conacyt",
|
||||||
|
name: "Formato Posgrado CONAHCYT.docx",
|
||||||
|
versions: ["v3.0 (2025)", "v2.8"],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const PLANTILLAS_ANEXO_2 = [
|
||||||
|
{
|
||||||
|
id: "sep-2017-xlsx",
|
||||||
|
name: "Licenciatura RVOE 2017.xlsx",
|
||||||
|
versions: ["v2017.0", "v2018.1", "v2019.2", "v2020.Final"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "interno-mix-xlsx",
|
||||||
|
name: "Estándar Institucional Mixto.xlsx",
|
||||||
|
versions: ["v1.0", "v1.5"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "conacyt-xlsx",
|
||||||
|
name: "Formato Posgrado CONAHCYT.xlsx",
|
||||||
|
versions: ["v1.0", "v2.0"],
|
||||||
|
},
|
||||||
|
];
|
||||||
@@ -2,19 +2,35 @@ import { useMemo, useState } from "react";
|
|||||||
|
|
||||||
import { CARRERAS } from "../catalogs";
|
import { CARRERAS } from "../catalogs";
|
||||||
|
|
||||||
import type { NewPlanWizardState, PlanPreview } from "../types";
|
import type { NewPlanWizardState, PlanPreview, TipoCiclo } from "../types";
|
||||||
|
|
||||||
export function useNuevoPlanWizard() {
|
export function useNuevoPlanWizard() {
|
||||||
const [wizard, setWizard] = useState<NewPlanWizardState>({
|
const [wizard, setWizard] = useState<NewPlanWizardState>({
|
||||||
step: 1,
|
step: 1,
|
||||||
modoCreacion: null,
|
modoCreacion: null,
|
||||||
|
// datosBasicos: {
|
||||||
|
// nombrePlan: "",
|
||||||
|
// carreraId: "",
|
||||||
|
// facultadId: "",
|
||||||
|
// nivel: "",
|
||||||
|
// tipoCiclo: "",
|
||||||
|
// numCiclos: undefined,
|
||||||
|
// plantillaPlanId: "",
|
||||||
|
// plantillaPlanVersion: "",
|
||||||
|
// plantillaMapaId: "",
|
||||||
|
// plantillaMapaVersion: "",
|
||||||
|
// },
|
||||||
datosBasicos: {
|
datosBasicos: {
|
||||||
nombrePlan: "",
|
nombrePlan: "Medicina",
|
||||||
carreraId: "",
|
carreraId: "medico",
|
||||||
facultadId: "",
|
facultadId: "med",
|
||||||
nivel: "",
|
nivel: "Licenciatura",
|
||||||
tipoCiclo: "SEMESTRE",
|
tipoCiclo: "SEMESTRE",
|
||||||
numCiclos: 8,
|
numCiclos: 8,
|
||||||
|
plantillaPlanId: "sep-2025",
|
||||||
|
plantillaPlanVersion: "v2025.2 (Vigente)",
|
||||||
|
plantillaMapaId: "sep-2017-xlsx",
|
||||||
|
plantillaMapaVersion: "v2017.0",
|
||||||
},
|
},
|
||||||
clonInterno: { planOrigenId: null },
|
clonInterno: { planOrigenId: null },
|
||||||
clonTradicional: {
|
clonTradicional: {
|
||||||
@@ -27,6 +43,8 @@ export function useNuevoPlanWizard() {
|
|||||||
poblacionObjetivo: "",
|
poblacionObjetivo: "",
|
||||||
notasAdicionales: "",
|
notasAdicionales: "",
|
||||||
archivosReferencia: [],
|
archivosReferencia: [],
|
||||||
|
repositoriosReferencia: [],
|
||||||
|
archivosAdjuntos: [],
|
||||||
},
|
},
|
||||||
resumen: {},
|
resumen: {},
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
@@ -46,12 +64,20 @@ export function useNuevoPlanWizard() {
|
|||||||
!!wizard.datosBasicos.carreraId &&
|
!!wizard.datosBasicos.carreraId &&
|
||||||
!!wizard.datosBasicos.facultadId &&
|
!!wizard.datosBasicos.facultadId &&
|
||||||
!!wizard.datosBasicos.nivel &&
|
!!wizard.datosBasicos.nivel &&
|
||||||
wizard.datosBasicos.numCiclos > 0;
|
(wizard.datosBasicos.numCiclos !== undefined &&
|
||||||
|
wizard.datosBasicos.numCiclos > 0) &&
|
||||||
|
// Requerir ambas plantillas (plan y mapa) con versión
|
||||||
|
!!wizard.datosBasicos.plantillaPlanId &&
|
||||||
|
!!wizard.datosBasicos.plantillaPlanVersion &&
|
||||||
|
!!wizard.datosBasicos.plantillaMapaId &&
|
||||||
|
!!wizard.datosBasicos.plantillaMapaVersion;
|
||||||
|
|
||||||
const canContinueDesdeDetalles = (() => {
|
const canContinueDesdeDetalles = (() => {
|
||||||
if (wizard.modoCreacion === "MANUAL") return true;
|
if (wizard.modoCreacion === "MANUAL") return true;
|
||||||
if (wizard.modoCreacion === "IA") {
|
if (wizard.modoCreacion === "IA") {
|
||||||
return !!wizard.iaConfig?.descripcionEnfoque;
|
// Requerimos descripción del enfoque y notas adicionales
|
||||||
|
return !!wizard.iaConfig?.descripcionEnfoque &&
|
||||||
|
!!wizard.iaConfig?.notasAdicionales;
|
||||||
}
|
}
|
||||||
if (wizard.modoCreacion === "CLONADO") {
|
if (wizard.modoCreacion === "CLONADO") {
|
||||||
if (wizard.subModoClonado === "INTERNO") {
|
if (wizard.subModoClonado === "INTERNO") {
|
||||||
@@ -72,12 +98,24 @@ export function useNuevoPlanWizard() {
|
|||||||
const generarPreviewIA = async () => {
|
const generarPreviewIA = async () => {
|
||||||
setWizard((w) => ({ ...w, isLoading: true, errorMessage: null }));
|
setWizard((w) => ({ ...w, isLoading: true, errorMessage: null }));
|
||||||
await new Promise((r) => setTimeout(r, 800));
|
await new Promise((r) => setTimeout(r, 800));
|
||||||
|
// Ensure preview has the stricter types required by `PlanPreview`.
|
||||||
|
let tipoCicloSafe: TipoCiclo;
|
||||||
|
if (wizard.datosBasicos.tipoCiclo === "") {
|
||||||
|
tipoCicloSafe = "SEMESTRE";
|
||||||
|
} else {
|
||||||
|
tipoCicloSafe = wizard.datosBasicos.tipoCiclo;
|
||||||
|
}
|
||||||
|
const numCiclosSafe: number =
|
||||||
|
typeof wizard.datosBasicos.numCiclos === "number"
|
||||||
|
? wizard.datosBasicos.numCiclos
|
||||||
|
: 1;
|
||||||
|
|
||||||
const preview: PlanPreview = {
|
const preview: PlanPreview = {
|
||||||
nombrePlan: wizard.datosBasicos.nombrePlan || "Plan sin nombre",
|
nombrePlan: wizard.datosBasicos.nombrePlan || "Plan sin nombre",
|
||||||
nivel: wizard.datosBasicos.nivel || "Licenciatura",
|
nivel: wizard.datosBasicos.nivel || "Licenciatura",
|
||||||
tipoCiclo: wizard.datosBasicos.tipoCiclo,
|
tipoCiclo: tipoCicloSafe,
|
||||||
numCiclos: wizard.datosBasicos.numCiclos,
|
numCiclos: numCiclosSafe,
|
||||||
numAsignaturasAprox: wizard.datosBasicos.numCiclos * 6,
|
numAsignaturasAprox: numCiclosSafe * 6,
|
||||||
secciones: [
|
secciones: [
|
||||||
{ id: "obj", titulo: "Objetivos", resumen: "Borrador de objetivos…" },
|
{ id: "obj", titulo: "Objetivos", resumen: "Borrador de objetivos…" },
|
||||||
{ id: "perfil", titulo: "Perfil de egreso", resumen: "Borrador…" },
|
{ id: "perfil", titulo: "Perfil de egreso", resumen: "Borrador…" },
|
||||||
@@ -20,20 +20,46 @@ export type NewPlanWizardState = {
|
|||||||
carreraId: string;
|
carreraId: string;
|
||||||
facultadId: string;
|
facultadId: string;
|
||||||
nivel: string;
|
nivel: string;
|
||||||
tipoCiclo: TipoCiclo;
|
tipoCiclo: TipoCiclo | "";
|
||||||
numCiclos: number;
|
numCiclos: number | undefined;
|
||||||
|
// Selección de plantillas (obligatorias)
|
||||||
|
plantillaPlanId?: string;
|
||||||
|
plantillaPlanVersion?: string;
|
||||||
|
plantillaMapaId?: string;
|
||||||
|
plantillaMapaVersion?: string;
|
||||||
};
|
};
|
||||||
clonInterno?: { planOrigenId: string | null };
|
clonInterno?: { planOrigenId: string | null };
|
||||||
clonTradicional?: {
|
clonTradicional?: {
|
||||||
archivoWordPlanId: string | null;
|
archivoWordPlanId:
|
||||||
archivoMapaExcelId: string | null;
|
| {
|
||||||
archivoAsignaturasExcelId: string | null;
|
id: string;
|
||||||
|
name: string;
|
||||||
|
size: string;
|
||||||
|
type: string;
|
||||||
|
}
|
||||||
|
| null;
|
||||||
|
archivoMapaExcelId: {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
size: string;
|
||||||
|
type: string;
|
||||||
|
} | null;
|
||||||
|
archivoAsignaturasExcelId: {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
size: string;
|
||||||
|
type: string;
|
||||||
|
} | null;
|
||||||
};
|
};
|
||||||
iaConfig?: {
|
iaConfig?: {
|
||||||
descripcionEnfoque: string;
|
descripcionEnfoque: string;
|
||||||
poblacionObjetivo: string;
|
poblacionObjetivo: string;
|
||||||
notasAdicionales: string;
|
notasAdicionales: string;
|
||||||
archivosReferencia: Array<string>;
|
archivosReferencia: Array<string>;
|
||||||
|
repositoriosReferencia?: Array<string>;
|
||||||
|
archivosAdjuntos?: Array<
|
||||||
|
{ id: string; name: string; size: string; type: string }
|
||||||
|
>;
|
||||||
};
|
};
|
||||||
resumen: { previewPlan?: PlanPreview };
|
resumen: { previewPlan?: PlanPreview };
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { createFileRoute } from '@tanstack/react-router'
|
import { createFileRoute } from '@tanstack/react-router'
|
||||||
|
|
||||||
import { NuevaAsignaturaModalContainer } from '@/features/asignaturas/new/NuevaAsignaturaModalContainer'
|
import { NuevaAsignaturaModalContainer } from '@/features/asignaturas/nueva/NuevaAsignaturaModalContainer'
|
||||||
|
|
||||||
export const Route = createFileRoute(
|
export const Route = createFileRoute(
|
||||||
'/planes/$planId/asignaturas/_lista/nueva',
|
'/planes/$planId/asignaturas/_lista/nueva',
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { createFileRoute } from '@tanstack/react-router'
|
import { createFileRoute } from '@tanstack/react-router'
|
||||||
|
|
||||||
import NuevoPlanModalContainer from '@/features/planes/new/NuevoPlanModalContainer'
|
import NuevoPlanModalContainer from '@/features/planes/nuevo/NuevoPlanModalContainer'
|
||||||
|
|
||||||
export const Route = createFileRoute('/planes/_lista/nuevo')({
|
export const Route = createFileRoute('/planes/_lista/nuevo')({
|
||||||
component: NuevoPlanModalContainer,
|
component: NuevoPlanModalContainer,
|
||||||
|
|||||||
@@ -183,10 +183,19 @@ function RouteComponent() {
|
|||||||
|
|
||||||
// Filtrado de planes
|
// Filtrado de planes
|
||||||
const filteredPlans = useMemo(() => {
|
const filteredPlans = useMemo(() => {
|
||||||
const term = search.trim().toLowerCase()
|
// Función helper para limpiar texto (quita acentos y hace minúsculas)
|
||||||
|
const cleanText = (text: string) => {
|
||||||
|
return text
|
||||||
|
.normalize('NFD') // Descompone "á" en "a" + "´"
|
||||||
|
.replace(/[\u0300-\u036f]/g, '') // Elimina los símbolos diacríticos
|
||||||
|
.toLowerCase() // Convierte a minúsculas
|
||||||
|
}
|
||||||
|
// Limpiamos el término de búsqueda una sola vez antes de filtrar
|
||||||
|
const term = cleanText(search.trim())
|
||||||
return planes.filter((p) => {
|
return planes.filter((p) => {
|
||||||
const matchName = term
|
const matchName = term
|
||||||
? p.nombrePrograma.toLowerCase().includes(term)
|
? // Limpiamos también el nombre del programa antes de comparar
|
||||||
|
cleanText(p.nombrePrograma).includes(term)
|
||||||
: true
|
: true
|
||||||
const matchFac =
|
const matchFac =
|
||||||
facultadSel === 'todas' ? true : p.facultadId === facultadSel
|
facultadSel === 'todas' ? true : p.facultadId === facultadSel
|
||||||
|
|||||||
Reference in New Issue
Block a user