Compare commits
28 Commits
feature/Pd
...
fix/typos
| Author | SHA1 | Date | |
|---|---|---|---|
| e3c1a0ce2b | |||
| 76170421aa | |||
| 2db3a0570a | |||
| d8ade3da75 | |||
| 6a28af26b5 | |||
| 9d9fb3d8a8 | |||
| a2dddae5f3 | |||
| a6efb496db | |||
| ef6cc7b96d | |||
| a6f0010a53 | |||
| 29231206c0 | |||
| 93c79eee77 | |||
| 6f97a83eb0 | |||
| 4ec2c2d533 | |||
| efe7faa65f | |||
| c9d66ce2e5 | |||
| f7a29ad510 | |||
| e7a47f56f8 | |||
| 214d17cf98 | |||
| 8c890d76e0 | |||
| 7105b286bf | |||
| 0e884f20c5 | |||
| 8bb8399ec5 | |||
| 6d264a8214 | |||
| 6e2b3d72f1 | |||
| 0c5c3f935b | |||
| 8da08b6bf1 | |||
| 1fe8f2b6a8 |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -4,6 +4,7 @@ dist
|
|||||||
dist-ssr
|
dist-ssr
|
||||||
*.local
|
*.local
|
||||||
count.txt
|
count.txt
|
||||||
.env
|
.env*
|
||||||
.nitro
|
.nitro
|
||||||
.tanstack
|
.tanstack
|
||||||
|
.cta.json
|
||||||
|
|||||||
48
bun.lock
48
bun.lock
@@ -32,6 +32,8 @@
|
|||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"cmdk": "^1.1.1",
|
"cmdk": "^1.1.1",
|
||||||
"gsap": "^3.13.0",
|
"gsap": "^3.13.0",
|
||||||
|
"jspdf": "^3.0.3",
|
||||||
|
"jspdf-autotable": "^5.0.2",
|
||||||
"lucide-react": "^0.540.0",
|
"lucide-react": "^0.540.0",
|
||||||
"next-themes": "^0.4.6",
|
"next-themes": "^0.4.6",
|
||||||
"react": "^19.0.0",
|
"react": "^19.0.0",
|
||||||
@@ -482,12 +484,18 @@
|
|||||||
|
|
||||||
"@types/node": ["@types/node@24.3.0", "", { "dependencies": { "undici-types": "~7.10.0" } }, "sha512-aPTXCrfwnDLj4VvXrm+UUCQjNEvJgNA8s5F1cvwQU+3KNltTOkBm1j30uNLyqqPNe7gE3KFzImYoZEfLhp4Yow=="],
|
"@types/node": ["@types/node@24.3.0", "", { "dependencies": { "undici-types": "~7.10.0" } }, "sha512-aPTXCrfwnDLj4VvXrm+UUCQjNEvJgNA8s5F1cvwQU+3KNltTOkBm1j30uNLyqqPNe7gE3KFzImYoZEfLhp4Yow=="],
|
||||||
|
|
||||||
|
"@types/pako": ["@types/pako@2.0.4", "", {}, "sha512-VWDCbrLeVXJM9fihYodcLiIv0ku+AlOa/TQ1SvYOaBuyrSKgEcro95LJyIsJ4vSo6BXIxOKxiJAat04CmST9Fw=="],
|
||||||
|
|
||||||
"@types/phoenix": ["@types/phoenix@1.6.6", "", {}, "sha512-PIzZZlEppgrpoT2QgbnDU+MMzuR6BbCjllj0bM70lWoejMeNJAxCchxnv7J3XFkI8MpygtRpzXrIlmWUBclP5A=="],
|
"@types/phoenix": ["@types/phoenix@1.6.6", "", {}, "sha512-PIzZZlEppgrpoT2QgbnDU+MMzuR6BbCjllj0bM70lWoejMeNJAxCchxnv7J3XFkI8MpygtRpzXrIlmWUBclP5A=="],
|
||||||
|
|
||||||
|
"@types/raf": ["@types/raf@3.4.3", "", {}, "sha512-c4YAvMedbPZ5tEyxzQdMoOhhJ4RD3rngZIdwC2/qDN3d7JpEhB6fiBRKVY1lg5B7Wk+uPBjn5f39j1/2MY1oOw=="],
|
||||||
|
|
||||||
"@types/react": ["@types/react@19.1.10", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-EhBeSYX0Y6ye8pNebpKrwFJq7BoQ8J5SO6NlvNwwHjSj6adXJViPQrKlsyPw7hLBLvckEMO1yxeGdR82YBBlDg=="],
|
"@types/react": ["@types/react@19.1.10", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-EhBeSYX0Y6ye8pNebpKrwFJq7BoQ8J5SO6NlvNwwHjSj6adXJViPQrKlsyPw7hLBLvckEMO1yxeGdR82YBBlDg=="],
|
||||||
|
|
||||||
"@types/react-dom": ["@types/react-dom@19.1.7", "", { "peerDependencies": { "@types/react": "^19.0.0" } }, "sha512-i5ZzwYpqjmrKenzkoLM2Ibzt6mAsM7pxB6BCIouEVVmgiqaMj1TjaK7hnA36hbW5aZv20kx7Lw6hWzPWg0Rurw=="],
|
"@types/react-dom": ["@types/react-dom@19.1.7", "", { "peerDependencies": { "@types/react": "^19.0.0" } }, "sha512-i5ZzwYpqjmrKenzkoLM2Ibzt6mAsM7pxB6BCIouEVVmgiqaMj1TjaK7hnA36hbW5aZv20kx7Lw6hWzPWg0Rurw=="],
|
||||||
|
|
||||||
|
"@types/trusted-types": ["@types/trusted-types@2.0.7", "", {}, "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw=="],
|
||||||
|
|
||||||
"@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="],
|
"@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="],
|
||||||
|
|
||||||
"@types/use-sync-external-store": ["@types/use-sync-external-store@0.0.6", "", {}, "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg=="],
|
"@types/use-sync-external-store": ["@types/use-sync-external-store@0.0.6", "", {}, "sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg=="],
|
||||||
@@ -536,6 +544,8 @@
|
|||||||
|
|
||||||
"bail": ["bail@2.0.2", "", {}, "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw=="],
|
"bail": ["bail@2.0.2", "", {}, "sha512-0xO6mYd7JB2YesxDKplafRpsiOzPt9V02ddPCLbY1xYGPOX24NTyN50qnUxgCPcSoYMhKpAuBTjQoRZCAkUDRw=="],
|
||||||
|
|
||||||
|
"base64-arraybuffer": ["base64-arraybuffer@1.0.2", "", {}, "sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ=="],
|
||||||
|
|
||||||
"binary-extensions": ["binary-extensions@2.3.0", "", {}, "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw=="],
|
"binary-extensions": ["binary-extensions@2.3.0", "", {}, "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw=="],
|
||||||
|
|
||||||
"braces": ["braces@3.0.3", "", { "dependencies": { "fill-range": "^7.1.1" } }, "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA=="],
|
"braces": ["braces@3.0.3", "", { "dependencies": { "fill-range": "^7.1.1" } }, "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA=="],
|
||||||
@@ -548,6 +558,8 @@
|
|||||||
|
|
||||||
"canvas-confetti": ["canvas-confetti@1.9.3", "", {}, "sha512-rFfTURMvmVEX1gyXFgn5QMn81bYk70qa0HLzcIOSVEyl57n6o9ItHeBtUSWdvKAPY0xlvBHno4/v3QPrT83q9g=="],
|
"canvas-confetti": ["canvas-confetti@1.9.3", "", {}, "sha512-rFfTURMvmVEX1gyXFgn5QMn81bYk70qa0HLzcIOSVEyl57n6o9ItHeBtUSWdvKAPY0xlvBHno4/v3QPrT83q9g=="],
|
||||||
|
|
||||||
|
"canvg": ["canvg@3.0.11", "", { "dependencies": { "@babel/runtime": "^7.12.5", "@types/raf": "^3.4.0", "core-js": "^3.8.3", "raf": "^3.4.1", "regenerator-runtime": "^0.13.7", "rgbcolor": "^1.0.1", "stackblur-canvas": "^2.0.0", "svg-pathdata": "^6.0.3" } }, "sha512-5ON+q7jCTgMp9cjpu4Jo6XbvfYwSB2Ow3kzHKfIyJfaCAOHLbdKPQqGKgfED/R5B+3TFFfe8pegYA+b423SRyA=="],
|
||||||
|
|
||||||
"ccount": ["ccount@2.0.1", "", {}, "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg=="],
|
"ccount": ["ccount@2.0.1", "", {}, "sha512-eyrF0jiFpY+3drT6383f1qhkbGsLSifNAjA61IUjZjmLCWjItY6LB9ft9YhoDgwfmclB2zhu51Lc7+95b8NRAg=="],
|
||||||
|
|
||||||
"chai": ["chai@5.3.1", "", { "dependencies": { "assertion-error": "^2.0.1", "check-error": "^2.1.1", "deep-eql": "^5.0.1", "loupe": "^3.1.0", "pathval": "^2.0.0" } }, "sha512-48af6xm9gQK8rhIcOxWwdGzIervm8BVTin+yRp9HEvU20BtVZ2lBywlIJBzwaDtvo0FvjeL7QdCADoUoqIbV3A=="],
|
"chai": ["chai@5.3.1", "", { "dependencies": { "assertion-error": "^2.0.1", "check-error": "^2.1.1", "deep-eql": "^5.0.1", "loupe": "^3.1.0", "pathval": "^2.0.0" } }, "sha512-48af6xm9gQK8rhIcOxWwdGzIervm8BVTin+yRp9HEvU20BtVZ2lBywlIJBzwaDtvo0FvjeL7QdCADoUoqIbV3A=="],
|
||||||
@@ -578,6 +590,10 @@
|
|||||||
|
|
||||||
"cookie-es": ["cookie-es@1.2.2", "", {}, "sha512-+W7VmiVINB+ywl1HGXJXmrqkOhpKrIiVZV6tQuV54ZyQC7MMuBt81Vc336GMLoHBq5hV/F9eXgt5Mnx0Rha5Fg=="],
|
"cookie-es": ["cookie-es@1.2.2", "", {}, "sha512-+W7VmiVINB+ywl1HGXJXmrqkOhpKrIiVZV6tQuV54ZyQC7MMuBt81Vc336GMLoHBq5hV/F9eXgt5Mnx0Rha5Fg=="],
|
||||||
|
|
||||||
|
"core-js": ["core-js@3.46.0", "", {}, "sha512-vDMm9B0xnqqZ8uSBpZ8sNtRtOdmfShrvT6h2TuQGLs0Is+cR0DYbj/KWP6ALVNbWPpqA/qPLoOuppJN07humpA=="],
|
||||||
|
|
||||||
|
"css-line-break": ["css-line-break@2.1.0", "", { "dependencies": { "utrie": "^1.0.2" } }, "sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w=="],
|
||||||
|
|
||||||
"cssstyle": ["cssstyle@4.6.0", "", { "dependencies": { "@asamuzakjp/css-color": "^3.2.0", "rrweb-cssom": "^0.8.0" } }, "sha512-2z+rWdzbbSZv6/rhtvzvqeZQHrBaqgogqt85sqFNbabZOuFbCVFb8kPeEtZjiKkbrm395irpNKiYeFeLiQnFPg=="],
|
"cssstyle": ["cssstyle@4.6.0", "", { "dependencies": { "@asamuzakjp/css-color": "^3.2.0", "rrweb-cssom": "^0.8.0" } }, "sha512-2z+rWdzbbSZv6/rhtvzvqeZQHrBaqgogqt85sqFNbabZOuFbCVFb8kPeEtZjiKkbrm395irpNKiYeFeLiQnFPg=="],
|
||||||
|
|
||||||
"csstype": ["csstype@3.1.3", "", {}, "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw=="],
|
"csstype": ["csstype@3.1.3", "", {}, "sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw=="],
|
||||||
@@ -628,6 +644,8 @@
|
|||||||
|
|
||||||
"dom-accessibility-api": ["dom-accessibility-api@0.5.16", "", {}, "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg=="],
|
"dom-accessibility-api": ["dom-accessibility-api@0.5.16", "", {}, "sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg=="],
|
||||||
|
|
||||||
|
"dompurify": ["dompurify@3.3.0", "", { "optionalDependencies": { "@types/trusted-types": "^2.0.7" } }, "sha512-r+f6MYR1gGN1eJv0TVQbhA7if/U7P87cdPl3HN5rikqaBSBxLiCb/b9O+2eG0cxz0ghyU+mU1QkbsOwERMYlWQ=="],
|
||||||
|
|
||||||
"electron-to-chromium": ["electron-to-chromium@1.5.204", "", {}, "sha512-s9VbBXWxfDrl67PlO4avwh0/GU2vcwx8Fph3wlR8LJl7ySGYId59EFE17VWVcuC3sLWNPENm6Z/uGqKbkPCcXA=="],
|
"electron-to-chromium": ["electron-to-chromium@1.5.204", "", {}, "sha512-s9VbBXWxfDrl67PlO4avwh0/GU2vcwx8Fph3wlR8LJl7ySGYId59EFE17VWVcuC3sLWNPENm6Z/uGqKbkPCcXA=="],
|
||||||
|
|
||||||
"enhanced-resolve": ["enhanced-resolve@5.18.3", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.2.0" } }, "sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww=="],
|
"enhanced-resolve": ["enhanced-resolve@5.18.3", "", { "dependencies": { "graceful-fs": "^4.2.4", "tapable": "^2.2.0" } }, "sha512-d4lC8xfavMeBjzGr2vECC3fsGXziXZQyJxD868h2M/mBI3PwAuODxAkLkq5HYuvrPYcUtiLzsTo8U3PgX3Ocww=="],
|
||||||
@@ -654,8 +672,12 @@
|
|||||||
|
|
||||||
"extend": ["extend@3.0.2", "", {}, "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g=="],
|
"extend": ["extend@3.0.2", "", {}, "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g=="],
|
||||||
|
|
||||||
|
"fast-png": ["fast-png@6.4.0", "", { "dependencies": { "@types/pako": "^2.0.3", "iobuffer": "^5.3.2", "pako": "^2.1.0" } }, "sha512-kAqZq1TlgBjZcLr5mcN6NP5Rv4V2f22z00c3g8vRrwkcqjerx7BEhPbOnWCPqaHUl2XWQBJQvOT/FQhdMT7X/Q=="],
|
||||||
|
|
||||||
"fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="],
|
"fdir": ["fdir@6.5.0", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg=="],
|
||||||
|
|
||||||
|
"fflate": ["fflate@0.8.2", "", {}, "sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A=="],
|
||||||
|
|
||||||
"fill-range": ["fill-range@7.1.1", "", { "dependencies": { "to-regex-range": "^5.0.1" } }, "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg=="],
|
"fill-range": ["fill-range@7.1.1", "", { "dependencies": { "to-regex-range": "^5.0.1" } }, "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg=="],
|
||||||
|
|
||||||
"fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="],
|
"fsevents": ["fsevents@2.3.3", "", { "os": "darwin" }, "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw=="],
|
||||||
@@ -682,6 +704,8 @@
|
|||||||
|
|
||||||
"html-url-attributes": ["html-url-attributes@3.0.1", "", {}, "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ=="],
|
"html-url-attributes": ["html-url-attributes@3.0.1", "", {}, "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ=="],
|
||||||
|
|
||||||
|
"html2canvas": ["html2canvas@1.4.1", "", { "dependencies": { "css-line-break": "^2.1.0", "text-segmentation": "^1.0.3" } }, "sha512-fPU6BHNpsyIhr8yyMpTLLxAbkaK8ArIBcmZIRiBLiDhjeqvXolaEmDGmELFuX9I4xDcaKKcJl+TKZLqruBbmWA=="],
|
||||||
|
|
||||||
"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=="],
|
||||||
|
|
||||||
"https-proxy-agent": ["https-proxy-agent@7.0.6", "", { "dependencies": { "agent-base": "^7.1.2", "debug": "4" } }, "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw=="],
|
"https-proxy-agent": ["https-proxy-agent@7.0.6", "", { "dependencies": { "agent-base": "^7.1.2", "debug": "4" } }, "sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw=="],
|
||||||
@@ -694,6 +718,8 @@
|
|||||||
|
|
||||||
"internmap": ["internmap@2.0.3", "", {}, "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg=="],
|
"internmap": ["internmap@2.0.3", "", {}, "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg=="],
|
||||||
|
|
||||||
|
"iobuffer": ["iobuffer@5.4.0", "", {}, "sha512-DRebOWuqDvxunfkNJAlc3IzWIPD5xVxwUNbHr7xKB8E6aLJxIPfNX3CoMJghcFjpv6RWQsrcJbghtEwSPoJqMA=="],
|
||||||
|
|
||||||
"is-alphabetical": ["is-alphabetical@2.0.1", "", {}, "sha512-FWyyY60MeTNyeSRpkM2Iry0G9hpr7/9kD40mD/cGQEuilcZYS4okz8SN2Q6rLCJ8gbCt6fN+rC+6tMGS99LaxQ=="],
|
"is-alphabetical": ["is-alphabetical@2.0.1", "", {}, "sha512-FWyyY60MeTNyeSRpkM2Iry0G9hpr7/9kD40mD/cGQEuilcZYS4okz8SN2Q6rLCJ8gbCt6fN+rC+6tMGS99LaxQ=="],
|
||||||
|
|
||||||
"is-alphanumerical": ["is-alphanumerical@2.0.1", "", { "dependencies": { "is-alphabetical": "^2.0.0", "is-decimal": "^2.0.0" } }, "sha512-hmbYhX/9MUMF5uh7tOXyK/n0ZvWpad5caBA17GsC6vyuCqaWliRG5K1qS9inmUhEMaOBIW7/whAnSwveW/LtZw=="],
|
"is-alphanumerical": ["is-alphanumerical@2.0.1", "", { "dependencies": { "is-alphabetical": "^2.0.0", "is-decimal": "^2.0.0" } }, "sha512-hmbYhX/9MUMF5uh7tOXyK/n0ZvWpad5caBA17GsC6vyuCqaWliRG5K1qS9inmUhEMaOBIW7/whAnSwveW/LtZw=="],
|
||||||
@@ -726,6 +752,10 @@
|
|||||||
|
|
||||||
"json5": ["json5@2.2.3", "", { "bin": { "json5": "lib/cli.js" } }, "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg=="],
|
"json5": ["json5@2.2.3", "", { "bin": { "json5": "lib/cli.js" } }, "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg=="],
|
||||||
|
|
||||||
|
"jspdf": ["jspdf@3.0.3", "", { "dependencies": { "@babel/runtime": "^7.26.9", "fast-png": "^6.2.0", "fflate": "^0.8.1" }, "optionalDependencies": { "canvg": "^3.0.11", "core-js": "^3.6.0", "dompurify": "^3.2.4", "html2canvas": "^1.0.0-rc.5" } }, "sha512-eURjAyz5iX1H8BOYAfzvdPfIKK53V7mCpBTe7Kb16PaM8JSXEcUQNBQaiWMI8wY5RvNOPj4GccMjTlfwRBd+oQ=="],
|
||||||
|
|
||||||
|
"jspdf-autotable": ["jspdf-autotable@5.0.2", "", { "peerDependencies": { "jspdf": "^2 || ^3" } }, "sha512-YNKeB7qmx3pxOLcNeoqAv3qTS7KuvVwkFe5AduCawpop3NOkBUtqDToxNc225MlNecxT4kP2Zy3z/y/yvGdXUQ=="],
|
||||||
|
|
||||||
"lightningcss": ["lightningcss@1.30.1", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-darwin-arm64": "1.30.1", "lightningcss-darwin-x64": "1.30.1", "lightningcss-freebsd-x64": "1.30.1", "lightningcss-linux-arm-gnueabihf": "1.30.1", "lightningcss-linux-arm64-gnu": "1.30.1", "lightningcss-linux-arm64-musl": "1.30.1", "lightningcss-linux-x64-gnu": "1.30.1", "lightningcss-linux-x64-musl": "1.30.1", "lightningcss-win32-arm64-msvc": "1.30.1", "lightningcss-win32-x64-msvc": "1.30.1" } }, "sha512-xi6IyHML+c9+Q3W0S4fCQJOym42pyurFiJUHEcEyHS0CeKzia4yZDEsLlqOFykxOdHpNy0NmvVO31vcSqAxJCg=="],
|
"lightningcss": ["lightningcss@1.30.1", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-darwin-arm64": "1.30.1", "lightningcss-darwin-x64": "1.30.1", "lightningcss-freebsd-x64": "1.30.1", "lightningcss-linux-arm-gnueabihf": "1.30.1", "lightningcss-linux-arm64-gnu": "1.30.1", "lightningcss-linux-arm64-musl": "1.30.1", "lightningcss-linux-x64-gnu": "1.30.1", "lightningcss-linux-x64-musl": "1.30.1", "lightningcss-win32-arm64-msvc": "1.30.1", "lightningcss-win32-x64-msvc": "1.30.1" } }, "sha512-xi6IyHML+c9+Q3W0S4fCQJOym42pyurFiJUHEcEyHS0CeKzia4yZDEsLlqOFykxOdHpNy0NmvVO31vcSqAxJCg=="],
|
||||||
|
|
||||||
"lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.30.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-c8JK7hyE65X1MHMN+Viq9n11RRC7hgin3HhYKhrMyaXflk5GVplZ60IxyoVtzILeKr+xAJwg6zK6sjTBJ0FKYQ=="],
|
"lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.30.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-c8JK7hyE65X1MHMN+Viq9n11RRC7hgin3HhYKhrMyaXflk5GVplZ60IxyoVtzILeKr+xAJwg6zK6sjTBJ0FKYQ=="],
|
||||||
@@ -836,6 +866,8 @@
|
|||||||
|
|
||||||
"nwsapi": ["nwsapi@2.2.21", "", {}, "sha512-o6nIY3qwiSXl7/LuOU0Dmuctd34Yay0yeuZRLFmDPrrdHpXKFndPj3hM+YEPVHYC5fx2otBx4Ilc/gyYSAUaIA=="],
|
"nwsapi": ["nwsapi@2.2.21", "", {}, "sha512-o6nIY3qwiSXl7/LuOU0Dmuctd34Yay0yeuZRLFmDPrrdHpXKFndPj3hM+YEPVHYC5fx2otBx4Ilc/gyYSAUaIA=="],
|
||||||
|
|
||||||
|
"pako": ["pako@2.1.0", "", {}, "sha512-w+eufiZ1WuJYgPXbV/PO3NCMEc3xqylkKHzp8bxp1uW4qaSNQUkwmLLEc3kKsfz8lpV1F8Ht3U1Cm+9Srog2ug=="],
|
||||||
|
|
||||||
"parse-entities": ["parse-entities@4.0.2", "", { "dependencies": { "@types/unist": "^2.0.0", "character-entities-legacy": "^3.0.0", "character-reference-invalid": "^2.0.0", "decode-named-character-reference": "^1.0.0", "is-alphanumerical": "^2.0.0", "is-decimal": "^2.0.0", "is-hexadecimal": "^2.0.0" } }, "sha512-GG2AQYWoLgL877gQIKeRPGO1xF9+eG1ujIb5soS5gPvLQ1y2o8FL90w2QWNdf9I361Mpp7726c+lj3U0qK1uGw=="],
|
"parse-entities": ["parse-entities@4.0.2", "", { "dependencies": { "@types/unist": "^2.0.0", "character-entities-legacy": "^3.0.0", "character-reference-invalid": "^2.0.0", "decode-named-character-reference": "^1.0.0", "is-alphanumerical": "^2.0.0", "is-decimal": "^2.0.0", "is-hexadecimal": "^2.0.0" } }, "sha512-GG2AQYWoLgL877gQIKeRPGO1xF9+eG1ujIb5soS5gPvLQ1y2o8FL90w2QWNdf9I361Mpp7726c+lj3U0qK1uGw=="],
|
||||||
|
|
||||||
"parse5": ["parse5@7.3.0", "", { "dependencies": { "entities": "^6.0.0" } }, "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw=="],
|
"parse5": ["parse5@7.3.0", "", { "dependencies": { "entities": "^6.0.0" } }, "sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw=="],
|
||||||
@@ -844,6 +876,8 @@
|
|||||||
|
|
||||||
"pathval": ["pathval@2.0.1", "", {}, "sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ=="],
|
"pathval": ["pathval@2.0.1", "", {}, "sha512-//nshmD55c46FuFw26xV/xFAaB5HF9Xdap7HJBBnrKdAd6/GxDBaNA1870O79+9ueg61cZLSVc+OaFlfmObYVQ=="],
|
||||||
|
|
||||||
|
"performance-now": ["performance-now@2.1.0", "", {}, "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow=="],
|
||||||
|
|
||||||
"picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="],
|
"picocolors": ["picocolors@1.1.1", "", {}, "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA=="],
|
||||||
|
|
||||||
"picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
"picomatch": ["picomatch@4.0.3", "", {}, "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q=="],
|
||||||
@@ -858,6 +892,8 @@
|
|||||||
|
|
||||||
"punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="],
|
"punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="],
|
||||||
|
|
||||||
|
"raf": ["raf@3.4.1", "", { "dependencies": { "performance-now": "^2.1.0" } }, "sha512-Sq4CW4QhwOHE8ucn6J34MqtZCeWFP2aQSmrlroYgqAV1PjStIhJXxYuTgUIfkEk7zTLjmIjLmU5q+fbD1NnOJA=="],
|
||||||
|
|
||||||
"react": ["react@19.1.1", "", {}, "sha512-w8nqGImo45dmMIfljjMwOGtbmC/mk4CMYhWIicdSflH91J9TyCyczcPFXJzrZ/ZXcgGRFeP6BU0BEJTw6tZdfQ=="],
|
"react": ["react@19.1.1", "", {}, "sha512-w8nqGImo45dmMIfljjMwOGtbmC/mk4CMYhWIicdSflH91J9TyCyczcPFXJzrZ/ZXcgGRFeP6BU0BEJTw6tZdfQ=="],
|
||||||
|
|
||||||
"react-dom": ["react-dom@19.1.1", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.1" } }, "sha512-Dlq/5LAZgF0Gaz6yiqZCf6VCcZs1ghAJyrsu84Q/GT0gV+mCxbfmKNoGRKBYMJ8IEdGPqu49YWXD02GCknEDkw=="],
|
"react-dom": ["react-dom@19.1.1", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.1" } }, "sha512-Dlq/5LAZgF0Gaz6yiqZCf6VCcZs1ghAJyrsu84Q/GT0gV+mCxbfmKNoGRKBYMJ8IEdGPqu49YWXD02GCknEDkw=="],
|
||||||
@@ -886,6 +922,8 @@
|
|||||||
|
|
||||||
"redux-thunk": ["redux-thunk@3.1.0", "", { "peerDependencies": { "redux": "^5.0.0" } }, "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw=="],
|
"redux-thunk": ["redux-thunk@3.1.0", "", { "peerDependencies": { "redux": "^5.0.0" } }, "sha512-NW2r5T6ksUKXCabzhL9z+h206HQw/NJkcLm1GPImRQ8IzfXwRGqjVhKJGauHirT0DAuyy6hjdnMZaRoAcy0Klw=="],
|
||||||
|
|
||||||
|
"regenerator-runtime": ["regenerator-runtime@0.13.11", "", {}, "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg=="],
|
||||||
|
|
||||||
"remark-parse": ["remark-parse@11.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "mdast-util-from-markdown": "^2.0.0", "micromark-util-types": "^2.0.0", "unified": "^11.0.0" } }, "sha512-FCxlKLNGknS5ba/1lmpYijMUzX2esxW5xQqjWxw2eHFfS2MSdaHVINFmhjo+qN1WhZhNimq0dZATN9pH0IDrpA=="],
|
"remark-parse": ["remark-parse@11.0.0", "", { "dependencies": { "@types/mdast": "^4.0.0", "mdast-util-from-markdown": "^2.0.0", "micromark-util-types": "^2.0.0", "unified": "^11.0.0" } }, "sha512-FCxlKLNGknS5ba/1lmpYijMUzX2esxW5xQqjWxw2eHFfS2MSdaHVINFmhjo+qN1WhZhNimq0dZATN9pH0IDrpA=="],
|
||||||
|
|
||||||
"remark-rehype": ["remark-rehype@11.1.2", "", { "dependencies": { "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "mdast-util-to-hast": "^13.0.0", "unified": "^11.0.0", "vfile": "^6.0.0" } }, "sha512-Dh7l57ianaEoIpzbp0PC9UKAdCSVklD8E5Rpw7ETfbTl3FqcOOgq5q2LVDhgGCkaBv7p24JXikPdvhhmHvKMsw=="],
|
"remark-rehype": ["remark-rehype@11.1.2", "", { "dependencies": { "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "mdast-util-to-hast": "^13.0.0", "unified": "^11.0.0", "vfile": "^6.0.0" } }, "sha512-Dh7l57ianaEoIpzbp0PC9UKAdCSVklD8E5Rpw7ETfbTl3FqcOOgq5q2LVDhgGCkaBv7p24JXikPdvhhmHvKMsw=="],
|
||||||
@@ -894,6 +932,8 @@
|
|||||||
|
|
||||||
"resolve-pkg-maps": ["resolve-pkg-maps@1.0.0", "", {}, "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw=="],
|
"resolve-pkg-maps": ["resolve-pkg-maps@1.0.0", "", {}, "sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw=="],
|
||||||
|
|
||||||
|
"rgbcolor": ["rgbcolor@1.0.1", "", {}, "sha512-9aZLIrhRaD97sgVhtJOW6ckOEh6/GnvQtdVNfdZ6s67+3/XwLS9lBcQYzEEhYVeUowN7pRzMLsyGhK2i/xvWbw=="],
|
||||||
|
|
||||||
"rollup": ["rollup@4.46.3", "", { "dependencies": { "@types/estree": "1.0.8" }, "optionalDependencies": { "@rollup/rollup-android-arm-eabi": "4.46.3", "@rollup/rollup-android-arm64": "4.46.3", "@rollup/rollup-darwin-arm64": "4.46.3", "@rollup/rollup-darwin-x64": "4.46.3", "@rollup/rollup-freebsd-arm64": "4.46.3", "@rollup/rollup-freebsd-x64": "4.46.3", "@rollup/rollup-linux-arm-gnueabihf": "4.46.3", "@rollup/rollup-linux-arm-musleabihf": "4.46.3", "@rollup/rollup-linux-arm64-gnu": "4.46.3", "@rollup/rollup-linux-arm64-musl": "4.46.3", "@rollup/rollup-linux-loongarch64-gnu": "4.46.3", "@rollup/rollup-linux-ppc64-gnu": "4.46.3", "@rollup/rollup-linux-riscv64-gnu": "4.46.3", "@rollup/rollup-linux-riscv64-musl": "4.46.3", "@rollup/rollup-linux-s390x-gnu": "4.46.3", "@rollup/rollup-linux-x64-gnu": "4.46.3", "@rollup/rollup-linux-x64-musl": "4.46.3", "@rollup/rollup-win32-arm64-msvc": "4.46.3", "@rollup/rollup-win32-ia32-msvc": "4.46.3", "@rollup/rollup-win32-x64-msvc": "4.46.3", "fsevents": "~2.3.2" }, "bin": { "rollup": "dist/bin/rollup" } }, "sha512-RZn2XTjXb8t5g13f5YclGoilU/kwT696DIkY3sywjdZidNSi3+vseaQov7D7BZXVJCPv3pDWUN69C78GGbXsKw=="],
|
"rollup": ["rollup@4.46.3", "", { "dependencies": { "@types/estree": "1.0.8" }, "optionalDependencies": { "@rollup/rollup-android-arm-eabi": "4.46.3", "@rollup/rollup-android-arm64": "4.46.3", "@rollup/rollup-darwin-arm64": "4.46.3", "@rollup/rollup-darwin-x64": "4.46.3", "@rollup/rollup-freebsd-arm64": "4.46.3", "@rollup/rollup-freebsd-x64": "4.46.3", "@rollup/rollup-linux-arm-gnueabihf": "4.46.3", "@rollup/rollup-linux-arm-musleabihf": "4.46.3", "@rollup/rollup-linux-arm64-gnu": "4.46.3", "@rollup/rollup-linux-arm64-musl": "4.46.3", "@rollup/rollup-linux-loongarch64-gnu": "4.46.3", "@rollup/rollup-linux-ppc64-gnu": "4.46.3", "@rollup/rollup-linux-riscv64-gnu": "4.46.3", "@rollup/rollup-linux-riscv64-musl": "4.46.3", "@rollup/rollup-linux-s390x-gnu": "4.46.3", "@rollup/rollup-linux-x64-gnu": "4.46.3", "@rollup/rollup-linux-x64-musl": "4.46.3", "@rollup/rollup-win32-arm64-msvc": "4.46.3", "@rollup/rollup-win32-ia32-msvc": "4.46.3", "@rollup/rollup-win32-x64-msvc": "4.46.3", "fsevents": "~2.3.2" }, "bin": { "rollup": "dist/bin/rollup" } }, "sha512-RZn2XTjXb8t5g13f5YclGoilU/kwT696DIkY3sywjdZidNSi3+vseaQov7D7BZXVJCPv3pDWUN69C78GGbXsKw=="],
|
||||||
|
|
||||||
"rrweb-cssom": ["rrweb-cssom@0.8.0", "", {}, "sha512-guoltQEx+9aMf2gDZ0s62EcV8lsXR+0w8915TC3ITdn2YueuNjdAYh/levpU9nFaoChh9RUS5ZdQMrKfVEN9tw=="],
|
"rrweb-cssom": ["rrweb-cssom@0.8.0", "", {}, "sha512-guoltQEx+9aMf2gDZ0s62EcV8lsXR+0w8915TC3ITdn2YueuNjdAYh/levpU9nFaoChh9RUS5ZdQMrKfVEN9tw=="],
|
||||||
@@ -924,6 +964,8 @@
|
|||||||
|
|
||||||
"stackback": ["stackback@0.0.2", "", {}, "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw=="],
|
"stackback": ["stackback@0.0.2", "", {}, "sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw=="],
|
||||||
|
|
||||||
|
"stackblur-canvas": ["stackblur-canvas@2.7.0", "", {}, "sha512-yf7OENo23AGJhBriGx0QivY5JP6Y1HbrrDI6WLt6C5auYZXlQrheoY8hD4ibekFKz1HOfE48Ww8kMWMnJD/zcQ=="],
|
||||||
|
|
||||||
"std-env": ["std-env@3.9.0", "", {}, "sha512-UGvjygr6F6tpH7o2qyqR6QYpwraIjKSdtzyBdyytFOHmPZY917kwdwLG0RbOjWOnKmnm3PeHjaoLLMie7kPLQw=="],
|
"std-env": ["std-env@3.9.0", "", {}, "sha512-UGvjygr6F6tpH7o2qyqR6QYpwraIjKSdtzyBdyytFOHmPZY917kwdwLG0RbOjWOnKmnm3PeHjaoLLMie7kPLQw=="],
|
||||||
|
|
||||||
"stringify-entities": ["stringify-entities@4.0.4", "", { "dependencies": { "character-entities-html4": "^2.0.0", "character-entities-legacy": "^3.0.0" } }, "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg=="],
|
"stringify-entities": ["stringify-entities@4.0.4", "", { "dependencies": { "character-entities-html4": "^2.0.0", "character-entities-legacy": "^3.0.0" } }, "sha512-IwfBptatlO+QCJUo19AqvrPNqlVMpW9YEL2LIVY+Rpv2qsjCGxaDLNRgeGsQWJhfItebuJhsGSLjaBbNSQ+ieg=="],
|
||||||
@@ -934,6 +976,8 @@
|
|||||||
|
|
||||||
"style-to-object": ["style-to-object@1.0.11", "", { "dependencies": { "inline-style-parser": "0.2.4" } }, "sha512-5A560JmXr7wDyGLK12Nq/EYS38VkGlglVzkis1JEdbGWSnbQIEhZzTJhzURXN5/8WwwFCs/f/VVcmkTppbXLow=="],
|
"style-to-object": ["style-to-object@1.0.11", "", { "dependencies": { "inline-style-parser": "0.2.4" } }, "sha512-5A560JmXr7wDyGLK12Nq/EYS38VkGlglVzkis1JEdbGWSnbQIEhZzTJhzURXN5/8WwwFCs/f/VVcmkTppbXLow=="],
|
||||||
|
|
||||||
|
"svg-pathdata": ["svg-pathdata@6.0.3", "", {}, "sha512-qsjeeq5YjBZ5eMdFuUa4ZosMLxgr5RZ+F+Y1OrDhuOCEInRMA3x74XdBtggJcj9kOeInz0WE+LgCPDkZFlBYJw=="],
|
||||||
|
|
||||||
"symbol-tree": ["symbol-tree@3.2.4", "", {}, "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw=="],
|
"symbol-tree": ["symbol-tree@3.2.4", "", {}, "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw=="],
|
||||||
|
|
||||||
"tailwind-merge": ["tailwind-merge@3.3.1", "", {}, "sha512-gBXpgUm/3rp1lMZZrM/w7D8GKqshif0zAymAhbCyIt8KMe+0v9DQ7cdYLR4FHH/cKpdTXb+A/tKKU3eolfsI+g=="],
|
"tailwind-merge": ["tailwind-merge@3.3.1", "", {}, "sha512-gBXpgUm/3rp1lMZZrM/w7D8GKqshif0zAymAhbCyIt8KMe+0v9DQ7cdYLR4FHH/cKpdTXb+A/tKKU3eolfsI+g=="],
|
||||||
@@ -944,6 +988,8 @@
|
|||||||
|
|
||||||
"tar": ["tar@7.4.3", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.0.1", "mkdirp": "^3.0.1", "yallist": "^5.0.0" } }, "sha512-5S7Va8hKfV7W5U6g3aYxXmlPoZVAwUMy9AOKyF2fVuZa2UD3qZjg578OrLRt8PcNN1PleVaL/5/yYATNL0ICUw=="],
|
"tar": ["tar@7.4.3", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.0.1", "mkdirp": "^3.0.1", "yallist": "^5.0.0" } }, "sha512-5S7Va8hKfV7W5U6g3aYxXmlPoZVAwUMy9AOKyF2fVuZa2UD3qZjg578OrLRt8PcNN1PleVaL/5/yYATNL0ICUw=="],
|
||||||
|
|
||||||
|
"text-segmentation": ["text-segmentation@1.0.3", "", { "dependencies": { "utrie": "^1.0.2" } }, "sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw=="],
|
||||||
|
|
||||||
"tiny-invariant": ["tiny-invariant@1.3.3", "", {}, "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg=="],
|
"tiny-invariant": ["tiny-invariant@1.3.3", "", {}, "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg=="],
|
||||||
|
|
||||||
"tiny-warning": ["tiny-warning@1.0.3", "", {}, "sha512-lBN9zLN/oAf68o3zNXYrdCt1kP8WsiGW8Oo2ka41b2IM5JL/S1CTyX1rW0mb/zSuJun0ZUrDxx4sqvYS2FWzPA=="],
|
"tiny-warning": ["tiny-warning@1.0.3", "", {}, "sha512-lBN9zLN/oAf68o3zNXYrdCt1kP8WsiGW8Oo2ka41b2IM5JL/S1CTyX1rW0mb/zSuJun0ZUrDxx4sqvYS2FWzPA=="],
|
||||||
@@ -1006,6 +1052,8 @@
|
|||||||
|
|
||||||
"use-sync-external-store": ["use-sync-external-store@1.5.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Rb46I4cGGVBmjamjphe8L/UnvJD+uPPtTkNvX5mZgqdbavhI4EbgIWJiIHXJ8bc/i9EQGPRh4DwEURJ552Do0A=="],
|
"use-sync-external-store": ["use-sync-external-store@1.5.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Rb46I4cGGVBmjamjphe8L/UnvJD+uPPtTkNvX5mZgqdbavhI4EbgIWJiIHXJ8bc/i9EQGPRh4DwEURJ552Do0A=="],
|
||||||
|
|
||||||
|
"utrie": ["utrie@1.0.2", "", { "dependencies": { "base64-arraybuffer": "^1.0.2" } }, "sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw=="],
|
||||||
|
|
||||||
"vaul": ["vaul@1.1.2", "", { "dependencies": { "@radix-ui/react-dialog": "^1.1.1" }, "peerDependencies": { "react": "^16.8 || ^17.0 || ^18.0 || ^19.0.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0.0 || ^19.0.0-rc" } }, "sha512-ZFkClGpWyI2WUQjdLJ/BaGuV6AVQiJ3uELGk3OYtP+B6yCO7Cmn9vPFXVJkRaGkOJu3m8bQMgtyzNHixULceQA=="],
|
"vaul": ["vaul@1.1.2", "", { "dependencies": { "@radix-ui/react-dialog": "^1.1.1" }, "peerDependencies": { "react": "^16.8 || ^17.0 || ^18.0 || ^19.0.0 || ^19.0.0-rc", "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0.0 || ^19.0.0-rc" } }, "sha512-ZFkClGpWyI2WUQjdLJ/BaGuV6AVQiJ3uELGk3OYtP+B6yCO7Cmn9vPFXVJkRaGkOJu3m8bQMgtyzNHixULceQA=="],
|
||||||
|
|
||||||
"vfile": ["vfile@6.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "vfile-message": "^4.0.0" } }, "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q=="],
|
"vfile": ["vfile@6.0.3", "", { "dependencies": { "@types/unist": "^3.0.0", "vfile-message": "^4.0.0" } }, "sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q=="],
|
||||||
|
|||||||
627
src/components/ai/AIChatModal.jsx
Normal file
627
src/components/ai/AIChatModal.jsx
Normal file
@@ -0,0 +1,627 @@
|
|||||||
|
import React, { useEffect, useRef, useState } from "react";
|
||||||
|
import { supabase } from "@/auth/supabase";
|
||||||
|
import ReactMarkdown from "react-markdown";
|
||||||
|
|
||||||
|
/* ---------- UI Mocks (sin cambios) ---------- */
|
||||||
|
const Paperclip = (props) => (
|
||||||
|
<svg {...props} xmlns="http://www.w3.org/2000/svg" width="24" height="24"
|
||||||
|
viewBox="0 0 24 24" fill="none" stroke="currentColor"
|
||||||
|
strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
||||||
|
<path d="m21.44 11.05-9.19 9.19a6 6 0 0 1-8.49-8.49l9.19-9.19a4 4 0 0 1 5.66 5.66l-9.2 9.19a2 2 0 0 1-2.83-2.83l8.49-8.48"/>
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
|
||||||
|
const Dialog = ({ open, onOpenChange, children }) =>
|
||||||
|
open ? <div className="fixed inset-0 bg-black/50 flex items-center justify-center z-50 p-4" onClick={onOpenChange}>{children}</div> : null;
|
||||||
|
const DialogContent = ({ className, children }) =>
|
||||||
|
<div className={`bg-white rounded-xl shadow-2xl transform transition-all max-w-6xl w-[95vw] h-[85vh] p-6 flex flex-col ${className}`}
|
||||||
|
onClick={(e) => e.stopPropagation()}>{children}</div>;
|
||||||
|
const DialogHeader = ({ children }) => <div className="pb-4 border-b border-gray-200">{children}</div>;
|
||||||
|
const DialogTitle = ({ className, children }) => <h2 className={`text-xl font-bold ${className}`}>{children}</h2>;
|
||||||
|
const Button = ({ onClick, disabled, className, variant, children }) => (
|
||||||
|
<button onClick={onClick} disabled={disabled}
|
||||||
|
className={`px-4 py-2 rounded-lg text-sm font-medium transition-colors
|
||||||
|
${variant === "outline"
|
||||||
|
? "bg-white border border-gray-300 text-gray-700 hover:bg-gray-50"
|
||||||
|
: "bg-blue-600 text-white hover:bg-blue-700"}
|
||||||
|
${disabled ? "opacity-50 cursor-not-allowed" : ""} ${className}`}>
|
||||||
|
{children}
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
const Card = ({ className, children }) => <div className={`bg-white rounded-2xl shadow-md ${className}`}>{children}</div>;
|
||||||
|
const CardContent = ({ className, children }) => <div className={`p-4 ${className}`}>{children}</div>;
|
||||||
|
const ScrollArea = ({ className, children }) => <div className={`overflow-y-auto ${className}`}>{children}</div>;
|
||||||
|
|
||||||
|
/* ------------- COMPONENT ------------- */
|
||||||
|
export default function AIChatModal({ open, onClose, context, onAccept, plan_format }) {
|
||||||
|
|
||||||
|
const [vectorStores, setVectorStores] = useState([]);
|
||||||
|
const [vectorFiles, setVectorFiles] = useState([]);
|
||||||
|
const [selectedVector, setSelectedVector] = useState(null);
|
||||||
|
const [selectedFiles, setSelectedFiles] = useState([]);
|
||||||
|
|
||||||
|
const [attachedFiles, setAttachedFiles] = useState([]);
|
||||||
|
const [attachedPreviews, setAttachedPreviews] = useState([]);
|
||||||
|
|
||||||
|
// chat
|
||||||
|
const [messages, setMessages] = useState([]);
|
||||||
|
const [input, setInput] = useState("");
|
||||||
|
|
||||||
|
// loading states
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
const [loadingFiles, setLoadingFiles] = useState(false);
|
||||||
|
const [loadingVectors, setLoadingVectors] = useState(false);
|
||||||
|
|
||||||
|
// conversation control
|
||||||
|
const [conversationId, setConversationId] = useState(null);
|
||||||
|
const [creatingConversation, setCreatingConversation] = useState(false);
|
||||||
|
|
||||||
|
const messagesEndRef = useRef(null);
|
||||||
|
const scrollToBottom = () => messagesEndRef.current?.scrollIntoView({ behavior: "smooth" });
|
||||||
|
useEffect(scrollToBottom, [messages]);
|
||||||
|
|
||||||
|
const normalizeInvokeResponse = (resp) => {
|
||||||
|
if (!resp) return null;
|
||||||
|
const raw = resp.data;
|
||||||
|
if (typeof raw === "string") {
|
||||||
|
try { return JSON.parse(raw); } catch (e) { console.warn("❗ No se pudo parsear resp.data:", raw); return null; }
|
||||||
|
}
|
||||||
|
if (typeof raw === "object" && raw !== null) return raw;
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Al abrir: reset o crear conversación
|
||||||
|
useEffect(() => {
|
||||||
|
if (!open) {
|
||||||
|
if (conversationId) {
|
||||||
|
deleteConversation(conversationId).catch((e) => console.error(e));
|
||||||
|
}
|
||||||
|
setMessages([]);
|
||||||
|
setInput("");
|
||||||
|
setSelectedFiles([]);
|
||||||
|
setAttachedFiles([]);
|
||||||
|
setAttachedPreviews([]);
|
||||||
|
setConversationId(null);
|
||||||
|
setSelectedVector(null);
|
||||||
|
setVectorFiles([]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (context) {
|
||||||
|
setMessages([
|
||||||
|
{
|
||||||
|
role: "system",
|
||||||
|
//content: `Contexto académico:\n${context.section || "—"}\n\nTexto original:\n${context.originalText || "—"}`
|
||||||
|
content: `Contexto académico:\n${context.section || "—"}\n\nTexto original:\n${context.originalText || "—"}`
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
await createConversation();
|
||||||
|
fetchVectorStores();
|
||||||
|
})();
|
||||||
|
|
||||||
|
}, [open]);
|
||||||
|
|
||||||
|
// ---------- CREATE CONVERSATION ----------
|
||||||
|
const createConversation = async () => {
|
||||||
|
try {
|
||||||
|
setCreatingConversation(true);
|
||||||
|
const { data: { session } } = await supabase.auth.getSession();
|
||||||
|
const token = session?.access_token;
|
||||||
|
|
||||||
|
const resp = await supabase.functions.invoke("conversation-format", {
|
||||||
|
headers: { Authorization: `Bearer ${token}` },
|
||||||
|
body: { action: "start", role: "system", content: context?.cont_conversation ?? "" }
|
||||||
|
});
|
||||||
|
|
||||||
|
let parsed = null;
|
||||||
|
if (typeof resp?.data === "string") {
|
||||||
|
try { parsed = JSON.parse(resp.data); } catch (e) { parsed = null; }
|
||||||
|
} else if (typeof resp?.data === "object" && resp.data !== null) parsed = resp.data;
|
||||||
|
else parsed = resp;
|
||||||
|
|
||||||
|
const convId =
|
||||||
|
parsed?.conversationId ||
|
||||||
|
parsed?.data?.conversationId ||
|
||||||
|
parsed?.data?.id ||
|
||||||
|
parsed?.id ||
|
||||||
|
parsed?.conversation_id ||
|
||||||
|
parsed?.data?.conversation_id;
|
||||||
|
|
||||||
|
if (!convId) { setCreatingConversation(false); return; }
|
||||||
|
setConversationId(convId);
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Error creando conversación:", err);
|
||||||
|
} finally {
|
||||||
|
setCreatingConversation(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// ---------- DELETE CONVERSATION ----------
|
||||||
|
const deleteConversation = async (convIdParam) => {
|
||||||
|
try {
|
||||||
|
const convIdToUse = convIdParam ?? conversationId;
|
||||||
|
if (!convIdToUse) return;
|
||||||
|
const { data: { session } } = await supabase.auth.getSession();
|
||||||
|
const token = session?.access_token;
|
||||||
|
|
||||||
|
await supabase.functions.invoke("conversation-format", {
|
||||||
|
headers: { Authorization: `Bearer ${token}` },
|
||||||
|
body: { action: "end", conversationId: convIdToUse }
|
||||||
|
});
|
||||||
|
|
||||||
|
setConversationId(null);
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Error eliminando conversación:", err);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// ---------- CONVERT FILE TO BASE64 ----------
|
||||||
|
const fileToBase64 = (file) =>
|
||||||
|
new Promise((resolve, reject) => {
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onerror = (e) => reject(e);
|
||||||
|
reader.onload = () => resolve(reader.result.split(",")[1]);
|
||||||
|
reader.readAsDataURL(file);
|
||||||
|
});
|
||||||
|
|
||||||
|
// ---------- HANDLE CONVERSATION (envío) ----------
|
||||||
|
const handleConversation = async ({ text }) => {
|
||||||
|
let contextText = "";
|
||||||
|
if (context?.originalText) contextText += `CONTEXTO DEL CAMPO:\n${context.originalText}\n`;
|
||||||
|
|
||||||
|
if (!conversationId) {
|
||||||
|
console.warn("No hay conversación activa todavía. conversationId:", conversationId);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
setLoading(true);
|
||||||
|
const { data: { session } } = await supabase.auth.getSession();
|
||||||
|
const token = session?.access_token;
|
||||||
|
|
||||||
|
// archivos adjuntos (locales) -> base64
|
||||||
|
let filesInput = [];
|
||||||
|
if (attachedFiles.length > 0) {
|
||||||
|
for (const file of attachedFiles) {
|
||||||
|
const base64 = await fileToBase64(file);
|
||||||
|
filesInput.push({
|
||||||
|
type: "input_file",
|
||||||
|
filename: file.name,
|
||||||
|
file_data: `data:${file.type};base64,${base64}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// archivos seleccionados del vector (por id)
|
||||||
|
if (selectedFiles.length > 0) {
|
||||||
|
const filesFromVectors = selectedFiles.map(f => ({
|
||||||
|
type: "input_file",
|
||||||
|
file_id: f.id
|
||||||
|
}));
|
||||||
|
filesInput = [...filesInput, ...filesFromVectors];
|
||||||
|
}
|
||||||
|
|
||||||
|
const promptFinal = `${contextText}\nPREGUNTA DEL USUARIO:\n${text}`;
|
||||||
|
const payload = {
|
||||||
|
action: "message",
|
||||||
|
format: plan_format,
|
||||||
|
conversationId,
|
||||||
|
vectorStoreId: selectedVector ?? null,
|
||||||
|
fileIds: selectedFiles.length ? selectedFiles.map(f => f.id) : [],
|
||||||
|
input: [
|
||||||
|
{
|
||||||
|
role: "user",
|
||||||
|
content: [
|
||||||
|
{ type: "input_text", text: promptFinal },
|
||||||
|
...filesInput
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
const { data: invokeData, error } = await supabase.functions.invoke(
|
||||||
|
"conversation-format",
|
||||||
|
{
|
||||||
|
headers: { Authorization: `Bearer ${token}` },
|
||||||
|
body: payload
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (error) throw error;
|
||||||
|
const parsed = normalizeInvokeResponse({ data: invokeData });
|
||||||
|
|
||||||
|
// Extraer texto del assistant (robusto)
|
||||||
|
let assistantText = null;
|
||||||
|
if (parsed?.data?.output_text) assistantText = parsed.data.output_text;
|
||||||
|
if (!assistantText && Array.isArray(parsed?.data?.output)) {
|
||||||
|
const msgBlock = parsed.data.output.find(o => o.type === "message");
|
||||||
|
if (msgBlock?.content?.[0]?.text) assistantText = msgBlock.content[0].text;
|
||||||
|
}
|
||||||
|
assistantText = assistantText || "Sin respuesta del modelo.";
|
||||||
|
|
||||||
|
setMessages(prev => [...prev, { role: "assistant", content: cleanAssistantResponse(assistantText) }]);
|
||||||
|
|
||||||
|
// limpiar attachments locales (pero mantener seleccionados del vector si quieres — aquí los limpiamos)
|
||||||
|
setAttachedFiles([]);
|
||||||
|
setAttachedPreviews([]);
|
||||||
|
// si quieres mantener los selectedFiles tras el envío, comenta la siguiente línea:
|
||||||
|
setSelectedFiles([]);
|
||||||
|
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Error en handleConversation:", err);
|
||||||
|
setMessages(prev => [...prev, { role: "assistant", content: "Ocurrió un error al procesar tu mensaje." }]);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// ---------- VECTORES ----------
|
||||||
|
const fetchVectorStores = async () => {
|
||||||
|
try {
|
||||||
|
setLoadingVectors(true);
|
||||||
|
const { data: { session } } = await supabase.auth.getSession();
|
||||||
|
const token = session?.access_token;
|
||||||
|
|
||||||
|
const { data, error } = await supabase.functions.invoke("files-and-vector-stores-api", {
|
||||||
|
headers: { Authorization: `Bearer ${token}` },
|
||||||
|
body: { module: "vectorStores", action: "list" }
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error) throw error;
|
||||||
|
setVectorStores(Array.isArray(data) ? data : (data?.data ?? []));
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Error loading vector stores:", err);
|
||||||
|
setVectorStores([]);
|
||||||
|
} finally {
|
||||||
|
setLoadingVectors(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (open) fetchVectorStores();
|
||||||
|
}, [open]);
|
||||||
|
|
||||||
|
const loadFilesForVector = async (vectorStoreId) => {
|
||||||
|
try {
|
||||||
|
setLoadingFiles(true);
|
||||||
|
const { data: { session } } = await supabase.auth.getSession();
|
||||||
|
const token = session?.access_token;
|
||||||
|
|
||||||
|
const { data, error } = await supabase.functions.invoke("files-and-vector-stores-api", {
|
||||||
|
headers: { Authorization: `Bearer ${token}` },
|
||||||
|
body: { module: "vectorStoreFiles", action: "list", params: { vector_store_id: vectorStoreId } }
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error) throw error;
|
||||||
|
setVectorFiles(Array.isArray(data) ? data : (data?.data ?? []));
|
||||||
|
} catch (err) {
|
||||||
|
console.error("Error loading vector files:", err);
|
||||||
|
setVectorFiles([]);
|
||||||
|
} finally {
|
||||||
|
setLoadingFiles(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// ---------- UI helpers ----------
|
||||||
|
const handleAttach = (e) => {
|
||||||
|
const files = Array.from(e.target.files);
|
||||||
|
if (!files.length) return;
|
||||||
|
setAttachedFiles(prev => [...prev, ...files]);
|
||||||
|
setAttachedPreviews(prev => [...prev, ...files.map(f => f.name)]);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Al hacer click en un vector: expandir (solo uno a la vez) y cargar sus archivos
|
||||||
|
const handleVectorClick = async (vector) => {
|
||||||
|
if (selectedVector === vector.id) {
|
||||||
|
// colapsar
|
||||||
|
setSelectedVector(null);
|
||||||
|
setVectorFiles([]);
|
||||||
|
setSelectedFiles([]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setSelectedVector(vector.id);
|
||||||
|
setSelectedFiles([]);
|
||||||
|
await loadFilesForVector(vector.id);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Toggle selección de archivo (checkbox)
|
||||||
|
const toggleFileSelection = (file) => {
|
||||||
|
if (selectedFiles.some(f => f.id === file.id)) {
|
||||||
|
setSelectedFiles(prev => prev.filter(f => f.id !== file.id));
|
||||||
|
} else {
|
||||||
|
setSelectedFiles(prev => [...prev, file]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const removeSelectedFile = (fileId) => {
|
||||||
|
setSelectedFiles(prev => prev.filter(f => f.id !== fileId));
|
||||||
|
};
|
||||||
|
|
||||||
|
// ---------- Send flow ----------
|
||||||
|
const handleSend = async () => {
|
||||||
|
// no permitir enviar si no hay nada
|
||||||
|
if (!input.trim() && attachedFiles.length === 0 && selectedFiles.length === 0) return;
|
||||||
|
|
||||||
|
if (creatingConversation) {
|
||||||
|
// no bloqueo visible aquí por diseño; simplemente ignoramos el envío si aún creando
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!conversationId) {
|
||||||
|
await createConversation();
|
||||||
|
if (!conversationId) {
|
||||||
|
setMessages(prev => [...prev, { role: "assistant", content: "No se pudo crear la conversación. Intenta de nuevo." }]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const userText = input.trim() || (selectedFiles.length ? `Consultar ${selectedFiles.length} archivo(s) del repositorio` : "");
|
||||||
|
setMessages(prev => [...prev, { role: "user", content: userText }]);
|
||||||
|
setInput("");
|
||||||
|
|
||||||
|
await handleConversation({ text: userText });
|
||||||
|
};
|
||||||
|
|
||||||
|
function cleanAIResponse(text) {
|
||||||
|
if (!text) return text;
|
||||||
|
|
||||||
|
let cleaned = text;
|
||||||
|
|
||||||
|
// -------------------------
|
||||||
|
// 1. Eliminar emojis
|
||||||
|
// -------------------------
|
||||||
|
cleaned = cleaned.replace(/[\p{Emoji}\uFE0F]/gu, "");
|
||||||
|
|
||||||
|
// -------------------------
|
||||||
|
// 2. Eliminar separadores tipo ---
|
||||||
|
// -------------------------
|
||||||
|
cleaned = cleaned.replace(/^---+$/gm, "");
|
||||||
|
|
||||||
|
// -------------------------
|
||||||
|
// 3. Eliminar saludos y frases meta
|
||||||
|
// -------------------------
|
||||||
|
const metaPatterns = [
|
||||||
|
/^hola[!¡., ]*/i,
|
||||||
|
/^buen(os|as) (días|tardes|noches)[!¡., ]*/i,
|
||||||
|
/estoy aquí para ayudarte[.! ]*/gi,
|
||||||
|
/aquí tienes[,:]*/gi,
|
||||||
|
/claro[,:]*/gi,
|
||||||
|
/como pediste[,:]*/gi,
|
||||||
|
/como solicitaste[,:]*/gi,
|
||||||
|
/el texto íntegro que compartiste.*$/gi,
|
||||||
|
/te lo dejo a continuación.*$/gi,
|
||||||
|
/¿te gustaría.*$/gi,
|
||||||
|
/¿en qué más puedo.*$/gi,
|
||||||
|
/si necesitas algo más.*$/gi,
|
||||||
|
/con gusto.*$/gi,
|
||||||
|
];
|
||||||
|
|
||||||
|
metaPatterns.forEach(p => {
|
||||||
|
cleaned = cleaned.replace(p, "").trim();
|
||||||
|
});
|
||||||
|
|
||||||
|
// -------------------------
|
||||||
|
// 4. Extraer solo contenido útil
|
||||||
|
// -------------------------
|
||||||
|
const startMarker = "CONTEXTO DEL CAMPO";
|
||||||
|
const startIndex = cleaned.indexOf(startMarker);
|
||||||
|
|
||||||
|
if (startIndex !== -1) {
|
||||||
|
cleaned = cleaned.substring(startIndex).trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
// -------------------------
|
||||||
|
// 5. Eliminar líneas vacías múltiples
|
||||||
|
// -------------------------
|
||||||
|
cleaned = cleaned.replace(/\n{2,}/g, "\n\n");
|
||||||
|
|
||||||
|
// -------------------------
|
||||||
|
// 6. Quitar numeraciones de cortesía (opcional)
|
||||||
|
// Ejemplo: “1. ” al inicio de líneas
|
||||||
|
// -------------------------
|
||||||
|
cleaned = cleaned.replace(/^\s*\d+\.\s+/gm, "");
|
||||||
|
|
||||||
|
return cleaned.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const handleApply = () => {
|
||||||
|
const last = [...messages].reverse().find(m => m.role === "assistant");
|
||||||
|
|
||||||
|
if (last && onAccept) {
|
||||||
|
const cleaned = cleanAIResponse(last.content);
|
||||||
|
onAccept(cleaned);
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
const cleanAssistantResponse = (text) => {
|
||||||
|
if (!text) return text;
|
||||||
|
const patterns = [/^claro[, ]*/i, /^por supuesto[, ]*/i, /^aquí tienes[, ]*/i, /^con gusto[, ]*/i, /^hola[, ]*/i, /^perfecto[, ]*/i, /^entendido[, ]*/i, /^muy bien[, ]*/i, /^ok[, ]*/i];
|
||||||
|
let cleaned = text.trim();
|
||||||
|
for (const p of patterns) cleaned = cleaned.replace(p, "").trim();
|
||||||
|
return cleaned;
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={open} onOpenChange={onClose} >
|
||||||
|
<DialogContent className="max-w-6xl w-[95vw] h-[85vh] p-6 flex flex-col relative"
|
||||||
|
|
||||||
|
>
|
||||||
|
<button onClick={onClose} className="absolute top-3 right-3 text-gray-400 hover:text-gray-600 transition z-50">✕</button>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>Asistente Inteligente</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
<div className="flex-1 pt-4 min-h-0">
|
||||||
|
<div className="flex gap-6 h-full min-h-0">
|
||||||
|
{/* Left: vectors */}
|
||||||
|
<Card className="w-1/3 min-w-[250px] max-w-sm flex flex-col bg-muted/20 border border-gray-200 rounded-2xl">
|
||||||
|
<CardContent className="flex flex-col flex-1 p-4">
|
||||||
|
<h3 className="font-semibold text-sm mb-3">Repositorio de archivos</h3>
|
||||||
|
<ScrollArea className="flex-1">
|
||||||
|
{loadingVectors ? (
|
||||||
|
<p className="text-gray-500 text-sm text-center mt-10">Cargando Repositorio de archivos...</p>
|
||||||
|
) : vectorStores.length === 0 ? (
|
||||||
|
<p className="text-gray-500 text-sm text-center mt-10">No hay Repositorio de archivos.</p>
|
||||||
|
) : (
|
||||||
|
<div className="space-y-3">
|
||||||
|
{vectorStores.map((vector) => (
|
||||||
|
<div key={vector.id}>
|
||||||
|
{/* VECTOR */}
|
||||||
|
<div
|
||||||
|
onClick={() => handleVectorClick(vector)}
|
||||||
|
className={`p-3 rounded-lg border cursor-pointer transition flex items-center justify-between
|
||||||
|
${selectedVector === vector.id ? "bg-blue-50 border-blue-400 shadow" : "bg-white border-gray-300"}`}
|
||||||
|
>
|
||||||
|
<div className="truncate">
|
||||||
|
<strong className="block truncate">{vector.name || vector.id}</strong>
|
||||||
|
<p className="text-xs text-gray-400 truncate">{vector.description || ""}</p>
|
||||||
|
</div>
|
||||||
|
<div className="text-xs text-gray-500">{selectedVector === vector.id ? "▼" : "▶"}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* ARCHIVOS cuando está expandido */}
|
||||||
|
{selectedVector === vector.id && (
|
||||||
|
<div className="ml-4 mt-2 mb-2 space-y-2">
|
||||||
|
{loadingFiles ? (
|
||||||
|
<p className="text-gray-400 text-sm">Cargando archivos...</p>
|
||||||
|
) : vectorFiles.length === 0 ? (
|
||||||
|
<p className="text-gray-400 text-sm">No hay archivos en este repositorio</p>
|
||||||
|
) : (
|
||||||
|
vectorFiles.map((file) => (
|
||||||
|
<label key={file.id} className="flex items-center gap-2 p-2 rounded-md hover:bg-gray-50 cursor-pointer">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={selectedFiles.some(f => f.id === file.id)}
|
||||||
|
onChange={() => toggleFileSelection(file)}
|
||||||
|
/>
|
||||||
|
<div className="text-sm">
|
||||||
|
<div className="font-medium">{file.filename ?? file.name ?? file.id}</div>
|
||||||
|
<div className="text-xs text-gray-400">{file.id}</div>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
</ScrollArea>
|
||||||
|
|
||||||
|
{/* Resumen de archivos seleccionados (de vectores) */}
|
||||||
|
<div className="mt-4">
|
||||||
|
<h4 className="font-semibold text-sm mb-2">Archivos seleccionados</h4>
|
||||||
|
{selectedFiles.length === 0 ? (
|
||||||
|
<p className="text-sm text-gray-500">No has seleccionado archivos del repositorio</p>
|
||||||
|
) : (
|
||||||
|
<ul className="space-y-2 max-h-40 overflow-auto">
|
||||||
|
{selectedFiles.map((f) => (
|
||||||
|
<li key={f.id} className="flex items-center justify-between p-2 rounded-md border bg-white">
|
||||||
|
<div className="text-sm">
|
||||||
|
<div className="font-medium">{f.filename ?? f.name ?? f.id}</div>
|
||||||
|
<div className="text-xs text-gray-400 truncate">{f.id}</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="text-xs text-gray-500">{/* optionally show vector id */}</span>
|
||||||
|
<button onClick={() => removeSelectedFile(f.id)} className="text-sm text-red-500 hover:underline">Quitar</button>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* <div className="mt-4 flex-shrink-0">
|
||||||
|
<Button variant="outline" className="w-full" onClick={() => alert("Funcionalidad Subir a vector store no implementada aquí")}>Subir archivo (vector)</Button>
|
||||||
|
</div> */}
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
{/* Right: Chat */}
|
||||||
|
<Card className="flex-1 flex flex-col min-w-[350px] bg-background border border-gray-200 rounded-2xl">
|
||||||
|
<CardContent className="flex flex-col flex-1 p-4 min-h-0">
|
||||||
|
<h3 className="font-semibold text-sm mb-3 flex-shrink-0">Chat con IA</h3>
|
||||||
|
|
||||||
|
<div className="flex-1 flex flex-col min-h-0">
|
||||||
|
{/* CONTENEDOR SCROLL DE LOS MENSAJES */}
|
||||||
|
<div className="flex-1 overflow-y-auto min-h-0 border border-gray-200 rounded-lg p-3 space-y-3 bg-gray-50 break-words whitespace-pre-wrap">
|
||||||
|
{messages.length === 0 ? (
|
||||||
|
<p className="text-gray-400 text-sm text-center mt-10">Inicia una conversación...</p>
|
||||||
|
) : (
|
||||||
|
messages.map((m, i) => (
|
||||||
|
<div key={i} className={`break-words whitespace-pre-wrap p-3 rounded-xl shadow-sm max-w-[85%] ${m.role === "user" ? "bg-blue-50 text-blue-800 ml-auto" : m.role === "assistant" ? "bg-white text-gray-800 mr-auto border border-gray-200" : "bg-gray-100 text-gray-700 mr-auto"}`}>
|
||||||
|
<strong className="font-bold">{m.role === "user" ? "Tú:" : m.role === "assistant" ? "IA:" : "Sistema:"}</strong>{" "}
|
||||||
|
<ReactMarkdown>{m.content}</ReactMarkdown>
|
||||||
|
</div>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
|
||||||
|
{loading && (
|
||||||
|
<div className="flex items-center space-x-2 p-3 bg-white border border-gray-200 rounded-xl mr-auto max-w-fit shadow-sm flex-shrink-0">
|
||||||
|
<svg className="animate-spin h-4 w-4 text-blue-500" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24">
|
||||||
|
<circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4"></circle>
|
||||||
|
<path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||||||
|
</svg>
|
||||||
|
<span className="text-sm text-gray-600">La IA está respondiendo...</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div ref={messagesEndRef} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{attachedPreviews.length > 0 && (
|
||||||
|
<ul className="text-xs text-gray-600 mt-2">
|
||||||
|
{attachedPreviews.map((name, i) => (
|
||||||
|
<li key={i}>📄 {name}</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className="flex gap-2 mt-4 items-end flex-shrink-0">
|
||||||
|
<label className="cursor-pointer text-gray-600 hover:text-blue-600 self-center">
|
||||||
|
<Paperclip className="w-5 h-5" />
|
||||||
|
<input type="file" accept=".pdf,.txt,.doc,.docx" multiple className="hidden" onChange={handleAttach} />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<textarea
|
||||||
|
value={input}
|
||||||
|
onChange={(e) => setInput(e.target.value)}
|
||||||
|
placeholder="Escribe tu pregunta..."
|
||||||
|
onKeyDown={(e) => {
|
||||||
|
if (e.key === "Enter" && !e.shiftKey) {
|
||||||
|
e.preventDefault();
|
||||||
|
handleSend();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
rows={1}
|
||||||
|
className="flex-1 resize-none rounded-xl border border-gray-300 p-3 text-sm focus:outline-none focus:ring-2 focus:ring-blue-500 max-h-32 overflow-y-auto bg-white shadow-inner"
|
||||||
|
style={{ minHeight: "38px" }}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button onClick={handleSend} disabled={loading || creatingConversation || (!input.trim() && attachedFiles.length === 0 && selectedFiles.length === 0)} className="shadow-md">
|
||||||
|
{creatingConversation ? "Preparando..." : loading ? "Enviando..." : "Enviar"}
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<Button onClick={handleApply} disabled={!messages.some((m) => m.role === "assistant")} className="shadow-md">
|
||||||
|
Aplicar mejora
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
91
src/components/historico/HistorialCambiosModal.tsx
Normal file
91
src/components/historico/HistorialCambiosModal.tsx
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
import { useQuery } from "@tanstack/react-query"
|
||||||
|
import { Dialog, DialogContent, DialogHeader, DialogTitle } from "@/components/ui/dialog"
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
|
import { supabase } from "@/auth/supabase"
|
||||||
|
import { useSupabaseAuth } from "@/auth/supabase"
|
||||||
|
|
||||||
|
export function HistorialCambiosModal({
|
||||||
|
open,
|
||||||
|
onClose,
|
||||||
|
planId,
|
||||||
|
onRestore, // 🔥 recibiremos una función del padre para restaurar
|
||||||
|
}: {
|
||||||
|
open: boolean
|
||||||
|
onClose: () => void
|
||||||
|
planId: string
|
||||||
|
onRestore: (key: string, value: string) => void
|
||||||
|
}) {
|
||||||
|
const auth = useSupabaseAuth()
|
||||||
|
|
||||||
|
const { data, isLoading, error } = useQuery({
|
||||||
|
queryKey: ["historico_cambios", planId, auth.user?.id],
|
||||||
|
queryFn: async () => {
|
||||||
|
const { data, error } = await supabase
|
||||||
|
.from("historico_cambios")
|
||||||
|
.select("id, json_cambios, user_id, created_at")
|
||||||
|
.eq("id_plan_estudios", planId)
|
||||||
|
.eq("user_id", auth.user?.id) // ✅ filtro por usuario actual
|
||||||
|
.order("created_at", { ascending: false })
|
||||||
|
|
||||||
|
if (error) throw error
|
||||||
|
return data
|
||||||
|
},
|
||||||
|
enabled: !!auth.user?.id, // ✅ solo corre si hay usuario autenticado
|
||||||
|
})
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={open} onOpenChange={onClose}>
|
||||||
|
<DialogContent className="max-w-3xl">
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>Histórico de cambios</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
{isLoading && <p className="text-sm text-gray-500">Cargando historial...</p>}
|
||||||
|
{error && <p className="text-red-500 text-sm">Error al cargar: {String(error)}</p>}
|
||||||
|
{!isLoading && !error && (!data || data.length === 0) && (
|
||||||
|
<p className="text-gray-500 text-sm">No hay cambios registrados.</p>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className="space-y-4 max-h-[60vh] overflow-y-auto">
|
||||||
|
{data?.map((item) => {
|
||||||
|
const diff = item.json_cambios?.[0]
|
||||||
|
const key = diff?.path?.replace("/", "")
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={item.id}
|
||||||
|
className="rounded-lg border p-3 bg-white/70 dark:bg-neutral-900/50"
|
||||||
|
>
|
||||||
|
<div className="flex justify-between text-xs text-neutral-500 mb-2">
|
||||||
|
<span>Usuario: {item.user_id || "Desconocido"}</span>
|
||||||
|
<span>{new Date(item.created_at).toLocaleString()}</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="text-xs text-gray-700 font-mono whitespace-pre-wrap">
|
||||||
|
<p><strong>Campo:</strong> {key}</p>
|
||||||
|
<p><strong>Antes:</strong> {diff?.from || "—"}</p>
|
||||||
|
<p><strong>Después:</strong> {diff?.value || "—"}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="outline"
|
||||||
|
className="mt-2"
|
||||||
|
onClick={() => onRestore(key, diff.from)}
|
||||||
|
>
|
||||||
|
Restaurar
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="mt-4 text-right">
|
||||||
|
<Button variant="outline" onClick={onClose}>
|
||||||
|
Cerrar
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
)
|
||||||
|
|
||||||
|
}
|
||||||
@@ -6,15 +6,12 @@ import { Input } from "@/components/ui/input"
|
|||||||
import { Textarea } from "@/components/ui/textarea"
|
import { Textarea } from "@/components/ui/textarea"
|
||||||
import { AuroraButton } from "@/components/effect/aurora-button"
|
import { AuroraButton } from "@/components/effect/aurora-button"
|
||||||
import confetti from "canvas-confetti"
|
import confetti from "canvas-confetti"
|
||||||
import { useQueryClient } from "@tanstack/react-query"
|
|
||||||
import { supabase, useSupabaseAuth } from "@/auth/supabase"
|
import { supabase, useSupabaseAuth } from "@/auth/supabase"
|
||||||
import { Field } from "./Field"
|
import { Field } from "./Field"
|
||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from "../ui/tabs"
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from "../ui/tabs"
|
||||||
import { asignaturaKeys } from "./planQueries"
|
|
||||||
import { useRouter } from "@tanstack/react-router"
|
import { useRouter } from "@tanstack/react-router"
|
||||||
|
|
||||||
export function AddAsignaturaButton({ planId, onAdded }: { planId: string; onAdded?: () => void }) {
|
export function AddAsignaturaButton({ planId, onAdded }: { planId: string; onAdded?: () => void }) {
|
||||||
const qc = useQueryClient()
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const supabaseAuth = useSupabaseAuth()
|
const supabaseAuth = useSupabaseAuth()
|
||||||
const [open, setOpen] = useState(false)
|
const [open, setOpen] = useState(false)
|
||||||
|
|||||||
@@ -104,7 +104,7 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
|||||||
setDbFiles((data || []).map((file: any) => ({
|
setDbFiles((data || []).map((file: any) => ({
|
||||||
id: file.documentos_id,
|
id: file.documentos_id,
|
||||||
titulo: file.titulo_archivo,
|
titulo: file.titulo_archivo,
|
||||||
s3_file_path: file.titulo_archivo,
|
s3_file_path: `prueba-referencias/documento_${file.documentos_id}.pdf`,
|
||||||
fecha_subida: file.fecha_subida,
|
fecha_subida: file.fecha_subida,
|
||||||
tags: file.tags || [],
|
tags: file.tags || [],
|
||||||
})));
|
})));
|
||||||
@@ -118,35 +118,35 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
|||||||
|
|
||||||
const isSelected = useCallback((path: string) => selectedFiles.includes(path), [selectedFiles]);
|
const isSelected = useCallback((path: string) => selectedFiles.includes(path), [selectedFiles]);
|
||||||
|
|
||||||
const toggleSelected = useCallback((path: string) => {
|
const toggleSelected = useCallback((id: string) => {
|
||||||
setSelectedFiles(prev => prev.includes(path) ? prev.filter(p => p !== path) : [...prev, path]);
|
setSelectedFiles(prev => prev.includes(id) ? prev.filter(p => p !== id) : [...prev, id]);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const replaceSelection = useCallback((path: string) => {
|
const replaceSelection = useCallback((id: string) => {
|
||||||
setSelectedFiles([path]);
|
setSelectedFiles([id]);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const rangeSelect = useCallback((start: number, end: number) => {
|
const rangeSelect = useCallback((start: number, end: number) => {
|
||||||
const [s, e] = start < end ? [start, end] : [end, start];
|
const [s, e] = start < end ? [start, end] : [end, start];
|
||||||
const paths = dbFiles.slice(s, e + 1).map(f => f.s3_file_path);
|
const ids = dbFiles.slice(s, e + 1).map(f => f.id);
|
||||||
setSelectedFiles(prev => Array.from(new Set([...prev, ...paths])));
|
setSelectedFiles(prev => Array.from(new Set([...prev, ...ids])));
|
||||||
}, [dbFiles]);
|
}, [dbFiles]);
|
||||||
|
|
||||||
const handleCardClick = useCallback((e: React.MouseEvent, index: number, file: { s3_file_path: string }) => {
|
const handleCardClick = useCallback((e: React.MouseEvent, index: number, file: { id: string }) => {
|
||||||
const path = file.s3_file_path;
|
const id = file.id;
|
||||||
|
|
||||||
if (e.shiftKey && lastSelectedIndex !== null) {
|
if (e.shiftKey && lastSelectedIndex !== null) {
|
||||||
rangeSelect(lastSelectedIndex, index);
|
rangeSelect(lastSelectedIndex, index);
|
||||||
} else if (e.metaKey || e.ctrlKey) {
|
} else if (e.metaKey || e.ctrlKey) {
|
||||||
toggleSelected(path);
|
toggleSelected(id);
|
||||||
setLastSelectedIndex(index);
|
setLastSelectedIndex(index);
|
||||||
} else {
|
} else {
|
||||||
if (isSelected(path) && selectedFiles.length === 1) {
|
if (isSelected(id) && selectedFiles.length === 1) {
|
||||||
// si ya es el único seleccionado, des-selecciona
|
// si ya es el único seleccionado, des-selecciona
|
||||||
setSelectedFiles([]);
|
setSelectedFiles([]);
|
||||||
setLastSelectedIndex(null);
|
setLastSelectedIndex(null);
|
||||||
} else {
|
} else {
|
||||||
replaceSelection(path);
|
replaceSelection(id);
|
||||||
setLastSelectedIndex(index);
|
setLastSelectedIndex(index);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -167,7 +167,7 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
|||||||
prompt: prompt,
|
prompt: prompt,
|
||||||
insert: true,
|
insert: true,
|
||||||
files: selectedFiles,
|
files: selectedFiles,
|
||||||
uuid: auth.user?.id,
|
created_by: auth.user?.id,
|
||||||
})
|
})
|
||||||
const newId = (res as any)?.id || (res as any)?.plan?.id || (res as any)?.data?.id
|
const newId = (res as any)?.id || (res as any)?.plan?.id || (res as any)?.data?.id
|
||||||
if (newId) {
|
if (newId) {
|
||||||
@@ -261,7 +261,7 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
|||||||
<div role="grid" className="grid gap-4 xs:grid-cols-2 sm:grid-cols-2 md:grid-cols-3 xl:grid-cols-4 2xl:grid-cols-5">
|
<div role="grid" className="grid gap-4 xs:grid-cols-2 sm:grid-cols-2 md:grid-cols-3 xl:grid-cols-4 2xl:grid-cols-5">
|
||||||
{dbFiles.map((file, index) => {
|
{dbFiles.map((file, index) => {
|
||||||
const ext = fileExt(file.titulo);
|
const ext = fileExt(file.titulo);
|
||||||
const selected = isSelected(file.s3_file_path);
|
const selected = isSelected(file.id);
|
||||||
console.log(file);
|
console.log(file);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -21,7 +21,8 @@ export function DownloadPlanPDF({ plan }: { plan: PlanLike }) {
|
|||||||
unit: "mm",
|
unit: "mm",
|
||||||
format: "letter",
|
format: "letter",
|
||||||
})
|
})
|
||||||
|
console.log(plan);
|
||||||
|
|
||||||
const pageWidth = doc.internal.pageSize.getWidth()
|
const pageWidth = doc.internal.pageSize.getWidth()
|
||||||
const pageHeight = doc.internal.pageSize.getHeight()
|
const pageHeight = doc.internal.pageSize.getHeight()
|
||||||
const margin = 20
|
const margin = 20
|
||||||
@@ -32,7 +33,6 @@ export function DownloadPlanPDF({ plan }: { plan: PlanLike }) {
|
|||||||
const sectionGap = 10 // Espacio entre recuadros de sección
|
const sectionGap = 10 // Espacio entre recuadros de sección
|
||||||
const bodyFontSize = 10.5
|
const bodyFontSize = 10.5
|
||||||
const headingFontSize = 12
|
const headingFontSize = 12
|
||||||
const subHeadingFontSize = 10
|
|
||||||
const bulletGlifo = "\u21D2" // Flecha doble (⇒) para el glifo
|
const bulletGlifo = "\u21D2" // Flecha doble (⇒) para el glifo
|
||||||
const bulletIndent = 6 // Sangría para el texto de la lista
|
const bulletIndent = 6 // Sangría para el texto de la lista
|
||||||
|
|
||||||
@@ -229,7 +229,7 @@ export function DownloadPlanPDF({ plan }: { plan: PlanLike }) {
|
|||||||
// LICENCIATURA EN INGENIERÍA CIBERNÉTICA Y SISTEMAS COMPUTACIONALES
|
// LICENCIATURA EN INGENIERÍA CIBERNÉTICA Y SISTEMAS COMPUTACIONALES
|
||||||
doc.setFontSize(18)
|
doc.setFontSize(18)
|
||||||
// Manejamos la conversión a string si es necesario
|
// Manejamos la conversión a string si es necesario
|
||||||
const mainTitle = (plan["titulo"] !== null && plan["titulo"] !== undefined ? String(plan["titulo"]) : "LICENCIATURA EN INGENIERÍA CIBERNÉTICA Y SISTEMAS COMPUTACIONALES").toUpperCase()
|
const mainTitle = (plan["nombre"] !== null && plan["nombre"] !== undefined ? String(plan["nombre"]) : "LICENCIATURA EN INGENIERÍA CIBERNÉTICA Y SISTEMAS COMPUTACIONALES").toUpperCase()
|
||||||
const mainTitleLines = doc.splitTextToSize(mainTitle, maxWidth - 20)
|
const mainTitleLines = doc.splitTextToSize(mainTitle, maxWidth - 20)
|
||||||
doc.text(mainTitleLines, pageWidth / 2, cursorY, { align: "center" })
|
doc.text(mainTitleLines, pageWidth / 2, cursorY, { align: "center" })
|
||||||
cursorY += mainTitleLines.length * 8
|
cursorY += mainTitleLines.length * 8
|
||||||
|
|||||||
@@ -7,6 +7,10 @@ import { Textarea } from "@/components/ui/textarea"
|
|||||||
import { supabase,useSupabaseAuth } from "@/auth/supabase"
|
import { supabase,useSupabaseAuth } from "@/auth/supabase"
|
||||||
import { toast } from "sonner"
|
import { toast } from "sonner"
|
||||||
import ReactMarkdown from 'react-markdown'
|
import ReactMarkdown from 'react-markdown'
|
||||||
|
import { HistorialCambiosModal } from "../historico/HistorialCambiosModal"
|
||||||
|
// @ts-ignore
|
||||||
|
import AIChatModal from "../ai/AIChatModal"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* =====================================================
|
/* =====================================================
|
||||||
@@ -27,6 +31,7 @@ export type PlanTextFields = {
|
|||||||
indicadores_desempeno?: string | string[] | null
|
indicadores_desempeno?: string | string[] | null
|
||||||
pertinencia?: string | string[] | null
|
pertinencia?: string | string[] | null
|
||||||
prompt?: string | null
|
prompt?: string | null
|
||||||
|
historico?: string | null
|
||||||
}
|
}
|
||||||
|
|
||||||
async function fetchPlanText(planId: string): Promise<PlanTextFields> {
|
async function fetchPlanText(planId: string): Promise<PlanTextFields> {
|
||||||
@@ -50,6 +55,8 @@ export const planTextOptions = (planId: string) =>
|
|||||||
staleTime: 60_000,
|
staleTime: 60_000,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* =====================================================
|
/* =====================================================
|
||||||
Color helpers
|
Color helpers
|
||||||
===================================================== */
|
===================================================== */
|
||||||
@@ -65,7 +72,7 @@ const rgba = (rgb: [number, number, number], a: number) => `rgba(${rgb[0]},${rgb
|
|||||||
/* =====================================================
|
/* =====================================================
|
||||||
Expandable text
|
Expandable text
|
||||||
===================================================== */
|
===================================================== */
|
||||||
function ExpandableText({ text, mono = false }: { text?: string | string[] | null; mono?: boolean }) {
|
function ExpandableText({ text }: { text?: string | string[] | null; mono?: boolean }) {
|
||||||
const [open, setOpen] = useState(false)
|
const [open, setOpen] = useState(false)
|
||||||
if (!text || (Array.isArray(text) && text.length === 0)) {
|
if (!text || (Array.isArray(text) && text.length === 0)) {
|
||||||
return <span className="text-neutral-400">—</span>
|
return <span className="text-neutral-400">—</span>
|
||||||
@@ -112,12 +119,15 @@ function SectionPanel({ title, icon: Icon, color, children, id }: { title: strin
|
|||||||
export function AcademicSections({ planId, color }: { planId: string; color?: string | null }) {
|
export function AcademicSections({ planId, color }: { planId: string; color?: string | null }) {
|
||||||
const qc = useQueryClient()
|
const qc = useQueryClient()
|
||||||
const auth = useSupabaseAuth()
|
const auth = useSupabaseAuth()
|
||||||
|
const [openHistorial, setOpenHistorial] = useState(false)
|
||||||
|
const [openModalIa, setopenModalIa] = useState(false)
|
||||||
if(!planId) return <div>Cargando…</div>
|
if(!planId) return <div>Cargando…</div>
|
||||||
const { data: plan } = useSuspenseQuery(planTextOptions(planId))
|
const { data: plan } = useSuspenseQuery(planTextOptions(planId))
|
||||||
|
|
||||||
const [editing, setEditing] = useState<null | { key: keyof PlanTextFields; title: string }>(null)
|
const [editing, setEditing] = useState<null | { key: keyof PlanTextFields; title: string }>(null)
|
||||||
const [draft, setDraft] = useState("")
|
const [draft, setDraft] = useState("")
|
||||||
|
|
||||||
|
|
||||||
// --- mutation con actualización optimista ---
|
// --- mutation con actualización optimista ---
|
||||||
const updateField = useMutation({
|
const updateField = useMutation({
|
||||||
mutationFn: async ({ key, value }: { key: keyof PlanTextFields; value: string | string[] | null }) => {
|
mutationFn: async ({ key, value }: { key: keyof PlanTextFields; value: string | string[] | null }) => {
|
||||||
@@ -155,46 +165,64 @@ export function AcademicSections({ planId, color }: { planId: string; color?: st
|
|||||||
{ id: "sec-ind", title: "Indicadores de desempeño", icon: Icons.LineChart, key: "indicadores_desempeno" as const, mono: false },
|
{ id: "sec-ind", title: "Indicadores de desempeño", icon: Icons.LineChart, key: "indicadores_desempeno" as const, mono: false },
|
||||||
{ id: "sec-per", title: "Pertinencia", icon: Icons.Lightbulb, key: "pertinencia" as const, mono: false },
|
{ id: "sec-per", title: "Pertinencia", icon: Icons.Lightbulb, key: "pertinencia" as const, mono: false },
|
||||||
{ id: "sec-prm", title: "Prompt (origen)", icon: Icons.Code2, key: "prompt" as const, mono: true },
|
{ id: "sec-prm", title: "Prompt (origen)", icon: Icons.Code2, key: "prompt" as const, mono: true },
|
||||||
|
{ id: "sec-hist", title: "Histórico de cambios", icon: Icons.History, key: "historico" as const, mono: false }
|
||||||
],
|
],
|
||||||
[]
|
[]
|
||||||
)
|
)
|
||||||
|
const [iaContext, setIaContext] = useState<{ key: keyof PlanTextFields; title: string; content: string } | null>(null)
|
||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="grid gap-5 md:grid-cols-2">
|
<div className="grid gap-5 md:grid-cols-2">
|
||||||
{sections.map((s) => {
|
{sections.map((s) => {
|
||||||
const text = plan[s.key] ?? null
|
const text = plan[s.key] ?? null
|
||||||
return (
|
return (
|
||||||
<SectionPanel key={s.id} id={s.id} title={s.title} icon={s.icon} color={color}>
|
<SectionPanel key={s.id} id={s.id} title={s.title} icon={s.icon} color={color}>
|
||||||
<ExpandableText text={text} mono={s.mono} />
|
{s.key === "historico" ? (
|
||||||
<div className="mt-4 flex flex-wrap gap-2">
|
<>
|
||||||
<Button
|
<Button variant="outline" size="sm" onClick={() => setOpenHistorial(true)}>
|
||||||
variant="outline"
|
Ver historial
|
||||||
size="sm"
|
|
||||||
disabled={!text || (Array.isArray(text) && text.length === 0)}
|
|
||||||
onClick={() => {
|
|
||||||
const toCopy = Array.isArray(text) ? text.join("\n") : (text ?? "")
|
|
||||||
if (toCopy) navigator.clipboard.writeText(toCopy)
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
Copiar
|
|
||||||
</Button>
|
</Button>
|
||||||
{s.key !== "prompt" &&
|
<Button variant="outline" size="sm" onClick={() => setopenModalIa(true)}>
|
||||||
(<Button
|
Promt
|
||||||
variant="ghost"
|
</Button>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<ExpandableText text={text} mono={s.mono} />
|
||||||
|
<div className="mt-4 flex flex-wrap gap-2">
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
size="sm"
|
size="sm"
|
||||||
|
disabled={!text || (Array.isArray(text) && text.length === 0)}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
const current = Array.isArray(text) ? text.join("\n") : (text ?? "")
|
const toCopy = Array.isArray(text) ? text.join("\n") : (text ?? "")
|
||||||
setEditing({ key: s.key, title: s.title })
|
if (toCopy) navigator.clipboard.writeText(toCopy)
|
||||||
setDraft(current)
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
Editar
|
Copiar
|
||||||
</Button>)}
|
</Button>
|
||||||
</div>
|
{s.key !== "prompt" && (
|
||||||
</SectionPanel>
|
<Button
|
||||||
)
|
variant="ghost"
|
||||||
})}
|
size="sm"
|
||||||
|
onClick={() => {
|
||||||
|
const current = Array.isArray(text) ? text.join("\n") : (text ?? "")
|
||||||
|
setEditing({ key: s.key, title: s.title })
|
||||||
|
setDraft(current)
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Editar
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</SectionPanel>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Diálogo de edición */}
|
{/* Diálogo de edición */}
|
||||||
@@ -254,10 +282,55 @@ export function AcademicSections({ planId, color }: { planId: string; color?: st
|
|||||||
>
|
>
|
||||||
{updateField.isPending ? "Guardando…" : "Guardar"}
|
{updateField.isPending ? "Guardando…" : "Guardar"}
|
||||||
</Button>
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="secondary"
|
||||||
|
onClick={() => {
|
||||||
|
if (!editing) return
|
||||||
|
const current = draft
|
||||||
|
setIaContext({
|
||||||
|
key: editing.key,
|
||||||
|
title: editing.title,
|
||||||
|
content: current,
|
||||||
|
})
|
||||||
|
setopenModalIa(true)
|
||||||
|
setEditing(null) // 🔹 Cierra el modal de edición
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Mejorar con IA
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
|
||||||
</DialogFooter>
|
</DialogFooter>
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
<HistorialCambiosModal
|
||||||
|
open={openHistorial}
|
||||||
|
onClose={() => setOpenHistorial(false)}
|
||||||
|
planId={planId}
|
||||||
|
onRestore={async (key, value) => {
|
||||||
|
updateField.mutate({ key: key as keyof PlanTextFields, value })
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<AIChatModal
|
||||||
|
//plan_format={plan_format}
|
||||||
|
open={openModalIa}
|
||||||
|
onClose={() => setopenModalIa(false)}
|
||||||
|
context={{
|
||||||
|
section: null,//,iaContext?.title,
|
||||||
|
fieldKey: null,//iaContext?.key,
|
||||||
|
originalText: iaContext?.content,
|
||||||
|
cont_conversation: `Eres un experto en craer planes de estudios basate en el id del plan ${planId}`,
|
||||||
|
}}
|
||||||
|
onAccept={(newText: string) => {
|
||||||
|
if (iaContext) {
|
||||||
|
updateField.mutate({ key: iaContext.key, value: newText })
|
||||||
|
setIaContext(null)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
|
||||||
|
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
200
src/components/ui/prompt-input.tsx
Normal file
200
src/components/ui/prompt-input.tsx
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import { Textarea } from "@/components/ui/textarea"
|
||||||
|
import {
|
||||||
|
Tooltip,
|
||||||
|
TooltipContent,
|
||||||
|
TooltipProvider,
|
||||||
|
TooltipTrigger,
|
||||||
|
} from "@/components/ui/tooltip"
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import React, {
|
||||||
|
createContext,
|
||||||
|
useContext,
|
||||||
|
useEffect,
|
||||||
|
useRef,
|
||||||
|
useState,
|
||||||
|
} from "react"
|
||||||
|
|
||||||
|
type PromptInputContextType = {
|
||||||
|
isLoading: boolean
|
||||||
|
value: string
|
||||||
|
setValue: (value: string) => void
|
||||||
|
maxHeight: number | string
|
||||||
|
onSubmit?: () => void
|
||||||
|
disabled?: boolean
|
||||||
|
textareaRef: React.RefObject<HTMLTextAreaElement | null>
|
||||||
|
}
|
||||||
|
|
||||||
|
const PromptInputContext = createContext<PromptInputContextType>({
|
||||||
|
isLoading: false,
|
||||||
|
value: "",
|
||||||
|
setValue: () => {},
|
||||||
|
maxHeight: 240,
|
||||||
|
onSubmit: undefined,
|
||||||
|
disabled: false,
|
||||||
|
textareaRef: React.createRef<HTMLTextAreaElement>(),
|
||||||
|
})
|
||||||
|
|
||||||
|
function usePromptInput() {
|
||||||
|
const context = useContext(PromptInputContext)
|
||||||
|
if (!context) {
|
||||||
|
throw new Error("usePromptInput must be used within a PromptInput")
|
||||||
|
}
|
||||||
|
return context
|
||||||
|
}
|
||||||
|
|
||||||
|
type PromptInputProps = {
|
||||||
|
isLoading?: boolean
|
||||||
|
value?: string
|
||||||
|
onValueChange?: (value: string) => void
|
||||||
|
maxHeight?: number | string
|
||||||
|
onSubmit?: () => void
|
||||||
|
children: React.ReactNode
|
||||||
|
className?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
function PromptInput({
|
||||||
|
className,
|
||||||
|
isLoading = false,
|
||||||
|
maxHeight = 240,
|
||||||
|
value,
|
||||||
|
onValueChange,
|
||||||
|
onSubmit,
|
||||||
|
children,
|
||||||
|
}: PromptInputProps) {
|
||||||
|
const [internalValue, setInternalValue] = useState(value || "")
|
||||||
|
const textareaRef = useRef<HTMLTextAreaElement>(null)
|
||||||
|
|
||||||
|
const handleChange = (newValue: string) => {
|
||||||
|
setInternalValue(newValue)
|
||||||
|
onValueChange?.(newValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<TooltipProvider>
|
||||||
|
<PromptInputContext.Provider
|
||||||
|
value={{
|
||||||
|
isLoading,
|
||||||
|
value: value ?? internalValue,
|
||||||
|
setValue: onValueChange ?? handleChange,
|
||||||
|
maxHeight,
|
||||||
|
onSubmit,
|
||||||
|
textareaRef,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"border-input bg-background cursor-text rounded-3xl border p-2 shadow-xs",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
onClick={() => textareaRef.current?.focus()}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</PromptInputContext.Provider>
|
||||||
|
</TooltipProvider>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export type PromptInputTextareaProps = {
|
||||||
|
disableAutosize?: boolean
|
||||||
|
} & React.ComponentProps<typeof Textarea>
|
||||||
|
|
||||||
|
function PromptInputTextarea({
|
||||||
|
className,
|
||||||
|
onKeyDown,
|
||||||
|
disableAutosize = false,
|
||||||
|
...props
|
||||||
|
}: PromptInputTextareaProps) {
|
||||||
|
const { value, setValue, maxHeight, onSubmit, disabled, textareaRef } =
|
||||||
|
usePromptInput()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (disableAutosize) return
|
||||||
|
|
||||||
|
if (!textareaRef.current) return
|
||||||
|
|
||||||
|
if (textareaRef.current.scrollTop === 0) {
|
||||||
|
textareaRef.current.style.height = "auto"
|
||||||
|
}
|
||||||
|
|
||||||
|
textareaRef.current.style.height =
|
||||||
|
typeof maxHeight === "number"
|
||||||
|
? `${Math.min(textareaRef.current.scrollHeight, maxHeight)}px`
|
||||||
|
: `min(${textareaRef.current.scrollHeight}px, ${maxHeight})`
|
||||||
|
}, [value, maxHeight, disableAutosize])
|
||||||
|
|
||||||
|
const handleKeyDown = (e: React.KeyboardEvent<HTMLTextAreaElement>) => {
|
||||||
|
if (e.key === "Enter" && !e.shiftKey) {
|
||||||
|
e.preventDefault()
|
||||||
|
onSubmit?.()
|
||||||
|
}
|
||||||
|
onKeyDown?.(e)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Textarea
|
||||||
|
ref={textareaRef}
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => setValue(e.target.value)}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
className={cn(
|
||||||
|
"text-primary min-h-[44px] w-full resize-none border-none bg-transparent shadow-none outline-none focus-visible:ring-0 focus-visible:ring-offset-0",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
rows={1}
|
||||||
|
disabled={disabled}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type PromptInputActionsProps = React.HTMLAttributes<HTMLDivElement>
|
||||||
|
|
||||||
|
function PromptInputActions({
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: PromptInputActionsProps) {
|
||||||
|
return (
|
||||||
|
<div className={cn("flex items-center gap-2", className)} {...props}>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
type PromptInputActionProps = {
|
||||||
|
className?: string
|
||||||
|
tooltip: React.ReactNode
|
||||||
|
children: React.ReactNode
|
||||||
|
side?: "top" | "bottom" | "left" | "right"
|
||||||
|
} & React.ComponentProps<typeof Tooltip>
|
||||||
|
|
||||||
|
function PromptInputAction({
|
||||||
|
tooltip,
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
side = "top",
|
||||||
|
...props
|
||||||
|
}: PromptInputActionProps) {
|
||||||
|
const { disabled } = usePromptInput()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tooltip {...props}>
|
||||||
|
<TooltipTrigger asChild disabled={disabled} onClick={event => event.stopPropagation()}>
|
||||||
|
{children}
|
||||||
|
</TooltipTrigger>
|
||||||
|
<TooltipContent side={side} className={className}>
|
||||||
|
{tooltip}
|
||||||
|
</TooltipContent>
|
||||||
|
</Tooltip>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
PromptInput,
|
||||||
|
PromptInputTextarea,
|
||||||
|
PromptInputActions,
|
||||||
|
PromptInputAction,
|
||||||
|
}
|
||||||
@@ -44,13 +44,13 @@ function TooltipContent({
|
|||||||
data-slot="tooltip-content"
|
data-slot="tooltip-content"
|
||||||
sideOffset={sideOffset}
|
sideOffset={sideOffset}
|
||||||
className={cn(
|
className={cn(
|
||||||
"bg-primary text-primary-foreground animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-fit origin-(--radix-tooltip-content-transform-origin) rounded-md px-3 py-1.5 text-xs text-balance",
|
"bg-foreground text-background animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-fit origin-(--radix-tooltip-content-transform-origin) rounded-md px-3 py-1.5 text-xs text-balance",
|
||||||
className
|
className
|
||||||
)}
|
)}
|
||||||
{...props}
|
{...props}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
<TooltipPrimitive.Arrow className="bg-primary fill-primary z-50 size-2.5 translate-y-[calc(-50%_-_2px)] rotate-45 rounded-[2px]" />
|
<TooltipPrimitive.Arrow className="bg-foreground fill-foreground z-50 size-2.5 translate-y-[calc(-50%_-_2px)] rotate-45 rounded-[2px]" />
|
||||||
</TooltipPrimitive.Content>
|
</TooltipPrimitive.Content>
|
||||||
</TooltipPrimitive.Portal>
|
</TooltipPrimitive.Portal>
|
||||||
)
|
)
|
||||||
|
|||||||
10
src/formatos/plan.json
Normal file
10
src/formatos/plan.json
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
"objetivo_general": "...",
|
||||||
|
"sistema_evaluacion": "...",
|
||||||
|
"perfil_ingreso": "...",
|
||||||
|
"perfil_egreso": "...",
|
||||||
|
"competencias_genericas": "...",
|
||||||
|
"competencias_especificas": "...",
|
||||||
|
"indicadores_desempeno": "...",
|
||||||
|
"pertinencia": "..."
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useMutation, useQueryClient } from "@tanstack/react-query";
|
import { useMutation } from "@tanstack/react-query";
|
||||||
import { supabase } from "@/auth/supabase";
|
import { supabase } from "@/auth/supabase";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -11,8 +11,6 @@ export function useSupabaseUpdateWithHistory<T extends Record<string, any>>(
|
|||||||
tableName: string,
|
tableName: string,
|
||||||
idKey: keyof T = "id" as keyof T
|
idKey: keyof T = "id" as keyof T
|
||||||
) {
|
) {
|
||||||
const qc = useQueryClient();
|
|
||||||
|
|
||||||
// Generar diferencias tipo JSON Patch
|
// Generar diferencias tipo JSON Patch
|
||||||
function generateDiff(oldData: T, newData: Partial<T>) {
|
function generateDiff(oldData: T, newData: Partial<T>) {
|
||||||
const changes: any[] = [];
|
const changes: any[] = [];
|
||||||
|
|||||||
39
src/lib/filesAndVectorStoresClient.ts
Normal file
39
src/lib/filesAndVectorStoresClient.ts
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
import { supabase } from "@/auth/supabase"
|
||||||
|
|
||||||
|
type EdgeModule = "files" | "vectorStores" | "vectorStoreFiles"
|
||||||
|
|
||||||
|
type EdgeArgs = {
|
||||||
|
module: EdgeModule
|
||||||
|
action: string
|
||||||
|
params?: Record<string, any>
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function callFilesAndVectorStoresApi<T = unknown>(
|
||||||
|
args: EdgeArgs,
|
||||||
|
): Promise<T> {
|
||||||
|
const { data, error } = await supabase.functions.invoke<any>(
|
||||||
|
"files-and-vector-stores-api",
|
||||||
|
{
|
||||||
|
body: args,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error(error)
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = data ?? {}
|
||||||
|
|
||||||
|
if (payload.error) {
|
||||||
|
const msg =
|
||||||
|
typeof payload.error === "string"
|
||||||
|
? payload.error
|
||||||
|
: payload.error.message ?? "Error en la función Edge"
|
||||||
|
throw new Error(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Soporta tanto `{ data: [...] }` como `[...]`
|
||||||
|
const result = payload.data !== undefined ? payload.data : payload
|
||||||
|
return result as T
|
||||||
|
}
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
// dummy test
|
||||||
import { StrictMode } from 'react'
|
import { StrictMode } from 'react'
|
||||||
import ReactDOM from 'react-dom/client'
|
import ReactDOM from 'react-dom/client'
|
||||||
import { RouterProvider, createRouter } from '@tanstack/react-router'
|
import { RouterProvider, createRouter } from '@tanstack/react-router'
|
||||||
|
|||||||
@@ -149,8 +149,6 @@ function Layout() {
|
|||||||
|
|
||||||
function Sidebar({ onNavigate }: { onNavigate?: () => void }) {
|
function Sidebar({ onNavigate }: { onNavigate?: () => void }) {
|
||||||
const { claims } = useSupabaseAuth()
|
const { claims } = useSupabaseAuth()
|
||||||
const isAdmin = claims?.role === 'lci' || claims?.role === 'vicerrectoria'
|
|
||||||
|
|
||||||
const canSeeCarreras = ["jefe_carrera", 'vicerrectoria', 'secretario_academico', 'lci'].includes(claims?.role ?? '')
|
const canSeeCarreras = ["jefe_carrera", 'vicerrectoria', 'secretario_academico', 'lci'].includes(claims?.role ?? '')
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
// routes/_authenticated/archivos.tsx
|
// routes/_authenticated/archivos.tsx
|
||||||
import { createFileRoute, useRouter } from "@tanstack/react-router"
|
import { createFileRoute, useRouter } from "@tanstack/react-router"
|
||||||
import { use, useMemo, useState } from "react"
|
import { useEffect, useMemo, useState } from "react"
|
||||||
import { supabase, useSupabaseAuth } from "@/auth/supabase"
|
import { supabase, useSupabaseAuth } from "@/auth/supabase"
|
||||||
import * as Icons from "lucide-react"
|
import * as Icons from "lucide-react"
|
||||||
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
|
||||||
@@ -9,92 +9,204 @@ import { Button } from "@/components/ui/button"
|
|||||||
import { Badge } from "@/components/ui/badge"
|
import { Badge } from "@/components/ui/badge"
|
||||||
import { Label } from "@/components/ui/label"
|
import { Label } from "@/components/ui/label"
|
||||||
import {
|
import {
|
||||||
Dialog, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogTitle,
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
} from "@/components/ui/dialog"
|
} from "@/components/ui/dialog"
|
||||||
import { Textarea } from "@/components/ui/textarea"
|
import { Textarea } from "@/components/ui/textarea"
|
||||||
import { Select, SelectTrigger, SelectContent, SelectItem, SelectValue } from "@/components/ui/select"
|
import {
|
||||||
import { DetailDialog } from "@/components/archivos/DetailDialog"
|
Select,
|
||||||
|
SelectContent,
|
||||||
|
SelectItem,
|
||||||
|
SelectTrigger,
|
||||||
|
SelectValue,
|
||||||
|
} from "@/components/ui/select"
|
||||||
|
|
||||||
import type { RefRow } from "@/types/RefRow"
|
type EdgeModule = "files" | "vectorStores" | "vectorStoreFiles"
|
||||||
import { uuid } from "zod"
|
|
||||||
|
interface VectorStore {
|
||||||
|
id: string
|
||||||
|
object: "vector_store"
|
||||||
|
created_at: number
|
||||||
|
name: string | null
|
||||||
|
description?: string | null
|
||||||
|
usage_bytes: number
|
||||||
|
file_counts: {
|
||||||
|
in_progress: number
|
||||||
|
completed: number
|
||||||
|
failed: number
|
||||||
|
cancelled: number
|
||||||
|
total: number
|
||||||
|
}
|
||||||
|
status: string
|
||||||
|
last_active_at?: number | null
|
||||||
|
metadata?: Record<string, any> | null
|
||||||
|
}
|
||||||
|
|
||||||
|
interface VectorStoreFile {
|
||||||
|
id: string
|
||||||
|
object: string
|
||||||
|
created_at: number
|
||||||
|
vector_store_id: string
|
||||||
|
status: string
|
||||||
|
usage_bytes: number
|
||||||
|
last_error?: { code: string; message: string } | null
|
||||||
|
}
|
||||||
|
|
||||||
|
interface VectorStoreFileMeta {
|
||||||
|
id: string
|
||||||
|
user_id: string | null
|
||||||
|
vector_store_id: string
|
||||||
|
openai_file_id: string
|
||||||
|
label: string | null
|
||||||
|
tags: string[] | null
|
||||||
|
created_at: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type EdgeArgs = {
|
||||||
|
module: EdgeModule
|
||||||
|
action: string
|
||||||
|
params?: Record<string, any>
|
||||||
|
}
|
||||||
|
|
||||||
|
async function callFilesAndVectorStoresApi<T = unknown>(
|
||||||
|
args: EdgeArgs,
|
||||||
|
): Promise<T> {
|
||||||
|
const { data, error } = await supabase.functions.invoke<any>(
|
||||||
|
"files-and-vector-stores-api",
|
||||||
|
{
|
||||||
|
body: args,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error(error)
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = data ?? {}
|
||||||
|
if (payload.error) {
|
||||||
|
const msg =
|
||||||
|
typeof payload.error === "string"
|
||||||
|
? payload.error
|
||||||
|
: payload.error.message ?? "Error en la función Edge"
|
||||||
|
throw new Error(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = payload.data !== undefined ? payload.data : payload
|
||||||
|
return result as T
|
||||||
|
}
|
||||||
|
|
||||||
export const Route = createFileRoute("/_authenticated/archivos")({
|
export const Route = createFileRoute("/_authenticated/archivos")({
|
||||||
component: RouteComponent,
|
component: RouteComponent,
|
||||||
loader: async () => {
|
loader: async () => {
|
||||||
const { data, error } = await supabase
|
const stores = await callFilesAndVectorStoresApi<VectorStore[]>({
|
||||||
.from("documentos")
|
module: "vectorStores",
|
||||||
.select("*")
|
action: "list",
|
||||||
.order("fecha_subida", { ascending: false })
|
params: {
|
||||||
.limit(200)
|
limit: 10,
|
||||||
if (error) throw error
|
},
|
||||||
return (data ?? []) as RefRow[]
|
})
|
||||||
|
return stores ?? []
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
function chipTint(ok?: boolean | null) {
|
/* ====== UI helpers ====== */
|
||||||
return ok
|
|
||||||
? "bg-emerald-50 text-emerald-700 border-emerald-200"
|
function StatusBadge({ status }: { status: string }) {
|
||||||
: "bg-amber-50 text-amber-800 border-amber-200"
|
const label =
|
||||||
|
status === "completed"
|
||||||
|
? "Completado"
|
||||||
|
: status === "in_progress"
|
||||||
|
? "Procesando"
|
||||||
|
: status
|
||||||
|
const base = "text-[10px] px-2 py-0.5 rounded-full border"
|
||||||
|
if (status === "completed") {
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={`${base} bg-emerald-50 text-emerald-700 border-emerald-200`}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</span>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
if (status === "in_progress") {
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={`${base} bg-amber-50 text-amber-800 border-amber-200`}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</span>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<span className={`${base} bg-neutral-50 text-neutral-700 border-neutral-200`}>
|
||||||
|
{label}
|
||||||
|
</span>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* ====== Página principal: lista repositorios (Vector Stores) ====== */
|
||||||
|
|
||||||
function RouteComponent() {
|
function RouteComponent() {
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const rows = Route.useLoaderData() as RefRow[]
|
const vectorStores = Route.useLoaderData() as VectorStore[]
|
||||||
|
|
||||||
const [q, setQ] = useState("")
|
const [q, setQ] = useState("")
|
||||||
const [estado, setEstado] = useState<"todos" | "proc" | "pend">("todos")
|
const [statusFilter, setStatusFilter] = useState<"all" | "completed" | "in_progress">("all")
|
||||||
const [scope, setScope] = useState<"todos" | "internos" | "externos">("todos")
|
const [selected, setSelected] = useState<VectorStore | null>(null)
|
||||||
|
const [dialogOpen, setDialogOpen] = useState(false)
|
||||||
const [viewing, setViewing] = useState<RefRow | null>(null)
|
const [createOpen, setCreateOpen] = useState(false)
|
||||||
const [uploadOpen, setUploadOpen] = useState(false)
|
const [deletingId, setDeletingId] = useState<string | null>(null)
|
||||||
|
|
||||||
const filtered = useMemo(() => {
|
const filtered = useMemo(() => {
|
||||||
const t = q.trim().toLowerCase()
|
const term = q.trim().toLowerCase()
|
||||||
return rows.filter((r) => {
|
return vectorStores.filter((vs) => {
|
||||||
if (estado === "proc" && !r.procesado) return false
|
if (statusFilter !== "all" && vs.status !== statusFilter) return false
|
||||||
if (estado === "pend" && r.procesado) return false
|
if (!term) return true
|
||||||
if (scope === "internos" && !r.interno) return false
|
return (
|
||||||
if (scope === "externos" && r.interno) return false
|
(vs.name ?? "").toLowerCase().includes(term) ||
|
||||||
|
(vs.description ?? "").toLowerCase().includes(term)
|
||||||
if (!t) return true
|
)
|
||||||
const hay =
|
|
||||||
[r.titulo_archivo, r.descripcion, r.fuente_autoridad, r.tipo_contenido, ...(r.tags ?? [])]
|
|
||||||
.filter(Boolean)
|
|
||||||
.some((v) => String(v).toLowerCase().includes(t))
|
|
||||||
return hay
|
|
||||||
})
|
})
|
||||||
}, [rows, q, estado, scope])
|
}, [vectorStores, q, statusFilter])
|
||||||
|
|
||||||
async function remove(id: string) {
|
function openDetails(vs: VectorStore) {
|
||||||
if (!confirm("¿Eliminar archivo de referencia?")) return
|
setSelected(vs)
|
||||||
const { error } = await supabase
|
setDialogOpen(true)
|
||||||
.from("documentos")
|
}
|
||||||
.delete()
|
|
||||||
.eq("documentos_id", id)
|
|
||||||
if (error) return alert(error.message)
|
|
||||||
|
|
||||||
|
async function handleDelete(id: string) {
|
||||||
|
if (!confirm("¿Eliminar este repositorio y sus archivos asociados en OpenAI?")) return
|
||||||
|
setDeletingId(id)
|
||||||
try {
|
try {
|
||||||
const res = await fetch(`${import.meta.env.VITE_BACK_ORIGIN}/api/eliminar/documento`, {
|
await callFilesAndVectorStoresApi({
|
||||||
method: "DELETE",
|
module: "vectorStores",
|
||||||
headers: { "Content-Type": "application/json" },
|
action: "delete",
|
||||||
body: JSON.stringify({ documentos_id: id }),
|
params: { vector_store_id: id },
|
||||||
})
|
})
|
||||||
|
|
||||||
if (!res.ok) {
|
await supabase
|
||||||
throw new Error("Se falló al eliminar el documento")
|
.from("vector_store_files_meta")
|
||||||
}
|
.delete()
|
||||||
} catch (err) {
|
.eq("vector_store_id", id)
|
||||||
console.error("Error al eliminar el documento:", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
router.invalidate()
|
router.invalidate()
|
||||||
|
} catch (err: any) {
|
||||||
|
alert(err?.message ?? "Error al eliminar el repositorio")
|
||||||
|
} finally {
|
||||||
|
setDeletingId(null)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="p-6 space-y-4">
|
<div className="p-6 space-y-4">
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeader className="flex flex-col gap-3 sm:flex-row sm:items-center sm:justify-between">
|
<CardHeader className="flex flex-col gap-3 sm:flex-row sm:items-center sm:justify-between">
|
||||||
<CardTitle className="font-mono">Archivos de referencia</CardTitle>
|
<CardTitle className="font-mono">Repositorios de archivos</CardTitle>
|
||||||
|
|
||||||
<div className="flex w-full flex-col gap-2 sm:w-auto sm:flex-row sm:items-center">
|
<div className="flex w-full flex-col gap-2 sm:w-auto sm:flex-row sm:items-center">
|
||||||
<div className="relative w-full sm:w-80">
|
<div className="relative w-full sm:w-80">
|
||||||
@@ -102,240 +214,502 @@ function RouteComponent() {
|
|||||||
<Input
|
<Input
|
||||||
value={q}
|
value={q}
|
||||||
onChange={(e) => setQ(e.target.value)}
|
onChange={(e) => setQ(e.target.value)}
|
||||||
placeholder="Buscar por título, etiqueta, fuente…"
|
placeholder="Buscar por nombre o descripción…"
|
||||||
className="pl-8"
|
className="pl-8"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Select value={estado} onValueChange={(v: any) => setEstado(v)}>
|
<Select
|
||||||
|
value={statusFilter}
|
||||||
|
onValueChange={(v) =>
|
||||||
|
setStatusFilter(v as "all" | "completed" | "in_progress")
|
||||||
|
}
|
||||||
|
>
|
||||||
<SelectTrigger className="sm:w-[160px]">
|
<SelectTrigger className="sm:w-[160px]">
|
||||||
<SelectValue placeholder="Estado" />
|
<SelectValue placeholder="Estado" />
|
||||||
</SelectTrigger>
|
</SelectTrigger>
|
||||||
<SelectContent>
|
<SelectContent>
|
||||||
<SelectItem value="todos">Todos</SelectItem>
|
<SelectItem value="all">Todos</SelectItem>
|
||||||
<SelectItem value="proc">Procesados</SelectItem>
|
<SelectItem value="completed">Completados</SelectItem>
|
||||||
<SelectItem value="pend">Pendientes</SelectItem>
|
<SelectItem value="in_progress">En proceso</SelectItem>
|
||||||
</SelectContent>
|
</SelectContent>
|
||||||
</Select>
|
</Select>
|
||||||
|
|
||||||
<Select value={scope} onValueChange={(v: any) => setScope(v)}>
|
<Button onClick={() => setCreateOpen(true)}>
|
||||||
<SelectTrigger className="sm:w-[160px]">
|
<Icons.FolderPlus className="w-4 h-4 mr-2" />
|
||||||
<SelectValue placeholder="Ámbito" />
|
Nuevo repositorio
|
||||||
</SelectTrigger>
|
|
||||||
<SelectContent>
|
|
||||||
<SelectItem value="todos">Todos</SelectItem>
|
|
||||||
<SelectItem value="internos">Internos</SelectItem>
|
|
||||||
<SelectItem value="externos">Externos</SelectItem>
|
|
||||||
</SelectContent>
|
|
||||||
</Select>
|
|
||||||
|
|
||||||
<Button onClick={() => setUploadOpen(true)}>
|
|
||||||
<Icons.Upload className="w-4 h-4 mr-2" /> Nuevo
|
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
|
|
||||||
<CardContent>
|
<CardContent>
|
||||||
<div className="grid gap-3 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4">
|
{filtered.length ? (
|
||||||
{filtered.map((r) => (
|
<div className="grid gap-3 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4">
|
||||||
<article
|
{filtered.map((vs) => (
|
||||||
key={r.documentos_id}
|
<article
|
||||||
className="rounded-2xl border bg-white/70 dark:bg-neutral-900/60 p-4 flex flex-col gap-3"
|
key={vs.id}
|
||||||
>
|
className="rounded-2xl border bg-white/70 dark:bg-neutral-900/60 p-4 flex flex-col gap-3"
|
||||||
<header className="min-w-0">
|
>
|
||||||
<div className="flex items-center justify-between gap-2">
|
<header className="min-w-0 space-y-1">
|
||||||
<h3 className="font-semibold truncate">{r.titulo_archivo ?? "(Sin título)"}</h3>
|
<div className="flex items-center justify-between gap-2">
|
||||||
<span className={`text-[10px] px-2 py-0.5 rounded-full border ${chipTint(r.procesado)}`}>
|
<h3 className="font-semibold truncate">
|
||||||
{r.procesado ? "Procesado" : "Pendiente"}
|
{vs.name || "(Sin nombre)"}
|
||||||
</span>
|
</h3>
|
||||||
|
<StatusBadge status={vs.status} />
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-wrap items-center gap-2 text-xs text-neutral-600">
|
||||||
|
<Badge variant="outline">
|
||||||
|
Archivos: {vs.file_counts?.completed ?? 0}
|
||||||
|
</Badge>
|
||||||
|
{typeof vs.usage_bytes === "number" && (
|
||||||
|
<span>
|
||||||
|
{(vs.usage_bytes / 1024 / 1024).toFixed(2)} MB
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
{vs.last_active_at && (
|
||||||
|
<span className="inline-flex items-center gap-1">
|
||||||
|
<Icons.Clock3 className="w-3 h-3" />
|
||||||
|
{new Date(vs.last_active_at * 1000).toLocaleDateString()}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
{vs.description && (
|
||||||
|
<p className="text-sm text-neutral-700 line-clamp-3">
|
||||||
|
{vs.description}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className="mt-auto flex items-center justify-between gap-2 pt-2">
|
||||||
|
<Button variant="ghost" size="sm" onClick={() => openDetails(vs)}>
|
||||||
|
<Icons.Eye className="w-4 h-4 mr-1" /> Abrir
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => handleDelete(vs.id)}
|
||||||
|
disabled={deletingId === vs.id}
|
||||||
|
>
|
||||||
|
<Icons.Trash2 className="w-4 h-4 mr-1" />
|
||||||
|
{deletingId === vs.id ? "Eliminando…" : "Eliminar"}
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<div className="mt-1 text-xs text-neutral-600 flex flex-wrap gap-2">
|
</article>
|
||||||
{r.tipo_contenido && <Badge variant="outline">{r.tipo_contenido}</Badge>}
|
))}
|
||||||
{r.interno != null && (
|
</div>
|
||||||
<Badge variant="outline">{r.interno ? "Interno" : "Externo"}</Badge>
|
) : (
|
||||||
)}
|
<div className="text-center text-sm text-neutral-500 py-10">
|
||||||
{r.fecha_subida && (
|
No hay repositorios todavía. Crea uno nuevo para empezar 🚀
|
||||||
<span className="inline-flex items-center gap-1">
|
</div>
|
||||||
<Icons.CalendarClock className="w-3 h-3" />
|
|
||||||
{new Date(r.fecha_subida).toLocaleDateString()}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
{r.descripcion && (
|
|
||||||
<p className="text-sm text-neutral-700 line-clamp-3">{r.descripcion}</p>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Tags
|
|
||||||
{r.tags && r.tags.length > 0 && (
|
|
||||||
<div className="flex flex-wrap gap-1.5">
|
|
||||||
{r.tags.map((t, i) => (
|
|
||||||
<span key={i} className="text-[10px] px-2 py-0.5 rounded-full border bg-white/60">
|
|
||||||
#{t}
|
|
||||||
</span>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)} */}
|
|
||||||
|
|
||||||
<div className="mt-auto flex items-center justify-between gap-2">
|
|
||||||
<Button variant="ghost" size="sm" onClick={() => setViewing(r)}>
|
|
||||||
<Icons.Eye className="w-4 h-4 mr-1" /> Ver
|
|
||||||
</Button>
|
|
||||||
<Button variant="ghost" size="sm" onClick={() => remove(r.documentos_id)}>
|
|
||||||
<Icons.Trash2 className="w-4 h-4 mr-1" /> Eliminar
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</article>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{!filtered.length && (
|
|
||||||
<div className="text-center text-sm text-neutral-500 py-10">No hay archivos</div>
|
|
||||||
)}
|
)}
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
|
|
||||||
{/* Detalle */}
|
<CreateVectorStoreDialog
|
||||||
<DetailDialog row={viewing} onClose={() => setViewing(null)} />
|
open={createOpen}
|
||||||
|
onOpenChange={setCreateOpen}
|
||||||
|
onCreated={() => router.invalidate()}
|
||||||
|
/>
|
||||||
|
|
||||||
{/* Subida */}
|
<VectorStoreDialog
|
||||||
<UploadDialog open={uploadOpen} onOpenChange={setUploadOpen} onDone={() => router.invalidate()} />
|
store={selected}
|
||||||
|
open={dialogOpen}
|
||||||
|
onOpenChange={(open) => {
|
||||||
|
setDialogOpen(open)
|
||||||
|
if (!open) setSelected(null)
|
||||||
|
}}
|
||||||
|
onUpdated={() => router.invalidate()}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ========= Subida ========= */
|
/* ====== Crear repositorio ====== */
|
||||||
function UploadDialog({
|
|
||||||
open, onOpenChange, onDone,
|
|
||||||
}: { open: boolean; onOpenChange: (o: boolean) => void; onDone: () => void }) {
|
|
||||||
const supabaseAuth = useSupabaseAuth()
|
|
||||||
const [file, setFile] = useState<File | null>(null)
|
|
||||||
const [instrucciones, setInstrucciones] = useState("")
|
|
||||||
const [tags, setTags] = useState("")
|
|
||||||
const [interno, setInterno] = useState(true)
|
|
||||||
const [fuente, setFuente] = useState("")
|
|
||||||
const [subiendo, setSubiendo] = useState(false)
|
|
||||||
|
|
||||||
async function toBase64(f: File): Promise<string> {
|
function CreateVectorStoreDialog({
|
||||||
const buf = await f.arrayBuffer()
|
open,
|
||||||
const bytes = new Uint8Array(buf)
|
onOpenChange,
|
||||||
let binary = ""
|
onCreated,
|
||||||
for (let i = 0; i < bytes.byteLength; i++) binary += String.fromCharCode(bytes[i])
|
}: {
|
||||||
return btoa(binary)
|
open: boolean
|
||||||
}
|
onOpenChange: (open: boolean) => void
|
||||||
|
onCreated: () => void
|
||||||
|
}) {
|
||||||
|
const [name, setName] = useState("")
|
||||||
|
const [description, setDescription] = useState("")
|
||||||
|
const [creating, setCreating] = useState(false)
|
||||||
|
|
||||||
async function upload() {
|
async function handleCreate() {
|
||||||
if (!file) { alert("Selecciona un archivo"); return }
|
if (!name.trim()) {
|
||||||
if (!instrucciones.trim()) { alert("Escribe las instrucciones"); return }
|
alert("Escribe un nombre para el repositorio")
|
||||||
|
return
|
||||||
setSubiendo(true)
|
}
|
||||||
|
setCreating(true)
|
||||||
try {
|
try {
|
||||||
const fileBase64 = await toBase64(file)
|
await callFilesAndVectorStoresApi<VectorStore>({
|
||||||
// Enviamos al motor (inserta en la tabla si insert=true)
|
module: "vectorStores",
|
||||||
const res = await fetch(`${import.meta.env.VITE_BACK_ORIGIN}/api/upload/documento`, {
|
action: "create",
|
||||||
method: "POST",
|
params: { name: name.trim(), description: description.trim() || undefined },
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({
|
|
||||||
prompt: instrucciones,
|
|
||||||
fileBase64,
|
|
||||||
insert: true,
|
|
||||||
uuid: supabaseAuth.user?.id ?? null,
|
|
||||||
}),
|
|
||||||
})
|
})
|
||||||
if (!res.ok) {
|
|
||||||
const txt = await res.text()
|
|
||||||
throw new Error(txt || "Error al subir")
|
|
||||||
}
|
|
||||||
// Ajustes extra (tags, interno, fuente) si el motor no los llenó
|
|
||||||
// Intentamos leer el id que regrese el servicio; si no, solo invalidamos.
|
|
||||||
let createdId: string | null = null
|
|
||||||
try {
|
|
||||||
const payload = await res.json()
|
|
||||||
createdId =
|
|
||||||
payload?.documentos_id ||
|
|
||||||
payload?.id ||
|
|
||||||
payload?.data?.documentos_id ||
|
|
||||||
null
|
|
||||||
} catch { /* noop */ }
|
|
||||||
|
|
||||||
if (createdId && (tags.trim() || fuente.trim() || typeof interno === "boolean")) {
|
|
||||||
await supabase
|
|
||||||
.from("documentos")
|
|
||||||
.update({
|
|
||||||
tags: tags.trim() ? tags.split(",").map((s) => s.trim()).filter(Boolean) : undefined,
|
|
||||||
fuente_autoridad: fuente.trim() || undefined,
|
|
||||||
interno,
|
|
||||||
})
|
|
||||||
.eq("documentos_id", createdId)
|
|
||||||
}
|
|
||||||
|
|
||||||
onOpenChange(false)
|
onOpenChange(false)
|
||||||
onDone()
|
setName("")
|
||||||
} catch (e: any) {
|
setDescription("")
|
||||||
alert(e?.message ?? "Error al subir el documento")
|
onCreated()
|
||||||
|
} catch (err: any) {
|
||||||
|
alert(err?.message ?? "Error al crear el repositorio")
|
||||||
} finally {
|
} finally {
|
||||||
setSubiendo(false)
|
setCreating(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
||||||
<DialogContent className="max-w-xl">
|
<DialogContent className="max-w-md">
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle className="font-mono" >Nuevo archivo de referencia</DialogTitle>
|
<DialogTitle className="font-mono">Nuevo repositorio</DialogTitle>
|
||||||
<DialogDescription>
|
<DialogDescription>
|
||||||
Sube un documento y escribe instrucciones para su procesamiento. Se guardará en la base y se marcará como
|
Crea un Vector Store para agrupar archivos relacionados.
|
||||||
<em> procesado </em> cuando termine el flujo.
|
|
||||||
</DialogDescription>
|
</DialogDescription>
|
||||||
</DialogHeader>
|
</DialogHeader>
|
||||||
|
|
||||||
<div className="grid gap-3">
|
<div className="grid gap-3">
|
||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<Label>Archivo</Label>
|
<Label>Nombre</Label>
|
||||||
<Input type="file" accept=".pdf,.doc,.docx,.txt,.md" onChange={(e) => setFile(e.target.files?.[0] ?? null)} />
|
<Input
|
||||||
{file && (
|
value={name}
|
||||||
<div className="text-xs text-neutral-600">{file.name} · {(file.size / 1024).toFixed(1)} KB</div>
|
onChange={(e) => setName(e.target.value)}
|
||||||
)}
|
placeholder="Planeación curricular, Entrevistas…"
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="space-y-1">
|
|
||||||
<Label>Instrucciones</Label>
|
|
||||||
<Textarea
|
|
||||||
value={instrucciones}
|
|
||||||
onChange={(e) => setInstrucciones(e.target.value)}
|
|
||||||
placeholder="Ej.: Extrae temario, resultados de aprendizaje y bibliografía; limpia ruido y normaliza formato."
|
|
||||||
className="min-h-[120px]"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="grid sm:grid-cols-2 gap-3">
|
|
||||||
<div className="space-y-1">
|
|
||||||
<Label>Tags (separados por coma)</Label>
|
|
||||||
<Input value={tags} onChange={(e) => setTags(e.target.value)} placeholder="normatividad, plan, lineamientos" />
|
|
||||||
</div>
|
|
||||||
<div className="space-y-1">
|
|
||||||
<Label>Fuente de autoridad</Label>
|
|
||||||
<Input value={fuente} onChange={(e) => setFuente(e.target.value)} placeholder="SEP, ANUIES…" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
<Label>Ámbito</Label>
|
<Label>Descripción (opcional)</Label>
|
||||||
<Select value={String(interno)} onValueChange={(v) => setInterno(v === "true")}>
|
<Textarea
|
||||||
<SelectTrigger><SelectValue /></SelectTrigger>
|
value={description}
|
||||||
<SelectContent>
|
onChange={(e) => setDescription(e.target.value)}
|
||||||
<SelectItem value="true">Interno</SelectItem>
|
placeholder="Breve descripción del contenido de este repositorio."
|
||||||
<SelectItem value="false">Externo</SelectItem>
|
className="min-h-[80px]"
|
||||||
</SelectContent>
|
/>
|
||||||
</Select>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<DialogFooter>
|
<DialogFooter>
|
||||||
<Button variant="outline" onClick={() => onOpenChange(false)}>Cancelar</Button>
|
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
||||||
<Button onClick={upload} disabled={subiendo || !file || !instrucciones.trim()}>
|
Cancelar
|
||||||
{subiendo ? "Subiendo…" : "Subir"}
|
</Button>
|
||||||
|
<Button onClick={handleCreate} disabled={creating || !name.trim()}>
|
||||||
|
{creating ? "Creando…" : "Crear repositorio"}
|
||||||
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ====== Detalle de un repositorio: archivos + subida ====== */
|
||||||
|
|
||||||
|
type FileRow = {
|
||||||
|
file: VectorStoreFile
|
||||||
|
meta: VectorStoreFileMeta | null
|
||||||
|
}
|
||||||
|
|
||||||
|
function VectorStoreDialog({
|
||||||
|
store,
|
||||||
|
open,
|
||||||
|
onOpenChange,
|
||||||
|
onUpdated,
|
||||||
|
}: {
|
||||||
|
store: VectorStore | null
|
||||||
|
open: boolean
|
||||||
|
onOpenChange: (open: boolean) => void
|
||||||
|
onUpdated: () => void
|
||||||
|
}) {
|
||||||
|
const supabaseAuth = useSupabaseAuth()
|
||||||
|
const [files, setFiles] = useState<FileRow[]>([])
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const [error, setError] = useState<string | null>(null)
|
||||||
|
const [uploading, setUploading] = useState(false)
|
||||||
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
|
const [file, setFile] = useState<File | null>(null)
|
||||||
|
const [label, setLabel] = useState("")
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!open || !store) return
|
||||||
|
void refreshFiles()
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [open, store?.id])
|
||||||
|
|
||||||
|
async function refreshFiles() {
|
||||||
|
if (!store) return
|
||||||
|
setLoading(true)
|
||||||
|
setError(null)
|
||||||
|
try {
|
||||||
|
const vectorFiles = await callFilesAndVectorStoresApi<VectorStoreFile[]>({
|
||||||
|
module: "vectorStoreFiles",
|
||||||
|
action: "list",
|
||||||
|
params: { vector_store_id: store.id },
|
||||||
|
})
|
||||||
|
|
||||||
|
const { data: metaRows, error: metaError } = await supabase
|
||||||
|
.from("vector_store_files_meta")
|
||||||
|
.select("*")
|
||||||
|
.eq("vector_store_id", store.id)
|
||||||
|
.order("created_at", { ascending: false })
|
||||||
|
|
||||||
|
if (metaError) throw metaError
|
||||||
|
|
||||||
|
const meta = (metaRows ?? []) as VectorStoreFileMeta[]
|
||||||
|
|
||||||
|
const merged: FileRow[] = (vectorFiles ?? []).map((vf) => ({
|
||||||
|
file: vf,
|
||||||
|
meta: meta.find((m) => m.openai_file_id === vf.id) ?? null,
|
||||||
|
}))
|
||||||
|
|
||||||
|
setFiles(merged)
|
||||||
|
} catch (err: any) {
|
||||||
|
console.error(err)
|
||||||
|
setError(err?.message ?? "No se pudieron cargar los archivos")
|
||||||
|
} finally {
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async function handleUpload() {
|
||||||
|
if (!store || !file) {
|
||||||
|
alert("Selecciona un archivo")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
setUploading(true)
|
||||||
|
try {
|
||||||
|
// 1) Subir archivo a OpenAI vía Edge con FormData (igual que en tu script)
|
||||||
|
const formData = new FormData()
|
||||||
|
formData.append("module", "files")
|
||||||
|
formData.append("action", "upload")
|
||||||
|
formData.append("file", file)
|
||||||
|
formData.append("purpose", "assistants") // o lo que uses en tu flujo
|
||||||
|
|
||||||
|
const { data, error } = await supabase.functions.invoke<any>(
|
||||||
|
"files-and-vector-stores-api",
|
||||||
|
{
|
||||||
|
body: formData,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error(error)
|
||||||
|
throw error
|
||||||
|
}
|
||||||
|
|
||||||
|
const uploaded = data
|
||||||
|
// La respuesta es el objeto "file" de OpenAI:
|
||||||
|
// { object: "file", id: "file-xxx", ... }
|
||||||
|
const openaiFileId: string | undefined = uploaded?.id
|
||||||
|
|
||||||
|
if (!openaiFileId) {
|
||||||
|
console.error("Respuesta Edge inesperada:", uploaded)
|
||||||
|
throw new Error("La Edge Function no devolvió el id del archivo")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2) Mapear archivo al Vector Store (JSON normal)
|
||||||
|
await callFilesAndVectorStoresApi<any>({
|
||||||
|
module: "vectorStoreFiles",
|
||||||
|
action: "create",
|
||||||
|
params: {
|
||||||
|
vector_store_id: store.id,
|
||||||
|
body: {
|
||||||
|
file_id: openaiFileId,
|
||||||
|
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// 3) Guardar metadata en Supabase
|
||||||
|
const { error: insertError } = await supabase
|
||||||
|
.from("vector_store_files_meta")
|
||||||
|
.insert({
|
||||||
|
user_id: supabaseAuth.user?.id ?? null,
|
||||||
|
vector_store_id: store.id,
|
||||||
|
openai_file_id: openaiFileId,
|
||||||
|
label: label.trim() || file.name,
|
||||||
|
})
|
||||||
|
|
||||||
|
if (insertError) throw insertError
|
||||||
|
|
||||||
|
setFile(null)
|
||||||
|
setLabel("")
|
||||||
|
await refreshFiles()
|
||||||
|
onUpdated()
|
||||||
|
} catch (err: any) {
|
||||||
|
console.error(err)
|
||||||
|
alert(err?.message ?? "Error al subir el archivo")
|
||||||
|
} finally {
|
||||||
|
setUploading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
async function handleDeleteFile(fileId: string) {
|
||||||
|
if (!store) return
|
||||||
|
if (!confirm("¿Eliminar este archivo del repositorio y de OpenAI?")) return
|
||||||
|
setRefreshing(true)
|
||||||
|
try {
|
||||||
|
await callFilesAndVectorStoresApi<any>({
|
||||||
|
module: "vectorStoreFiles",
|
||||||
|
action: "delete",
|
||||||
|
params: {
|
||||||
|
vector_store_id: store.id,
|
||||||
|
file_id: fileId,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// Opcional: eliminar también el archivo global de OpenAI
|
||||||
|
await callFilesAndVectorStoresApi<any>({
|
||||||
|
module: "files",
|
||||||
|
action: "delete",
|
||||||
|
params: { id: fileId },
|
||||||
|
})
|
||||||
|
|
||||||
|
await supabase
|
||||||
|
.from("vector_store_files_meta")
|
||||||
|
.delete()
|
||||||
|
.eq("openai_file_id", fileId)
|
||||||
|
|
||||||
|
await refreshFiles()
|
||||||
|
onUpdated()
|
||||||
|
} catch (err: any) {
|
||||||
|
console.error(err)
|
||||||
|
alert(err?.message ?? "Error al eliminar el archivo")
|
||||||
|
} finally {
|
||||||
|
setRefreshing(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!store) return null
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={open} onOpenChange={onOpenChange}>
|
||||||
|
<DialogContent className="max-w-2xl">
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle className="flex items-center gap-2">
|
||||||
|
<Icons.Folder className="h-4 w-4" />
|
||||||
|
{store.name || "(Sin nombre)"}
|
||||||
|
</DialogTitle>
|
||||||
|
<DialogDescription>
|
||||||
|
Gestiona los archivos asociados a este repositorio (Vector Store).
|
||||||
|
</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
<div className="space-y-4">
|
||||||
|
<div className="flex flex-wrap items-center gap-3 text-xs text-neutral-600">
|
||||||
|
<StatusBadge status={store.status} />
|
||||||
|
<Badge variant="outline">
|
||||||
|
Archivos completados: {store.file_counts?.completed ?? 0}
|
||||||
|
</Badge>
|
||||||
|
<Badge variant="outline">
|
||||||
|
Total archivos: {store.file_counts?.total ?? 0}
|
||||||
|
</Badge>
|
||||||
|
{typeof store.usage_bytes === "number" && (
|
||||||
|
<span>{(store.usage_bytes / 1024 / 1024).toFixed(2)} MB</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Subida de archivo */}
|
||||||
|
<div className="space-y-2 rounded-lg border bg-muted/50 p-4">
|
||||||
|
<Label className="text-xs font-semibold uppercase tracking-wide text-neutral-500">
|
||||||
|
Agregar archivo al repositorio
|
||||||
|
</Label>
|
||||||
|
<div className="grid gap-3 sm:grid-cols-[minmax(0,1fr)_minmax(0,1fr)] sm:items-end">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Input
|
||||||
|
type="file"
|
||||||
|
accept=".pdf,.doc,.docx,.txt,.md"
|
||||||
|
onChange={(e) => setFile(e.target.files?.[0] ?? null)}
|
||||||
|
/>
|
||||||
|
{file && (
|
||||||
|
<div className="text-xs text-neutral-600">
|
||||||
|
{file.name} · {(file.size / 1024).toFixed(1)} KB
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Label>Título / etiqueta</Label>
|
||||||
|
<Input
|
||||||
|
value={label}
|
||||||
|
onChange={(e) => setLabel(e.target.value)}
|
||||||
|
placeholder="Ej.: Plan 2025, Entrevista 3…"
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
className="mt-2 w-full sm:w-auto"
|
||||||
|
onClick={handleUpload}
|
||||||
|
disabled={uploading || !file}
|
||||||
|
>
|
||||||
|
{uploading ? "Subiendo…" : "Subir al repositorio"}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Lista de archivos */}
|
||||||
|
<div className="space-y-2">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<Label>Archivos en este repositorio</Label>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => refreshFiles()}
|
||||||
|
disabled={loading || refreshing}
|
||||||
|
>
|
||||||
|
<Icons.RefreshCw className="h-4 w-4 mr-1" />
|
||||||
|
Actualizar
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{loading ? (
|
||||||
|
<div className="text-xs text-neutral-500 py-4">
|
||||||
|
Cargando archivos…
|
||||||
|
</div>
|
||||||
|
) : error ? (
|
||||||
|
<div className="text-xs text-red-500 py-4">{error}</div>
|
||||||
|
) : files.length === 0 ? (
|
||||||
|
<div className="text-xs text-neutral-500 py-4">
|
||||||
|
Todavía no hay archivos en este repositorio ⚡
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<ul className="space-y-2 max-h-64 overflow-y-auto pr-1">
|
||||||
|
{files.map(({ file, meta }) => (
|
||||||
|
<li
|
||||||
|
key={file.id}
|
||||||
|
className="flex items-center justify-between gap-3 rounded-md border bg-background px-3 py-2"
|
||||||
|
>
|
||||||
|
<div className="min-w-0">
|
||||||
|
<p className="font-medium truncate">
|
||||||
|
{meta?.label || file.id}
|
||||||
|
</p>
|
||||||
|
<p className="text-xs text-neutral-500 truncate">
|
||||||
|
{new Date(file.created_at * 1000).toLocaleString()} ·{" "}
|
||||||
|
{(file.usage_bytes / 1024).toFixed(1)} KB
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<StatusBadge status={file.status} />
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => handleDeleteFile(file.id)}
|
||||||
|
>
|
||||||
|
<Icons.Trash2 className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<DialogFooter>
|
||||||
|
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
||||||
|
Cerrar
|
||||||
</Button>
|
</Button>
|
||||||
</DialogFooter>
|
</DialogFooter>
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import { createFileRoute, Link, useRouter } from '@tanstack/react-router'
|
|||||||
import { useSuspenseQuery, queryOptions, useQueryClient } from '@tanstack/react-query'
|
import { useSuspenseQuery, queryOptions, useQueryClient } from '@tanstack/react-query'
|
||||||
import { supabase } from '@/auth/supabase'
|
import { supabase } from '@/auth/supabase'
|
||||||
import * as Icons from 'lucide-react'
|
import * as Icons from 'lucide-react'
|
||||||
import { useEffect, useMemo, useState } from 'react'
|
import { useMemo, useState } from 'react'
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
import { Input } from '@/components/ui/input'
|
import { Input } from '@/components/ui/input'
|
||||||
import { Select, SelectTrigger, SelectContent, SelectItem, SelectValue } from '@/components/ui/select'
|
import { Select, SelectTrigger, SelectContent, SelectItem, SelectValue } from '@/components/ui/select'
|
||||||
@@ -81,7 +81,7 @@ async function fetchAsignaturas(search: SearchState): Promise<Asignatura[]> {
|
|||||||
const planIds = await fetchPlanIdsByScope(search)
|
const planIds = await fetchPlanIdsByScope(search)
|
||||||
if (planIds && planIds.length === 0) return []
|
if (planIds && planIds.length === 0) return []
|
||||||
console.log(AsignaturaCard);
|
console.log(AsignaturaCard);
|
||||||
|
|
||||||
let query = supabase
|
let query = supabase
|
||||||
.from('asignaturas')
|
.from('asignaturas')
|
||||||
.select(`
|
.select(`
|
||||||
@@ -168,25 +168,25 @@ function RouteComponent() {
|
|||||||
const [q, setQ] = useState(search.q ?? '')
|
const [q, setQ] = useState(search.q ?? '')
|
||||||
const [sem, setSem] = useState<string>('todos')
|
const [sem, setSem] = useState<string>('todos')
|
||||||
const [tipo, setTipo] = useState<string>('todos')
|
const [tipo, setTipo] = useState<string>('todos')
|
||||||
const [groupBy, setGroupBy] = useState<'semestre' | 'ninguno'>('semestre')
|
const [groupBy] = useState<'semestre' | 'ninguno'>('semestre')
|
||||||
const [flag, setFlag] = useState<'' | 'sinBibliografia' | 'sinCriterios' | 'sinContenidos'>(search.f ?? '')
|
const [flag, setFlag] = useState<'' | 'sinBibliografia' | 'sinCriterios' | 'sinContenidos'>(search.f ?? '')
|
||||||
|
|
||||||
const [facultad, setFacultad] = useState("todas")
|
const [facultad, setFacultad] = useState("todas")
|
||||||
const [carrera, setCarrera] = useState("todas")
|
const [carrera, setCarrera] = useState("todas")
|
||||||
|
|
||||||
/* useEffect(() => {
|
/* useEffect(() => {
|
||||||
const timeout = setTimeout(() => {
|
const timeout = setTimeout(() => {
|
||||||
router.navigate({
|
router.navigate({
|
||||||
to: '/asignaturas',
|
to: '/asignaturas',
|
||||||
search: { ...search, q },
|
search: { ...search, q },
|
||||||
replace: true,
|
replace: true,
|
||||||
})
|
})
|
||||||
}, 400)
|
}, 400)
|
||||||
return () => clearTimeout(timeout)
|
return () => clearTimeout(timeout)
|
||||||
}, [q]) */
|
}, [q]) */
|
||||||
|
|
||||||
|
|
||||||
function handleChange(e: React.ChangeEvent<HTMLInputElement>) {
|
function handleChange(e: React.ChangeEvent<HTMLInputElement>) {
|
||||||
const value = e.target.value
|
const value = e.target.value
|
||||||
setQ(value)
|
setQ(value)
|
||||||
router.navigate({
|
router.navigate({
|
||||||
@@ -199,30 +199,30 @@ function handleChange(e: React.ChangeEvent<HTMLInputElement>) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// 🟣 Lista única de facultades
|
// 🟣 Lista única de facultades
|
||||||
const facultadesList = useMemo(() => {
|
const facultadesList = useMemo(() => {
|
||||||
const unique = new Map<string, string>()
|
const unique = new Map<string, string>()
|
||||||
planes?.forEach((p) => {
|
planes?.forEach((p) => {
|
||||||
const fac = p.carrera?.facultad
|
const fac = p.carrera?.facultad
|
||||||
if (fac?.id && fac?.nombre) unique.set(fac.id, fac.nombre)
|
if (fac?.id && fac?.nombre) unique.set(fac.id, fac.nombre)
|
||||||
})
|
})
|
||||||
return Array.from(unique.entries())
|
return Array.from(unique.entries())
|
||||||
}, [planes])
|
}, [planes])
|
||||||
|
|
||||||
// 🎓 Lista de carreras según la facultad seleccionada
|
// 🎓 Lista de carreras según la facultad seleccionada
|
||||||
const carrerasList = useMemo(() => {
|
const carrerasList = useMemo(() => {
|
||||||
const unique = new Map<string, string>()
|
const unique = new Map<string, string>()
|
||||||
planes?.forEach((p) => {
|
planes?.forEach((p) => {
|
||||||
if (
|
if (
|
||||||
p.carrera?.id &&
|
p.carrera?.id &&
|
||||||
p.carrera?.nombre &&
|
p.carrera?.nombre &&
|
||||||
(!facultad || facultad === "todas" || p.carrera?.facultad?.id === facultad)
|
(!facultad || facultad === "todas" || p.carrera?.facultad?.id === facultad)
|
||||||
) {
|
) {
|
||||||
unique.set(p.carrera.id, p.carrera.nombre)
|
unique.set(p.carrera.id, p.carrera.nombre)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return Array.from(unique.entries())
|
return Array.from(unique.entries())
|
||||||
}, [planes, facultad])
|
}, [planes, facultad])
|
||||||
|
|
||||||
|
|
||||||
// NEW: Clonado individual
|
// NEW: Clonado individual
|
||||||
@@ -256,12 +256,6 @@ const carrerasList = useMemo(() => {
|
|||||||
return Array.from(s).sort((a, b) => (a === '—' ? 1 : 0) - (b === '—' ? 1 : 0) || Number(a) - Number(b))
|
return Array.from(s).sort((a, b) => (a === '—' ? 1 : 0) - (b === '—' ? 1 : 0) || Number(a) - Number(b))
|
||||||
}, [asignaturas])
|
}, [asignaturas])
|
||||||
|
|
||||||
const tipos = useMemo(() => {
|
|
||||||
const s = new Set<string>()
|
|
||||||
asignaturas.forEach(a => s.add(a.tipo ?? '—'))
|
|
||||||
return Array.from(s).sort()
|
|
||||||
}, [asignaturas])
|
|
||||||
|
|
||||||
// Salud
|
// Salud
|
||||||
const salud = useMemo(() => {
|
const salud = useMemo(() => {
|
||||||
let sinBibliografia = 0, sinCriterios = 0, sinContenidos = 0
|
let sinBibliografia = 0, sinCriterios = 0, sinContenidos = 0
|
||||||
@@ -274,29 +268,29 @@ const carrerasList = useMemo(() => {
|
|||||||
}, [asignaturas])
|
}, [asignaturas])
|
||||||
|
|
||||||
const filtered = useMemo(() => {
|
const filtered = useMemo(() => {
|
||||||
const t = q.trim().toLowerCase()
|
const t = q.trim().toLowerCase()
|
||||||
return asignaturas.filter(a => {
|
return asignaturas.filter(a => {
|
||||||
const matchesQ =
|
const matchesQ =
|
||||||
!t ||
|
!t ||
|
||||||
[a.nombre, a.clave, a.tipo, a.objetivos, a.plan?.nombre, a.plan?.carrera?.nombre, a.plan?.carrera?.facultad?.nombre]
|
[a.nombre, a.clave, a.tipo, a.objetivos, a.plan?.nombre, a.plan?.carrera?.nombre, a.plan?.carrera?.facultad?.nombre]
|
||||||
.filter(Boolean)
|
.filter(Boolean)
|
||||||
.some(v => String(v).toLowerCase().includes(t))
|
.some(v => String(v).toLowerCase().includes(t))
|
||||||
|
|
||||||
const semOK = sem === 'todos' || String(a.semestre ?? '—') === sem
|
const semOK = sem === 'todos' || String(a.semestre ?? '—') === sem
|
||||||
const tipoOK = tipo === 'todos' || (a.tipo ?? '—') === tipo
|
const tipoOK = tipo === 'todos' || (a.tipo ?? '—') === tipo
|
||||||
const carreraOK = carrera === 'todas' || a.plan?.carrera?.id === carrera
|
const carreraOK = carrera === 'todas' || a.plan?.carrera?.id === carrera
|
||||||
const facultadOK = facultad === 'todas' || a.plan?.carrera?.facultad?.id === facultad
|
const facultadOK = facultad === 'todas' || a.plan?.carrera?.facultad?.id === facultad
|
||||||
const planOK = !search.planId || a.plan?.id === search.planId
|
const planOK = !search.planId || a.plan?.id === search.planId
|
||||||
|
|
||||||
const flagOK =
|
const flagOK =
|
||||||
!flag ||
|
!flag ||
|
||||||
(flag === 'sinBibliografia' && (!a.bibliografia || a.bibliografia.length === 0)) ||
|
(flag === 'sinBibliografia' && (!a.bibliografia || a.bibliografia.length === 0)) ||
|
||||||
(flag === 'sinCriterios' && (!a.criterios_evaluacion || !a.criterios_evaluacion.trim())) ||
|
(flag === 'sinCriterios' && (!a.criterios_evaluacion || !a.criterios_evaluacion.trim())) ||
|
||||||
(flag === 'sinContenidos' && (!a.contenidos || Object.keys(a.contenidos ?? {}).length === 0))
|
(flag === 'sinContenidos' && (!a.contenidos || Object.keys(a.contenidos ?? {}).length === 0))
|
||||||
|
|
||||||
return matchesQ && semOK && tipoOK && flagOK && carreraOK && facultadOK && planOK
|
return matchesQ && semOK && tipoOK && flagOK && carreraOK && facultadOK && planOK
|
||||||
})
|
})
|
||||||
}, [q, sem, tipo, flag, carrera, facultad, asignaturas])
|
}, [q, sem, tipo, flag, carrera, facultad, asignaturas])
|
||||||
|
|
||||||
|
|
||||||
// Agrupación
|
// Agrupación
|
||||||
@@ -316,18 +310,19 @@ const carrerasList = useMemo(() => {
|
|||||||
}, [filtered, groupBy])
|
}, [filtered, groupBy])
|
||||||
|
|
||||||
// Helpers
|
// Helpers
|
||||||
const clearFilters = () => { setQ(''); setSem('todos'); setTipo('todos'); setCarrera('todas'); setFlag('') ; setFacultad('todas')
|
const clearFilters = () => {
|
||||||
|
setQ(''); setSem('todos'); setTipo('todos'); setCarrera('todas'); setFlag(''); setFacultad('todas')
|
||||||
// Actualiza la URL limpiando todos los query params
|
// Actualiza la URL limpiando todos los query params
|
||||||
router.navigate({
|
router.navigate({
|
||||||
to: '/asignaturas',
|
to: '/asignaturas',
|
||||||
search: {
|
search: {
|
||||||
q: '',
|
q: '',
|
||||||
planId: '',
|
planId: '',
|
||||||
carreraId: '',
|
carreraId: '',
|
||||||
facultadId: '',
|
facultadId: '',
|
||||||
f: ''
|
f: ''
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// NEW: util para clonar 1 asignatura
|
// NEW: util para clonar 1 asignatura
|
||||||
@@ -363,7 +358,7 @@ const carrerasList = useMemo(() => {
|
|||||||
if (error) throw error
|
if (error) throw error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// NEW: abrir modal clon individual
|
// NEW: abrir modal clon individual
|
||||||
function openClone(a: Asignatura) {
|
function openClone(a: Asignatura) {
|
||||||
@@ -550,7 +545,12 @@ const carrerasList = useMemo(() => {
|
|||||||
value={search.planId ?? "todos"}
|
value={search.planId ?? "todos"}
|
||||||
onValueChange={(val) => {
|
onValueChange={(val) => {
|
||||||
router.navigate({
|
router.navigate({
|
||||||
search: { ...search, planId: val === "todos" ? "" : val },
|
to: '/asignaturas',
|
||||||
|
search: {
|
||||||
|
...search,
|
||||||
|
planId: val === 'todos' ? '' : val,
|
||||||
|
},
|
||||||
|
replace: true,
|
||||||
})
|
})
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -828,15 +828,14 @@ function AsignaturaCard({ a, onClone, onAddToCart }: { a: Asignatura; onClone: (
|
|||||||
const horasT = a.horas_teoricas ?? 0
|
const horasT = a.horas_teoricas ?? 0
|
||||||
const horasP = a.horas_practicas ?? 0
|
const horasP = a.horas_practicas ?? 0
|
||||||
const meta = tipoMeta(a.tipo)
|
const meta = tipoMeta(a.tipo)
|
||||||
const FacIcon = (Icons as any)[a.plan?.carrera?.facultad?.icon ?? 'Building2'] || Icons.Building2
|
|
||||||
console.log(a);
|
console.log(a);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li className="group relative overflow-hidden rounded-2xl border bg-white/75 dark:bg-neutral-900/60 shadow-sm hover:shadow-md transition-all"
|
<li className="group relative overflow-hidden rounded-2xl border bg-white/75 dark:bg-neutral-900/60 shadow-sm hover:shadow-md transition-all"
|
||||||
style={{
|
style={{
|
||||||
borderColor: a.plan?.carrera?.facultad?.color ?? '#ccc',
|
borderColor: a.plan?.carrera?.facultad?.color ?? '#ccc',
|
||||||
backgroundColor: `${a.plan?.carrera?.facultad?.color}15`, // 15 = transparencia HEX
|
backgroundColor: `${a.plan?.carrera?.facultad?.color}15`, // 15 = transparencia HEX
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div className="p-3">
|
<div className="p-3">
|
||||||
<div className="flex items-start gap-3">
|
<div className="flex items-start gap-3">
|
||||||
@@ -890,15 +889,15 @@ function AsignaturaCard({ a, onClone, onAddToCart }: { a: Asignatura; onClone: (
|
|||||||
{a.plan.carrera && (
|
{a.plan.carrera && (
|
||||||
<InfoChip
|
<InfoChip
|
||||||
icon={<Icons.GraduationCap className="h-3 w-3" />}
|
icon={<Icons.GraduationCap className="h-3 w-3" />}
|
||||||
label={a.plan.carrera.nombre}
|
label={a.plan.carrera.nombre}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{a.plan.carrera?.facultad && (
|
{a.plan.carrera?.facultad && (
|
||||||
<InfoChip
|
<InfoChip
|
||||||
icon={<Icons.Building2 className="h-3 w-3" />}
|
icon={<Icons.Building2 className="h-3 w-3" />}
|
||||||
label={a.plan.carrera.facultad.nombre}
|
label={a.plan.carrera.facultad.nombre}
|
||||||
tint={a.plan.carrera.facultad.color}
|
tint={a.plan.carrera.facultad.color}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
|
|||||||
import { Badge } from "@/components/ui/badge"
|
import { Badge } from "@/components/ui/badge"
|
||||||
import { AcademicSections, planKeys } from "@/components/planes/academic-sections"
|
import { AcademicSections, planKeys } from "@/components/planes/academic-sections"
|
||||||
import { GradientMesh } from "../../../components/planes/GradientMesh"
|
import { GradientMesh } from "../../../components/planes/GradientMesh"
|
||||||
import { asignaturaExtraOptions, asignaturaKeys, asignaturasCountOptions, asignaturasPreviewOptions, planByIdOptions, type AsignaturaLite, type PlanFull } from "@/components/planes/planQueries"
|
import { asignaturaExtraOptions, asignaturaKeys, asignaturasPreviewOptions, planByIdOptions, type AsignaturaLite, type PlanFull } from "@/components/planes/planQueries"
|
||||||
import { softAccentStyle } from "@/components/planes/planHelpers"
|
import { softAccentStyle } from "@/components/planes/planHelpers"
|
||||||
import { Dialog, DialogContent, DialogDescription, DialogTitle } from "@/components/ui/dialog"
|
import { Dialog, DialogContent, DialogDescription, DialogTitle } from "@/components/ui/dialog"
|
||||||
import { DialogFooter, DialogHeader } from "@/components/ui/dialog"
|
import { DialogFooter, DialogHeader } from "@/components/ui/dialog"
|
||||||
|
|||||||
@@ -17,13 +17,6 @@ import { toast } from "sonner"
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* -------------------- Tipos -------------------- */
|
|
||||||
|
|
||||||
|
|
||||||
const SCOPED_ROLES = ["director_facultad", "secretario_academico", "jefe_carrera"] as const
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* -------------------- Query Keys & Fetcher -------------------- */
|
/* -------------------- Query Keys & Fetcher -------------------- */
|
||||||
const usersKeys = {
|
const usersKeys = {
|
||||||
@@ -149,35 +142,6 @@ function RouteComponent() {
|
|||||||
carrera_id?: string | null
|
carrera_id?: string | null
|
||||||
}>({ email: "", password: "" })
|
}>({ email: "", password: "" })
|
||||||
|
|
||||||
function genPassword() {
|
|
||||||
/*
|
|
||||||
Supabase requiere que las contraseñas tengan las siguientes características:
|
|
||||||
- Mínimo de 6 caracteres
|
|
||||||
- Debe contener al menos una letra minúscula
|
|
||||||
- Debe contener al menos una letra mayúscula
|
|
||||||
- Debe contener al menos un número
|
|
||||||
- Debe contener al menos un carácter especial
|
|
||||||
Para garantizar la seguridad, generaremos contraseñas de 12 caracteres en vez del mínimo de 6
|
|
||||||
*/
|
|
||||||
|
|
||||||
// 1. Generar una permutación de los números de 1 al 12 con el método Fisher-Yates
|
|
||||||
|
|
||||||
const positions = Array.from({ length: 12 }, (_, i) => i);
|
|
||||||
for (let i = positions.length - 1; i > 0; i--) {
|
|
||||||
const j = Math.floor(Math.random() * (i + 1));
|
|
||||||
[positions[i], positions[j]] = [positions[j], positions[i]];
|
|
||||||
}
|
|
||||||
|
|
||||||
// 2. Las correspondencias son las siguientes:
|
|
||||||
// - El primer número indica la posición de la letra minúscula
|
|
||||||
// - El segundo número indica la posición de la letra mayúscula
|
|
||||||
// - El tercer número indica la posición del número
|
|
||||||
// - El cuarto número indica la posición del carácter especial
|
|
||||||
// - En las demás posiciones puede haber cualquier caracter alfanumérico
|
|
||||||
|
|
||||||
const s = Array.from(crypto.getRandomValues(new Uint32Array(4))).map((n) => n.toString(36)).join("")
|
|
||||||
return s.slice(0, 14)
|
|
||||||
}
|
|
||||||
|
|
||||||
function RolePill({ role }: { role: Role }) {
|
function RolePill({ role }: { role: Role }) {
|
||||||
const meta = ROLE_META[role]
|
const meta = ROLE_META[role]
|
||||||
@@ -197,61 +161,6 @@ function RouteComponent() {
|
|||||||
router.invalidate()
|
router.invalidate()
|
||||||
}
|
}
|
||||||
|
|
||||||
const upsertNombramiento = useMutation({
|
|
||||||
mutationFn: async (opts: {
|
|
||||||
user_id: string
|
|
||||||
puesto: "director_facultad" | "secretario_academico" | "jefe_carrera"
|
|
||||||
facultad_id?: string | null
|
|
||||||
carrera_id?: string | null
|
|
||||||
}) => {
|
|
||||||
// cierra vigentes
|
|
||||||
if (opts.puesto === "jefe_carrera") {
|
|
||||||
if (!opts.carrera_id) throw new Error("Selecciona carrera")
|
|
||||||
await supabase
|
|
||||||
.from("nombramientos")
|
|
||||||
.update({ hasta: new Date().toISOString().slice(0, 10) })
|
|
||||||
.eq("puesto", "jefe_carrera")
|
|
||||||
.eq("carrera_id", opts.carrera_id)
|
|
||||||
.is("hasta", null)
|
|
||||||
} else {
|
|
||||||
if (!opts.facultad_id) throw new Error("Selecciona facultad")
|
|
||||||
await supabase
|
|
||||||
.from("nombramientos")
|
|
||||||
.update({ hasta: new Date().toISOString().slice(0, 10) })
|
|
||||||
.eq("puesto", opts.puesto)
|
|
||||||
.eq("facultad_id", opts.facultad_id)
|
|
||||||
.is("hasta", null)
|
|
||||||
}
|
|
||||||
const { error } = await supabase.from("nombramientos").insert({
|
|
||||||
user_id: opts.user_id,
|
|
||||||
puesto: opts.puesto,
|
|
||||||
facultad_id: opts.facultad_id ?? null,
|
|
||||||
carrera_id: opts.carrera_id ?? null,
|
|
||||||
desde: new Date().toISOString().slice(0, 10),
|
|
||||||
hasta: null,
|
|
||||||
})
|
|
||||||
if (error) throw error
|
|
||||||
},
|
|
||||||
onError: (e: any) => toast.error(e?.message || "Error al registrar nombramiento"),
|
|
||||||
})
|
|
||||||
|
|
||||||
const toggleBan = useMutation({
|
|
||||||
mutationFn: async (u: UserClaims) => {
|
|
||||||
throw new Error("Funcionalidad de baneo no implementada aún.")
|
|
||||||
const banned = false // cuando se tenga acceso a ese campo
|
|
||||||
// const banned = !!u.banned_until && new Date(u.banned_until) > new Date()
|
|
||||||
const payload = banned ? { banned_until: null } : { banned_until: new Date(Date.now() + 100 * 365 * 24 * 60 * 60 * 1000).toISOString() }
|
|
||||||
// const { error } = await supabase.auth.admin.updateUserById(u.id, payload as any)
|
|
||||||
// if (error) throw new Error(error.message)
|
|
||||||
return !banned
|
|
||||||
},
|
|
||||||
onSuccess: async (isBanned) => {
|
|
||||||
toast.success(isBanned ? "Usuario baneado" : "Usuario desbaneado")
|
|
||||||
await invalidateAll()
|
|
||||||
},
|
|
||||||
onError: (e: any) => toast.error(e?.message || "Error al cambiar estado de baneo"),
|
|
||||||
})
|
|
||||||
|
|
||||||
const createUser = useMutation({
|
const createUser = useMutation({
|
||||||
mutationFn: async (payload: typeof createForm) => {
|
mutationFn: async (payload: typeof createForm) => {
|
||||||
// Validaciones previas
|
// Validaciones previas
|
||||||
@@ -409,7 +318,7 @@ function RouteComponent() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center gap-1">
|
<div className="flex items-center gap-1">
|
||||||
<Button variant="outline" size="sm" onClick={() => toggleBan.mutate(u)} title={banned ? "Restaurar acceso" : "Inhabilitar la cuenta"} className="hidden sm:inline-flex">
|
<Button variant="outline" size="sm" onClick={() => {}} title={banned ? "Restaurar acceso" : "Inhabilitar la cuenta"} className="hidden sm:inline-flex">
|
||||||
<Icons.BanIcon className="w-4 h-4 mr-1" /> {banned ? "Restaurar acceso" : "Inhabilitar la cuenta"}
|
<Icons.BanIcon className="w-4 h-4 mr-1" /> {banned ? "Restaurar acceso" : "Inhabilitar la cuenta"}
|
||||||
</Button>
|
</Button>
|
||||||
<Button variant="ghost" size="sm" className="hidden sm:inline-flex shrink-0" onClick={() => openEdit(u)}>
|
<Button variant="ghost" size="sm" className="hidden sm:inline-flex shrink-0" onClick={() => openEdit(u)}>
|
||||||
@@ -425,7 +334,7 @@ function RouteComponent() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="sm:hidden self-start shrink-0 flex gap-1">
|
<div className="sm:hidden self-start shrink-0 flex gap-1">
|
||||||
<Button variant="outline" size="icon" onClick={() => toggleBan.mutate(u)} aria-label="Ban/Unban"><Icons.BanIcon className="w-4 h-4" /></Button>
|
<Button variant="outline" size="icon" onClick={() => {}} aria-label="Ban/Unban"><Icons.BanIcon className="w-4 h-4" /></Button>
|
||||||
<Button variant="ghost" size="icon" onClick={() => openEdit(u)} aria-label="Editar"><Icons.Pencil className="w-4 h-4" /></Button>
|
<Button variant="ghost" size="icon" onClick={() => openEdit(u)} aria-label="Editar"><Icons.Pencil className="w-4 h-4" /></Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -138,7 +138,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.animate-aurora {
|
.animate-aurora {
|
||||||
background: radial-gradient(at 20% 30%, oklch(27.5% 0.13488 262.73), transparent 50%),
|
background: radial-gradient(at 20% 30%, oklch(27.5% 0.13488 262.73), transparent 50%),
|
||||||
radial-gradient(at 80% 70%, oklch(0.704 0.191 22.216), transparent 50%),
|
radial-gradient(at 80% 70%, oklch(0.704 0.191 22.216), transparent 50%),
|
||||||
|
|||||||
14
staticwebapp.config.json
Normal file
14
staticwebapp.config.json
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
"navigationFallback": {
|
||||||
|
"rewrite": "/index.html",
|
||||||
|
"exclude": [
|
||||||
|
"/assets/*",
|
||||||
|
"/*.css",
|
||||||
|
"/*.js",
|
||||||
|
"/*.ico",
|
||||||
|
"/*.png",
|
||||||
|
"/*.jpg",
|
||||||
|
"/*.svg"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
12
swa-cli.config.json
Normal file
12
swa-cli.config.json
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://aka.ms/azure/static-web-apps-cli/schema",
|
||||||
|
"configurations": {
|
||||||
|
"acad-ia": {
|
||||||
|
"appLocation": ".",
|
||||||
|
"outputLocation": "dist",
|
||||||
|
"appBuildCommand": "npm run build",
|
||||||
|
"run": "npm run dev",
|
||||||
|
"appDevserverUrl": "http://localhost:5173"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"include": ["**/*.ts", "**/*.tsx"],
|
"include": ["**/*.ts", "**/*.tsx", "src/components/ai/AIChatModal.jsx", "src/components/ai/AIChatModal.js"],
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"target": "ES2022",
|
"target": "ES2022",
|
||||||
"jsx": "react-jsx",
|
"jsx": "react-jsx",
|
||||||
|
|||||||
Reference in New Issue
Block a user