Compare commits
9 Commits
a6efb496db
...
feat/reest
| Author | SHA1 | Date | |
|---|---|---|---|
| 1475a65938 | |||
| 9a1d8279a1 | |||
| 0456a1063d | |||
| a41136a224 | |||
| 102c21927e | |||
| 566e23ad34 | |||
| 872c495d30 | |||
| 7951f9d8c5 | |||
| 4894543c57 |
@@ -1,4 +0,0 @@
|
||||
VITE_SUPABASE_URL=http://127.0.0.1:54321
|
||||
VITE_SUPABASE_ANON_KEY=eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6ImV4ZGtzc3Vyem1qbm5oZ3RpYW1hIiwicm9sZSI6ImFub24iLCJpYXQiOjE3NDEzNzg2MzIsImV4cCI6MjA1Njk1NDYzMn0.g1mBmsw-i6F6e-tPv5gWkHZacyPM2Y9X0fiKVYmVYKE
|
||||
#VITE_BACK_ORIGIN=http://localhost:3001
|
||||
VITE_BACK_ORIGIN=http://localhost:3001
|
||||
25
package-lock.json
generated
25
package-lock.json
generated
@@ -30,6 +30,7 @@
|
||||
"@tanstack/router-plugin": "^1.121.2",
|
||||
"@types/canvas-confetti": "^1.9.0",
|
||||
"canvas-confetti": "^1.9.3",
|
||||
"carbone-sdk-js": "^1.2.2",
|
||||
"class-variance-authority": "^0.7.1",
|
||||
"clsx": "^2.1.1",
|
||||
"cmdk": "^1.1.1",
|
||||
@@ -2819,6 +2820,12 @@
|
||||
"node": ">=10.0.0"
|
||||
}
|
||||
},
|
||||
"node_modules/carbone-sdk-js": {
|
||||
"version": "1.2.2",
|
||||
"resolved": "https://registry.npmjs.org/carbone-sdk-js/-/carbone-sdk-js-1.2.2.tgz",
|
||||
"integrity": "sha512-3bc4F04DizC8ULB6j6JsOq8G0sW/pwdwvfbbZ8exBZbcOCV4WE8KHsY6GiED/3tmH++Z1I5XMppjkBhjg60zjA==",
|
||||
"license": "Apache-2.0"
|
||||
},
|
||||
"node_modules/ccount": {
|
||||
"version": "2.0.1",
|
||||
"license": "MIT",
|
||||
@@ -3372,6 +3379,20 @@
|
||||
"node": ">=8"
|
||||
}
|
||||
},
|
||||
"node_modules/fsevents": {
|
||||
"version": "2.3.3",
|
||||
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
|
||||
"integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
|
||||
"hasInstallScript": true,
|
||||
"license": "MIT",
|
||||
"optional": true,
|
||||
"os": [
|
||||
"darwin"
|
||||
],
|
||||
"engines": {
|
||||
"node": "^8.16.0 || ^10.6.0 || >=11.0.0"
|
||||
}
|
||||
},
|
||||
"node_modules/gensync": {
|
||||
"version": "1.0.0-beta.2",
|
||||
"license": "MIT",
|
||||
@@ -5514,7 +5535,9 @@
|
||||
}
|
||||
},
|
||||
"node_modules/vite": {
|
||||
"version": "6.3.5",
|
||||
"version": "6.4.1",
|
||||
"resolved": "https://registry.npmjs.org/vite/-/vite-6.4.1.tgz",
|
||||
"integrity": "sha512-+Oxm7q9hDoLMyJOYfUYBuHQo+dkAloi33apOPP56pzj+vsdJDzr+j1NISE5pyaAuKL4A3UD34qd0lx5+kfKp2g==",
|
||||
"license": "MIT",
|
||||
"peer": true,
|
||||
"dependencies": {
|
||||
|
||||
@@ -34,6 +34,7 @@
|
||||
"@tanstack/router-plugin": "^1.121.2",
|
||||
"@types/canvas-confetti": "^1.9.0",
|
||||
"canvas-confetti": "^1.9.3",
|
||||
"carbone-sdk-js": "^1.2.2",
|
||||
"class-variance-authority": "^0.7.1",
|
||||
"clsx": "^2.1.1",
|
||||
"cmdk": "^1.1.1",
|
||||
|
||||
@@ -1,15 +1,24 @@
|
||||
import { useRouter } from "@tanstack/react-router"
|
||||
import { useSupabaseAuth } from "@/auth/supabase"
|
||||
import { useState, useEffect, useCallback } from "react"
|
||||
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter } from "@/components/ui/dialog"
|
||||
import { Label } from "@/components/ui/label"
|
||||
import { Textarea } from "@/components/ui/textarea"
|
||||
import { Input } from "@/components/ui/input"
|
||||
import { CarreraCombobox, FacultadCombobox } from "@/components/users/procedencia-combobox"
|
||||
import { Button } from "@/components/ui/button"
|
||||
import { postAPI } from "@/lib/api"
|
||||
import { supabase } from "@/auth/supabase"
|
||||
import { DetailDialog } from "@/components/archivos/DetailDialog"
|
||||
import { useRouter } from "@tanstack/react-router";
|
||||
import { useSupabaseAuth } from "@/auth/supabase";
|
||||
import { useState, useEffect, useCallback } from "react";
|
||||
import {
|
||||
Dialog,
|
||||
DialogContent,
|
||||
DialogHeader,
|
||||
DialogTitle,
|
||||
DialogFooter,
|
||||
} from "@/components/ui/dialog";
|
||||
import { Label } from "@/components/ui/label";
|
||||
import { Textarea } from "@/components/ui/textarea";
|
||||
import { Input } from "@/components/ui/input";
|
||||
import {
|
||||
CarreraCombobox,
|
||||
FacultadCombobox,
|
||||
} from "@/components/users/procedencia-combobox";
|
||||
import { Button } from "@/components/ui/button";
|
||||
import { postAPI } from "@/lib/api";
|
||||
import { supabase } from "@/auth/supabase";
|
||||
import { DetailDialog } from "@/components/archivos/DetailDialog";
|
||||
import type { RefRow } from "@/types/RefRow";
|
||||
|
||||
// ————————————————————————————————————————————————————————————————
|
||||
@@ -50,42 +59,51 @@ function extIcon(ext: string) {
|
||||
// ————————————————————————————————————————————————————————————————
|
||||
// Componente principal
|
||||
// ————————————————————————————————————————————————————————————————
|
||||
export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpenChange: (v: boolean) => void }) {
|
||||
const router = useRouter()
|
||||
const auth = useSupabaseAuth()
|
||||
const role = auth.claims?.role
|
||||
export function CreatePlanDialog({
|
||||
open,
|
||||
onOpenChange,
|
||||
}: {
|
||||
open: boolean;
|
||||
onOpenChange: (v: boolean) => void;
|
||||
}) {
|
||||
const router = useRouter();
|
||||
const auth = useSupabaseAuth();
|
||||
const role = auth.claims?.role;
|
||||
|
||||
const [saving, setSaving] = useState(false)
|
||||
const [err, setErr] = useState<string | null>(null)
|
||||
const [saving, setSaving] = useState(false);
|
||||
const [err, setErr] = useState<string | null>(null);
|
||||
|
||||
const [facultadId, setFacultadId] = useState(auth.claims?.facultad_id ?? "")
|
||||
const [carreraId, setCarreraId] = useState(auth.claims?.carrera_id ?? "")
|
||||
const [nivel, setNivel] = useState("")
|
||||
const [facultadId, setFacultadId] = useState(auth.claims?.facultad_id ?? "");
|
||||
const [carreraId, setCarreraId] = useState(auth.claims?.carrera_id ?? "");
|
||||
const [nivel, setNivel] = useState("");
|
||||
const [prompt, setPrompt] = useState(
|
||||
"Genera un plan de estudios claro y realista: "
|
||||
)
|
||||
);
|
||||
|
||||
const [dbFiles, setDbFiles] = useState<{
|
||||
id: string;
|
||||
titulo: string;
|
||||
s3_file_path: string;
|
||||
fecha_subida?: string;
|
||||
tags?: string[];
|
||||
}[]>([])
|
||||
const [selectedFiles, setSelectedFiles] = useState<string[]>([])
|
||||
const [lastSelectedIndex, setLastSelectedIndex] = useState<number | null>(null)
|
||||
const [searchTerm, setSearchTerm] = useState("")
|
||||
const [currentPage, setCurrentPage] = useState(1)
|
||||
const itemsPerPage = 10
|
||||
const debouncedSearchTerm = useDebounce(searchTerm, 300)
|
||||
const [dbFiles, setDbFiles] = useState<
|
||||
{
|
||||
id: string;
|
||||
titulo: string;
|
||||
s3_file_path: string;
|
||||
fecha_subida?: string;
|
||||
tags?: string[];
|
||||
}[]
|
||||
>([]);
|
||||
const [selectedFiles, setSelectedFiles] = useState<string[]>([]);
|
||||
const [lastSelectedIndex, setLastSelectedIndex] = useState<number | null>(
|
||||
null
|
||||
);
|
||||
const [searchTerm, setSearchTerm] = useState("");
|
||||
const [currentPage, setCurrentPage] = useState(1);
|
||||
const itemsPerPage = 10;
|
||||
const debouncedSearchTerm = useDebounce(searchTerm, 300);
|
||||
const totalPages = Math.ceil(dbFiles.length / itemsPerPage);
|
||||
|
||||
const [previewRow, setPreviewRow] = useState<RefRow | null>(null);
|
||||
|
||||
const lockFacultad = role === "secretario_academico" || role === "jefe_carrera"
|
||||
const lockCarrera = role === "jefe_carrera"
|
||||
|
||||
|
||||
const lockFacultad =
|
||||
role === "secretario_academico" || role === "jefe_carrera";
|
||||
const lockCarrera = role === "jefe_carrera";
|
||||
|
||||
useEffect(() => {
|
||||
async function fetchDbFiles() {
|
||||
@@ -94,20 +112,25 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
.from("documentos")
|
||||
.select("documentos_id, titulo_archivo, fecha_subida, tags")
|
||||
.ilike("titulo_archivo", `%${debouncedSearchTerm}%`)
|
||||
.range((currentPage - 1) * itemsPerPage, currentPage * itemsPerPage - 1);
|
||||
.range(
|
||||
(currentPage - 1) * itemsPerPage,
|
||||
currentPage * itemsPerPage - 1
|
||||
);
|
||||
|
||||
if (error) {
|
||||
console.error("Error fetching files from database:", error);
|
||||
return;
|
||||
}
|
||||
|
||||
setDbFiles((data || []).map((file: any) => ({
|
||||
id: file.documentos_id,
|
||||
titulo: file.titulo_archivo,
|
||||
s3_file_path: `prueba-referencias/documento_${file.documentos_id}.pdf`,
|
||||
fecha_subida: file.fecha_subida,
|
||||
tags: file.tags || [],
|
||||
})));
|
||||
setDbFiles(
|
||||
(data || []).map((file: any) => ({
|
||||
id: file.documentos_id,
|
||||
titulo: file.titulo_archivo,
|
||||
s3_file_path: `prueba-referencias/documento_${file.documentos_id}.pdf`,
|
||||
fecha_subida: file.fecha_subida,
|
||||
tags: file.tags || [],
|
||||
}))
|
||||
);
|
||||
} catch (err) {
|
||||
console.error("Unexpected error fetching files:", err);
|
||||
}
|
||||
@@ -116,41 +139,59 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
if (open) fetchDbFiles();
|
||||
}, [open, debouncedSearchTerm, currentPage]);
|
||||
|
||||
const isSelected = useCallback((path: string) => selectedFiles.includes(path), [selectedFiles]);
|
||||
const isSelected = useCallback(
|
||||
(path: string) => selectedFiles.includes(path),
|
||||
[selectedFiles]
|
||||
);
|
||||
|
||||
const toggleSelected = useCallback((id: string) => {
|
||||
setSelectedFiles(prev => prev.includes(id) ? prev.filter(p => p !== id) : [...prev, id]);
|
||||
setSelectedFiles((prev) =>
|
||||
prev.includes(id) ? prev.filter((p) => p !== id) : [...prev, id]
|
||||
);
|
||||
}, []);
|
||||
|
||||
const replaceSelection = useCallback((id: string) => {
|
||||
setSelectedFiles([id]);
|
||||
}, []);
|
||||
|
||||
const rangeSelect = useCallback((start: number, end: number) => {
|
||||
const [s, e] = start < end ? [start, end] : [end, start];
|
||||
const ids = dbFiles.slice(s, e + 1).map(f => f.id);
|
||||
setSelectedFiles(prev => Array.from(new Set([...prev, ...ids])));
|
||||
}, [dbFiles]);
|
||||
const rangeSelect = useCallback(
|
||||
(start: number, end: number) => {
|
||||
const [s, e] = start < end ? [start, end] : [end, start];
|
||||
const ids = dbFiles.slice(s, e + 1).map((f) => f.id);
|
||||
setSelectedFiles((prev) => Array.from(new Set([...prev, ...ids])));
|
||||
},
|
||||
[dbFiles]
|
||||
);
|
||||
|
||||
const handleCardClick = useCallback((e: React.MouseEvent, index: number, file: { id: string }) => {
|
||||
const id = file.id;
|
||||
const handleCardClick = useCallback(
|
||||
(e: React.MouseEvent, index: number, file: { id: string }) => {
|
||||
const id = file.id;
|
||||
|
||||
if (e.shiftKey && lastSelectedIndex !== null) {
|
||||
rangeSelect(lastSelectedIndex, index);
|
||||
} else if (e.metaKey || e.ctrlKey) {
|
||||
toggleSelected(id);
|
||||
setLastSelectedIndex(index);
|
||||
} else {
|
||||
if (isSelected(id) && selectedFiles.length === 1) {
|
||||
// si ya es el único seleccionado, des-selecciona
|
||||
setSelectedFiles([]);
|
||||
setLastSelectedIndex(null);
|
||||
} else {
|
||||
replaceSelection(id);
|
||||
if (e.shiftKey && lastSelectedIndex !== null) {
|
||||
rangeSelect(lastSelectedIndex, index);
|
||||
} else if (e.metaKey || e.ctrlKey) {
|
||||
toggleSelected(id);
|
||||
setLastSelectedIndex(index);
|
||||
} else {
|
||||
if (isSelected(id) && selectedFiles.length === 1) {
|
||||
// si ya es el único seleccionado, des-selecciona
|
||||
setSelectedFiles([]);
|
||||
setLastSelectedIndex(null);
|
||||
} else {
|
||||
replaceSelection(id);
|
||||
setLastSelectedIndex(index);
|
||||
}
|
||||
}
|
||||
}
|
||||
}, [isSelected, lastSelectedIndex, rangeSelect, replaceSelection, selectedFiles.length, toggleSelected]);
|
||||
},
|
||||
[
|
||||
isSelected,
|
||||
lastSelectedIndex,
|
||||
rangeSelect,
|
||||
replaceSelection,
|
||||
selectedFiles.length,
|
||||
toggleSelected,
|
||||
]
|
||||
);
|
||||
|
||||
const clearSelection = () => {
|
||||
setSelectedFiles([]);
|
||||
@@ -158,30 +199,51 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
};
|
||||
|
||||
async function crearConIA() {
|
||||
setErr(null)
|
||||
if (!carreraId) { setErr("Selecciona una carrera."); return }
|
||||
setSaving(true)
|
||||
setErr(null);
|
||||
if (!carreraId) {
|
||||
setErr("Selecciona una carrera.");
|
||||
return;
|
||||
}
|
||||
setSaving(true);
|
||||
try {
|
||||
const res = await postAPI("/api/generar/plan", {
|
||||
carreraId,
|
||||
prompt: prompt,
|
||||
insert: true,
|
||||
files: selectedFiles,
|
||||
created_by: auth.user?.id,
|
||||
})
|
||||
const newId = (res as any)?.id || (res as any)?.plan?.id || (res as any)?.data?.id
|
||||
|
||||
const {
|
||||
data: { session },
|
||||
} = await supabase.auth.getSession();
|
||||
const token = session?.access_token;
|
||||
|
||||
const { data, error } = await supabase.functions.invoke(
|
||||
"crear-plan-estudios",
|
||||
{
|
||||
headers: { Authorization: `Bearer ${token}` },
|
||||
body: {
|
||||
carrera_id: carreraId,
|
||||
prompt_usuario: prompt,
|
||||
insert: true,
|
||||
archivos_a_usar: [],
|
||||
},
|
||||
}
|
||||
);
|
||||
if (error) throw error;
|
||||
|
||||
const res = data;
|
||||
|
||||
const newId =
|
||||
(res as any)?.id || (res as any)?.plan?.id || (res as any)?.data?.id;
|
||||
if (newId) {
|
||||
onOpenChange(false)
|
||||
router.invalidate()
|
||||
router.navigate({ to: "/plan/$planId", params: { planId: newId } })
|
||||
onOpenChange(false);
|
||||
router.invalidate();
|
||||
router.navigate({ to: "/plan/$planId", params: { planId: newId } });
|
||||
} else {
|
||||
onOpenChange(false)
|
||||
router.invalidate()
|
||||
onOpenChange(false);
|
||||
router.invalidate();
|
||||
}
|
||||
} catch (e: any) {
|
||||
setErr(typeof e?.message === "string" ? e.message : "Error al generar el plan.")
|
||||
setErr(
|
||||
typeof e?.message === "string" ? e.message : "Error al generar el plan."
|
||||
);
|
||||
} finally {
|
||||
setSaving(false)
|
||||
setSaving(false);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -192,7 +254,9 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
||||
<DialogContent className="min-w-[65vw] max-w-4xl max-h-[90vh] overflow-y-auto">
|
||||
<DialogHeader>
|
||||
<DialogTitle className="font-mono">Nuevo plan de estudios (IA)</DialogTitle>
|
||||
<DialogTitle className="font-mono">
|
||||
Nuevo plan de estudios (IA)
|
||||
</DialogTitle>
|
||||
</DialogHeader>
|
||||
|
||||
<div className="grid gap-4 md:grid-cols-2">
|
||||
@@ -215,7 +279,10 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
<Label>Facultad</Label>
|
||||
<FacultadCombobox
|
||||
value={facultadId}
|
||||
onChange={(id) => { setFacultadId(id); setCarreraId("") }}
|
||||
onChange={(id) => {
|
||||
setFacultadId(id);
|
||||
setCarreraId("");
|
||||
}}
|
||||
disabled={lockFacultad}
|
||||
placeholder="Elige una facultad…"
|
||||
/>
|
||||
@@ -228,7 +295,11 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
value={carreraId}
|
||||
onChange={setCarreraId}
|
||||
disabled={!facultadId || lockCarrera}
|
||||
placeholder={facultadId ? "Elige una carrera…" : "Selecciona una facultad primero"}
|
||||
placeholder={
|
||||
facultadId
|
||||
? "Elige una carrera…"
|
||||
: "Selecciona una facultad primero"
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
|
||||
@@ -246,11 +317,19 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
<div className="text-sm text-neutral-600">
|
||||
{selectedFiles.length > 0 ? (
|
||||
<span>
|
||||
{selectedFiles.length} seleccionado{selectedFiles.length > 1 ? 's' : ''}
|
||||
<button className="ml-3 underline hover:no-underline" onClick={clearSelection}>Limpiar</button>
|
||||
{selectedFiles.length} seleccionado
|
||||
{selectedFiles.length > 1 ? "s" : ""}
|
||||
<button
|
||||
className="ml-3 underline hover:no-underline"
|
||||
onClick={clearSelection}
|
||||
>
|
||||
Limpiar
|
||||
</button>
|
||||
</span>
|
||||
) : (
|
||||
<span>Tip: ⇧ para seleccionar rango, ⌘/Ctrl para múltiples.</span>
|
||||
<span>
|
||||
Tip: ⇧ para seleccionar rango, ⌘/Ctrl para múltiples.
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
@@ -258,12 +337,15 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
{/* Grid de archivos con selección tipo file manager */}
|
||||
<div className="md:col-span-2 space-y-1">
|
||||
<Label>Archivos de referencia (opcional)</Label>
|
||||
<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) => {
|
||||
const ext = fileExt(file.titulo);
|
||||
const selected = isSelected(file.id);
|
||||
console.log(file);
|
||||
|
||||
|
||||
return (
|
||||
<div
|
||||
key={file.id}
|
||||
@@ -285,10 +367,10 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
fecha_subida: file.fecha_subida ?? null,
|
||||
tags: file.tags ?? null,
|
||||
instrucciones: "",
|
||||
})
|
||||
});
|
||||
}}
|
||||
onKeyDown={(e) => {
|
||||
if (e.key === 'Enter' || e.key === ' ') {
|
||||
if (e.key === "Enter" || e.key === " ") {
|
||||
e.preventDefault();
|
||||
handleCardClick(e as any, index, file);
|
||||
}
|
||||
@@ -296,31 +378,51 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
className={[
|
||||
"group relative rounded-2xl border bg-white p-4 text-left shadow-sm transition",
|
||||
"hover:shadow-md focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-blue-500",
|
||||
selected ? "border-blue-500 ring-2 ring-blue-500 shadow-md" : "border-neutral-200 hover:border-neutral-300",
|
||||
].join(' ')}
|
||||
selected
|
||||
? "border-blue-500 ring-2 ring-blue-500 shadow-md"
|
||||
: "border-neutral-200 hover:border-neutral-300",
|
||||
].join(" ")}
|
||||
>
|
||||
{/* Outline animado tipo file manager */}
|
||||
<span className={[
|
||||
"pointer-events-none absolute inset-0 rounded-2xl ring-2 ring-blue-500",
|
||||
"opacity-0 group-aria-selected:opacity-100 group-focus:opacity-100 transition-opacity",
|
||||
].join(' ')} />
|
||||
<span
|
||||
className={[
|
||||
"pointer-events-none absolute inset-0 rounded-2xl ring-2 ring-blue-500",
|
||||
"opacity-0 group-aria-selected:opacity-100 group-focus:opacity-100 transition-opacity",
|
||||
].join(" ")}
|
||||
/>
|
||||
|
||||
<div className="flex items-start gap-3">
|
||||
<div className="flex h-10 w-10 shrink-0 items-center justify-center rounded-xl border bg-neutral-50">
|
||||
<span className="text-lg" aria-hidden>{extIcon(ext)}</span>
|
||||
<span className="text-lg" aria-hidden>
|
||||
{extIcon(ext)}
|
||||
</span>
|
||||
</div>
|
||||
<div className="min-w-0 flex-1">
|
||||
<h3 className="font-semibold text-sm md:text-base truncate" title={file.titulo}>{file.titulo}</h3>
|
||||
<h3
|
||||
className="font-semibold text-sm md:text-base truncate"
|
||||
title={file.titulo}
|
||||
>
|
||||
{file.titulo}
|
||||
</h3>
|
||||
{file.fecha_subida ? (
|
||||
<p className="text-xs text-neutral-600">{new Date(file.fecha_subida).toLocaleDateString()}</p>
|
||||
<p className="text-xs text-neutral-600">
|
||||
{new Date(file.fecha_subida).toLocaleDateString()}
|
||||
</p>
|
||||
) : (
|
||||
<p className="text-xs text-neutral-500">Fecha desconocida</p>
|
||||
<p className="text-xs text-neutral-500">
|
||||
Fecha desconocida
|
||||
</p>
|
||||
)}
|
||||
|
||||
|
||||
{file.tags && file.tags.length > 0 && (
|
||||
<div className="mt-2 flex flex-wrap gap-1.5">
|
||||
{file.tags.map((tag, i) => (
|
||||
<span key={i} className="text-[10px] px-2 py-0.5 bg-neutral-100 text-neutral-700 rounded-full">#{tag}</span>
|
||||
<span
|
||||
key={i}
|
||||
className="text-[10px] px-2 py-0.5 bg-neutral-100 text-neutral-700 rounded-full"
|
||||
>
|
||||
#{tag}
|
||||
</span>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
@@ -347,50 +449,69 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
fecha_subida: file.fecha_subida ?? null,
|
||||
tags: file.tags ?? null,
|
||||
instrucciones: "",
|
||||
})
|
||||
});
|
||||
}}
|
||||
>Previsualizar</Button>
|
||||
>
|
||||
Previsualizar
|
||||
</Button>
|
||||
</div>
|
||||
|
||||
{/* Footer compacto */}
|
||||
<div className="mt-4 flex items-center justify-between text-xs text-neutral-600">
|
||||
<span className="truncate">{ext.toUpperCase()}</span>
|
||||
{selected ? <span className="font-medium">Seleccionado</span> : <span className="opacity-60">Click para seleccionar</span>}
|
||||
{selected ? (
|
||||
<span className="font-medium">Seleccionado</span>
|
||||
) : (
|
||||
<span className="opacity-60">
|
||||
Click para seleccionar
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
);
|
||||
})}
|
||||
|
||||
{dbFiles.length === 0 && (
|
||||
<p className="text-sm text-neutral-500">No se encontraron archivos.</p>
|
||||
<p className="text-sm text-neutral-500">
|
||||
No se encontraron archivos.
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/* Paginación mejorada */}
|
||||
{dbFiles.length > itemsPerPage && (
|
||||
<div className="mt-4 flex flex-wrap items-center justify-between gap-3">
|
||||
<div className="text-sm text-neutral-700">Página {currentPage} de {totalPages}</div>
|
||||
<div className="text-sm text-neutral-700">
|
||||
Página {currentPage} de {totalPages}
|
||||
</div>
|
||||
<div className="flex items-center gap-2">
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
disabled={currentPage === 1}
|
||||
onClick={() => setCurrentPage((p) => Math.max(p - 1, 1))}
|
||||
>Anterior</Button>
|
||||
>
|
||||
Anterior
|
||||
</Button>
|
||||
<Input
|
||||
className="h-8 w-16 text-center"
|
||||
value={currentPage}
|
||||
onChange={(e) => {
|
||||
const v = parseInt(e.target.value || '1', 10);
|
||||
if (!isNaN(v)) setCurrentPage(Math.min(Math.max(v, 1), totalPages));
|
||||
const v = parseInt(e.target.value || "1", 10);
|
||||
if (!isNaN(v))
|
||||
setCurrentPage(Math.min(Math.max(v, 1), totalPages));
|
||||
}}
|
||||
/>
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
disabled={currentPage === totalPages}
|
||||
onClick={() => setCurrentPage((p) => Math.min(p + 1, totalPages))}
|
||||
>Siguiente</Button>
|
||||
onClick={() =>
|
||||
setCurrentPage((p) => Math.min(p + 1, totalPages))
|
||||
}
|
||||
>
|
||||
Siguiente
|
||||
</Button>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
@@ -400,19 +521,26 @@ export function CreatePlanDialog({ open, onOpenChange }: { open: boolean; onOpen
|
||||
{err && <div className="text-sm text-red-600">{err}</div>}
|
||||
|
||||
<DialogFooter className="flex flex-col-reverse sm:flex-row gap-2">
|
||||
<Button variant="outline" className="w-full sm:w-auto" onClick={() => onOpenChange(false)}>Cancelar</Button>
|
||||
<Button className="w-full sm:w-auto" onClick={crearConIA} disabled={saving}>
|
||||
<Button
|
||||
variant="outline"
|
||||
className="w-full sm:w-auto"
|
||||
onClick={() => onOpenChange(false)}
|
||||
>
|
||||
Cancelar
|
||||
</Button>
|
||||
<Button
|
||||
className="w-full sm:w-auto"
|
||||
onClick={crearConIA}
|
||||
disabled={saving}
|
||||
>
|
||||
{saving ? "Generando…" : "Generar y crear"}
|
||||
</Button>
|
||||
</DialogFooter>
|
||||
</DialogContent>
|
||||
|
||||
{previewRow && (
|
||||
<DetailDialog
|
||||
row={previewRow}
|
||||
onClose={() => setPreviewRow(null)}
|
||||
/>
|
||||
<DetailDialog row={previewRow} onClose={() => setPreviewRow(null)} />
|
||||
)}
|
||||
</Dialog>
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
@@ -1,428 +1,145 @@
|
||||
import { jsPDF } from "jspdf"
|
||||
import { Button } from "../ui/button"
|
||||
import { Download } from "lucide-react"
|
||||
// Importamos 'react' para poder usar el hook de estado si fuera necesario.
|
||||
import { supabase } from "@/auth/supabase";
|
||||
import { Button } from "../ui/button";
|
||||
import { Download } from "lucide-react";
|
||||
|
||||
/**
|
||||
* Tipo mínimo para el plan. Hemos añadido 'number' a la unión
|
||||
* para permitir propiedades como 'total_creditos' que son numéricas,
|
||||
* lo cual resuelve el error de asignación con PlanFull.
|
||||
*/
|
||||
export type PlanLike = Record<string, string | number | object | null | undefined> // CORREGIDO: Se agregó 'object'
|
||||
export type PlanLike = Record<
|
||||
string,
|
||||
string | number | object | null | undefined
|
||||
>;
|
||||
|
||||
// Usamos el tipo corregido PlanLike en la prop 'plan'
|
||||
export function DownloadPlanPDF({ plan }: { plan: PlanLike }) {
|
||||
// console.log(plan) // Mantener el log para debug
|
||||
export function DownloadPlanPDF({ plan }: { plan: Record<string, any> }) {
|
||||
async function fetchPDF() {
|
||||
const planObj = {
|
||||
...plan,
|
||||
nivel_y_nombre_del_plan_de_estudios: `${plan["nivel"]} en ${plan["nombre"]}`,
|
||||
nivel: undefined,
|
||||
nombre: undefined,
|
||||
};
|
||||
const fileName = `Plan_${planObj.nivel_y_nombre_del_plan_de_estudios || "Desconocido"}.pdf`;
|
||||
// const jsonData = JSON.stringify(planObj);
|
||||
|
||||
function generatePDF() {
|
||||
// Inicialización del documento
|
||||
const doc = new jsPDF({
|
||||
orientation: "portrait",
|
||||
unit: "mm",
|
||||
format: "letter",
|
||||
})
|
||||
console.log(plan);
|
||||
|
||||
const pageWidth = doc.internal.pageSize.getWidth()
|
||||
const pageHeight = doc.internal.pageSize.getHeight()
|
||||
const margin = 20
|
||||
const maxWidth = pageWidth - margin * 2
|
||||
|
||||
// Parámetros de estilo institucional (basados en las capturas)
|
||||
const lineHeight = 5.0 // mm por línea (ajustado para más texto por página)
|
||||
const sectionGap = 10 // Espacio entre recuadros de sección
|
||||
const bodyFontSize = 10.5
|
||||
const headingFontSize = 12
|
||||
const subHeadingFontSize = 10
|
||||
const bulletGlifo = "\u21D2" // Flecha doble (⇒) para el glifo
|
||||
const bulletIndent = 6 // Sangría para el texto de la lista
|
||||
const triggerDownload = (blob: Blob, name: string) => {
|
||||
const url = URL.createObjectURL(blob);
|
||||
const a = document.createElement("a");
|
||||
a.href = url;
|
||||
a.setAttribute("download", name);
|
||||
document.body.appendChild(a);
|
||||
a.click();
|
||||
a.remove();
|
||||
URL.revokeObjectURL(url);
|
||||
};
|
||||
|
||||
let cursorY = margin
|
||||
|
||||
// Variable para controlar si ya se dibujaron todas las secciones (para la última caja)
|
||||
let totalSections = 0;
|
||||
let drawnSections = 0;
|
||||
const fetchBinaryFallback = async () => {
|
||||
// Intenta construir la URL del runtime de Functions
|
||||
const anyClient = supabase as any;
|
||||
const baseUrl =
|
||||
anyClient?.functions?.url ||
|
||||
`${(anyClient?.supabaseUrl || "").replace(/\/$/, "")}/functions/v1`;
|
||||
const { data: sess } = await supabase.auth.getSession();
|
||||
const token = sess?.session?.access_token;
|
||||
|
||||
// --- Utilidades de Dibujo ---
|
||||
console.log(JSON.stringify(planObj, null, 2));
|
||||
console.log(planObj);
|
||||
|
||||
|
||||
|
||||
// Dibuja el encabezado ("Anexo 1") y pie de página (Numeración) en todas las páginas
|
||||
const drawHeaderAndFooter = () => {
|
||||
// FIX: Usamos (doc as any) para acceder a getNumberOfPages() y evitar el error de TS
|
||||
const pageCount = (doc as any).internal.getNumberOfPages()
|
||||
for (let i = 1; i <= pageCount; i++) {
|
||||
doc.setPage(i)
|
||||
|
||||
// Encabezado (Anexo 1)
|
||||
doc.setFont("helvetica", "normal")
|
||||
doc.setFontSize(10)
|
||||
doc.text("Anexo 1", pageWidth - margin, margin - 5, { align: "right" })
|
||||
const resp = await fetch(`${baseUrl}/carbone-io-api`, {
|
||||
method: "POST",
|
||||
headers: {
|
||||
"Content-Type": "application/json",
|
||||
Accept: "application/pdf",
|
||||
...(token ? { Authorization: `Bearer ${token}` } : {}),
|
||||
},
|
||||
body: JSON.stringify({
|
||||
action: "downloadReport",
|
||||
templateId: "1302213091201757023",
|
||||
fileName,
|
||||
convertTo: "pdf",
|
||||
data: planObj,
|
||||
}),
|
||||
});
|
||||
|
||||
// Pie de página (Numeración)
|
||||
// Usamos el mismo tamaño y posición que en el ejemplo
|
||||
doc.setFontSize(8)
|
||||
doc.text(
|
||||
`Página ${i} de ${pageCount}`,
|
||||
pageWidth - margin, // Posicionado a la derecha
|
||||
pageHeight - 10,
|
||||
{ align: "right" }
|
||||
)
|
||||
}
|
||||
// Regresar al último estado de la página para continuar dibujando
|
||||
doc.setPage(pageCount)
|
||||
}
|
||||
if (!resp.ok) throw new Error(`HTTP ${resp.status}`);
|
||||
const blob = await resp.blob();
|
||||
triggerDownload(blob, fileName);
|
||||
};
|
||||
|
||||
// Verifica si se necesita una nueva página antes de dibujar una línea o un elemento.
|
||||
const addPageIfNeeded = (neededHeight: number = lineHeight) => {
|
||||
// Aseguramos que haya espacio para la altura necesaria + un poco de margen de seguridad
|
||||
// El margen de seguridad ayuda a que la línea de pie de página no se solape
|
||||
if (cursorY + neededHeight > pageHeight - 15) {
|
||||
doc.addPage()
|
||||
cursorY = margin
|
||||
// El encabezado "Anexo 1" se dibuja al final en drawHeaderAndFooter()
|
||||
}
|
||||
}
|
||||
try {
|
||||
// const { data, error } = await supabase.functions.invoke(
|
||||
// "carbone-io-api",
|
||||
// {
|
||||
// method: "POST",
|
||||
// headers: { Accept: "application/octet-stream" }, // preferir binario
|
||||
// body: {
|
||||
// action: "downloadReport",
|
||||
// templateId: "1302213091201757023",
|
||||
// fileName,
|
||||
// convertTo: "pdf",
|
||||
// data: planObj,
|
||||
// },
|
||||
// }
|
||||
// );
|
||||
|
||||
/**
|
||||
* Dibuja un título de sección con el estilo de recuadro gris (como en las capturas).
|
||||
* Retorna la altura ocupada por el recuadro para el cálculo de la altura total de la sección.
|
||||
*/
|
||||
const drawHeadingBox = (text: string, marginTop: number = 0): number => {
|
||||
doc.setFont("helvetica", "bold")
|
||||
doc.setFontSize(headingFontSize)
|
||||
|
||||
// Espacio antes del título
|
||||
cursorY += marginTop
|
||||
|
||||
const titleLines = doc.splitTextToSize(text.toUpperCase(), maxWidth - 4) // Pequeño padding
|
||||
const titleHeight = titleLines.length * lineHeight + 4 // Texto + padding vertical
|
||||
// if (error) throw error;
|
||||
|
||||
// 1. Verificar si el recuadro cabe en la página
|
||||
addPageIfNeeded(titleHeight + 5) // 5mm de margen de seguridad
|
||||
|
||||
// 2. Dibujar Recuadro Gris (Relleno)
|
||||
doc.setFillColor(230, 230, 230) // Gris claro
|
||||
doc.rect(margin, cursorY, maxWidth, titleHeight, "F")
|
||||
|
||||
// 3. Dibujar texto centrado
|
||||
const textX = pageWidth / 2
|
||||
const textY = cursorY + titleHeight / 2 + 0.8 // 0.8mm para centrado óptico
|
||||
doc.text(titleLines, textX, textY, { align: "center" })
|
||||
// // Si ya viene binario, descargar directo
|
||||
// if (typeof Blob !== "undefined" && data instanceof Blob) {
|
||||
// triggerDownload(data, fileName);
|
||||
// return;
|
||||
// }
|
||||
// if (data instanceof ArrayBuffer) {
|
||||
// triggerDownload(
|
||||
// new Blob([data], { type: "application/pdf" }),
|
||||
// fileName
|
||||
// );
|
||||
// return;
|
||||
// }
|
||||
|
||||
cursorY += titleHeight // Avanzar el cursor justo después del recuadro
|
||||
return titleHeight
|
||||
}
|
||||
// // Si vino como string (ej. empieza con %PDF), usa el fallback binario
|
||||
// if (typeof data === "string") {
|
||||
// await fetchBinaryFallback();
|
||||
// return;
|
||||
// }
|
||||
|
||||
/**
|
||||
* Dibuja un bloque de texto (párrafo o lista) manejando el salto de página línea por línea,
|
||||
* y envuelto en un recuadro.
|
||||
*/
|
||||
const drawContentBox = (text?: string | null, isList: boolean = false, isLastSection: boolean = false) => {
|
||||
// Manejamos 'text' que ahora puede ser string o number
|
||||
const content = (text !== null && text !== undefined) ? String(text).trim() : "Sin información."
|
||||
|
||||
doc.setFont("helvetica", "normal")
|
||||
doc.setFontSize(bodyFontSize)
|
||||
|
||||
let initialY = cursorY // Guardar Y inicial para dibujar el recuadro final
|
||||
// // Si vino JSON con base64, decodificar y descargar
|
||||
// if (data && typeof data === "object") {
|
||||
// const b64 =
|
||||
// (data as any).file || (data as any).buffer || (data as any).base64;
|
||||
// if (typeof b64 === "string") {
|
||||
// const clean = b64.replace(/^data:.*;base64,/, "");
|
||||
// const binary = atob(clean);
|
||||
// const bytes = new Uint8Array(binary.length);
|
||||
// for (let i = 0; i < binary.length; i++)
|
||||
// bytes[i] = binary.charCodeAt(i);
|
||||
// triggerDownload(
|
||||
// new Blob([bytes], { type: "application/pdf" }),
|
||||
// fileName
|
||||
// );
|
||||
// return;
|
||||
// }
|
||||
// }
|
||||
|
||||
// El contenido se dibuja en un recuadro. Dejamos un padding interno.
|
||||
const innerMargin = margin + 2
|
||||
const innerMaxWidth = maxWidth - 4
|
||||
let currentContentY = cursorY + 2 // Iniciar con 2mm de padding superior
|
||||
// console.warn("Respuesta no reconocida para descarga de PDF.", {
|
||||
// type: typeof data,
|
||||
// });
|
||||
|
||||
// Dividir el contenido en bloques (párrafos o ítems de lista)
|
||||
const blocks = isList ?
|
||||
content.split('\n').filter(line => line.trim().length > 0) :
|
||||
content.split('\n').filter(line => line.trim().length > 0)
|
||||
|
||||
let contentDrawn = false
|
||||
|
||||
for (const block of blocks) {
|
||||
let cleanBlock = block.trim()
|
||||
|
||||
// Si es lista, limpiamos los posibles marcadores (1., *, -)
|
||||
if (isList) {
|
||||
cleanBlock = cleanBlock.replace(/^(\d+\.|\*|[\-\•]|\u27A2|\u21D2)\s*/, '').trim()
|
||||
}
|
||||
|
||||
if (!cleanBlock) continue
|
||||
|
||||
// Líneas que componen el bloque actual
|
||||
const textWidth = isList ? innerMaxWidth - bulletIndent : innerMaxWidth
|
||||
const lines = doc.splitTextToSize(cleanBlock, textWidth)
|
||||
|
||||
for (let i = 0; i < lines.length; i++) {
|
||||
// 1. Verificar si se necesita un salto de página ANTES de dibujar la línea
|
||||
if (currentContentY + lineHeight > pageHeight - 15) {
|
||||
// Cierra el recuadro en la página actual
|
||||
doc.rect(margin, initialY, maxWidth, pageHeight - 15 - initialY)
|
||||
|
||||
doc.addPage()
|
||||
|
||||
// En la nueva página, el punto de inicio del recuadro es el margen superior
|
||||
initialY = margin
|
||||
currentContentY = margin + 2 // Iniciar con padding
|
||||
cursorY = margin // El cursorY global se actualiza para la siguiente sección/línea
|
||||
}
|
||||
|
||||
const currentLine = lines[i]
|
||||
|
||||
if (isList && i === 0) {
|
||||
// Dibujar el glifo solo en la primera línea del ítem
|
||||
doc.text(bulletGlifo, innerMargin, currentContentY)
|
||||
doc.text(currentLine, innerMargin + bulletIndent, currentContentY)
|
||||
} else if (isList && i > 0) {
|
||||
// Dibujar líneas subsiguientes con sangría (sin glifo)
|
||||
doc.text(currentLine, innerMargin + bulletIndent, currentContentY)
|
||||
} else {
|
||||
// Dibujar párrafo normal
|
||||
doc.text(currentLine, innerMargin, currentContentY)
|
||||
}
|
||||
|
||||
currentContentY += lineHeight // Avanzar el cursor de contenido
|
||||
}
|
||||
|
||||
// Espacio entre ítems de lista o entre párrafos
|
||||
currentContentY += isList ? 1.5 : 4
|
||||
contentDrawn = true
|
||||
}
|
||||
|
||||
// 2. Después de dibujar todo el contenido, dibujar el recuadro exterior
|
||||
if (contentDrawn) {
|
||||
let finalY = currentContentY - 2 // Ajuste final de padding y espacio
|
||||
|
||||
// FIX: Usamos (doc as any) para acceder a los métodos internos y evitar el error de TS
|
||||
if (isLastSection &&
|
||||
(doc as any).internal.getCurrentPageInfo().pageNumber === (doc as any).internal.getNumberOfPages()) {
|
||||
|
||||
// Si es la ÚLTIMA sección Y estamos en la ÚLTIMA página,
|
||||
// forzamos el recuadro a ir hasta el final (pageHeight - 15)
|
||||
finalY = pageHeight - 15;
|
||||
}
|
||||
|
||||
// Dibujar el recuadro completo (desde el Y inicial guardado hasta el Y final)
|
||||
doc.rect(margin, initialY, maxWidth, finalY - initialY)
|
||||
|
||||
cursorY = finalY + sectionGap // Actualizar el cursor global para la siguiente sección
|
||||
} else {
|
||||
// Si no se dibuja contenido, solo saltar la altura del recuadro vacío.
|
||||
doc.rect(margin, initialY, maxWidth, 10) // Dibuja una caja vacía de 10mm
|
||||
cursorY += 10 + sectionGap
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// --- Portada (Estilo Institucional) ---
|
||||
|
||||
const drawTitlePage = () => {
|
||||
cursorY = 40 // Empezar más abajo
|
||||
|
||||
// UNIVERSIDAD LA SALLE
|
||||
doc.setFont("helvetica", "bold")
|
||||
doc.setFontSize(14)
|
||||
doc.text("UNIVERSIDAD LA SALLE", pageWidth / 2, cursorY, { align: "center" })
|
||||
cursorY += 5
|
||||
|
||||
// Separador horizontal
|
||||
doc.line(margin, cursorY, pageWidth - margin, cursorY)
|
||||
cursorY += 15
|
||||
|
||||
// LICENCIATURA EN INGENIERÍA CIBERNÉTICA Y SISTEMAS COMPUTACIONALES
|
||||
doc.setFontSize(18)
|
||||
// Manejamos la conversión a string si es necesario
|
||||
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)
|
||||
doc.text(mainTitleLines, pageWidth / 2, cursorY, { align: "center" })
|
||||
cursorY += mainTitleLines.length * 8
|
||||
|
||||
// Nivel y Nombre del Plan de Estudios
|
||||
doc.setFont("helvetica", "normal")
|
||||
doc.setFontSize(10)
|
||||
doc.text("Nivel y Nombre del Plan de Estudios", pageWidth / 2, cursorY, { align: "center" })
|
||||
cursorY += 5
|
||||
|
||||
// Separador horizontal
|
||||
doc.line(margin, cursorY, pageWidth - margin, cursorY)
|
||||
cursorY += 10
|
||||
|
||||
// Escolar / Presencial (Modalidad Educativa)
|
||||
doc.setFont("helvetica", "bold")
|
||||
doc.setFontSize(14)
|
||||
doc.text("Escolar / Presencial", pageWidth / 2, cursorY, { align: "center" })
|
||||
doc.setFont("helvetica", "normal")
|
||||
doc.setFontSize(10)
|
||||
cursorY += 5
|
||||
doc.text("Modalidad Educativa", pageWidth / 2, cursorY, { align: "center" })
|
||||
cursorY += 15
|
||||
|
||||
// Recuadros de Vigencia, Antecedente y Área (Simulación del Layout)
|
||||
|
||||
// Recuadro Vigencia (Parte superior central)
|
||||
const boxWidth = maxWidth * 0.5
|
||||
const boxX = (pageWidth - boxWidth) / 2
|
||||
const boxY = cursorY
|
||||
doc.rect(boxX, boxY, boxWidth, 20)
|
||||
doc.rect(boxX, boxY + 15, boxWidth, 5)
|
||||
doc.setFontSize(10)
|
||||
doc.text("Vigencia", boxX + boxWidth / 2, boxY + 18, { align: "center" })
|
||||
cursorY += 30 // Espacio para el primer recuadro
|
||||
|
||||
// Recuadro Antecedente Académico (Izquierda)
|
||||
const smallBoxWidth = maxWidth * 0.4
|
||||
const smallBoxHeight = 35
|
||||
const smallBoxX1 = margin
|
||||
doc.rect(smallBoxX1, cursorY, smallBoxWidth, smallBoxHeight)
|
||||
doc.rect(smallBoxX1, cursorY + smallBoxHeight - 5, smallBoxWidth, 5)
|
||||
doc.setFontSize(10)
|
||||
doc.text("Bachillerato", smallBoxX1 + smallBoxWidth / 2, cursorY + smallBoxHeight / 2, { align: "center" })
|
||||
doc.text("Antecedente Académico", smallBoxX1 + smallBoxWidth / 2, cursorY + smallBoxHeight - 2, { align: "center" })
|
||||
|
||||
// Recuadro Área de Estudio (Derecha)
|
||||
const smallBoxX2 = pageWidth - margin - smallBoxWidth
|
||||
doc.rect(smallBoxX2, cursorY, smallBoxWidth, smallBoxHeight)
|
||||
doc.rect(smallBoxX2, cursorY + smallBoxHeight - 5, smallBoxWidth, 5)
|
||||
doc.setFontSize(10)
|
||||
doc.text("Ingeniería, manufactura y construcción", smallBoxX2 + smallBoxWidth / 2, cursorY + smallBoxHeight / 2, { align: "center" })
|
||||
doc.text("Área de Estudio", smallBoxX2 + smallBoxWidth / 2, cursorY + smallBoxHeight - 2, { align: "center" })
|
||||
cursorY += smallBoxHeight + 10
|
||||
|
||||
// Datos Fijos (Abajo)
|
||||
doc.setFont("helvetica", "normal")
|
||||
doc.setFontSize(10)
|
||||
|
||||
const drawDataPair = (label: string, value: string) => {
|
||||
const labelX = margin
|
||||
const valueX = margin + maxWidth * 0.4
|
||||
doc.text(label + ":", labelX, cursorY)
|
||||
doc.setFont("helvetica", "bold")
|
||||
doc.text(value, valueX, cursorY)
|
||||
doc.setFont("helvetica", "normal")
|
||||
cursorY += 5
|
||||
}
|
||||
|
||||
drawDataPair("Clave del Plan de Estudios", "2020")
|
||||
drawDataPair("Diseño Curricular", "Rígido")
|
||||
// Usamos plan.total_ciclos si existe
|
||||
drawDataPair("Total de Ciclos del Plan de Estudios", plan["total_ciclos"] ? String(plan["total_ciclos"]) : "9")
|
||||
drawDataPair("Duración del Ciclo Escolar", "16 semanas")
|
||||
drawDataPair("Carga Horaria a la Semana", "27")
|
||||
|
||||
// Pie de página institucional (simulado)
|
||||
doc.text(
|
||||
"Dirección de Asuntos Académicos - Anexo 1",
|
||||
pageWidth / 2,
|
||||
pageHeight - margin,
|
||||
{ align: "center" }
|
||||
)
|
||||
}
|
||||
|
||||
|
||||
// --- Ejecución Principal ---
|
||||
|
||||
// 1. Dibuja la portada
|
||||
drawTitlePage()
|
||||
|
||||
// 2. Comienza el contenido del plan en la nueva página
|
||||
doc.addPage()
|
||||
cursorY = margin
|
||||
|
||||
// Las secciones se ajustan a las que generas, pero también a las adicionales del documento de referencia
|
||||
const SECTIONS: Array<{ key: string; title: string; isList: boolean }> = [
|
||||
{ key: "objetivo_general", title: "Objetivo General", isList: false },
|
||||
// La sección FIN DE APRENDIZAJE O FORMACIÓN es el Objetivo General del documento institucional, la mapearemos aquí.
|
||||
{ key: "fin_aprendizaje", title: "FIN DE APRENDIZAJE O FORMACIÓN", isList: false }, // Mapea al objetivo general
|
||||
|
||||
{ key: "perfil_ingreso", title: "PERFIL DE INGRESO", isList: true },
|
||||
{ key: "perfil_egreso", title: "PERFIL DE EGRESO", isList: true },
|
||||
{ key: "competencias_genericas", title: "COMPETENCIAS GENÉRICAS", isList: true },
|
||||
{ key: "competencias_especificas", title: "COMPETENCIAS ESPECÍFICAS", isList: true },
|
||||
{ key: "indicadores_desempeno", title: "INDICADORES DE DESEMPEÑO", isList: true },
|
||||
{ key: "sistema_evaluacion", title: "SISTEMA DE EVALUACIÓN", isList: false },
|
||||
{ key: "pertinencia", title: "PERTINENCIA", isList: false },
|
||||
|
||||
// Nuevas secciones basadas en las imágenes que suelen ir con "No aplica"
|
||||
{ key: "administracion", title: "ADMINISTRACIÓN Y OPERATIVIDAD DEL PLAN DE ESTUDIOS", isList: false },
|
||||
{ key: "sustento_teorico", title: "SUSTENTO TEÓRICO DEL MODELO CURRICULAR", isList: false },
|
||||
{ key: "justificacion_curricular", title: "JUSTIFICACIÓN DE LA PROPUESTA CURRICULAR EN LA MODALIDAD NO ESCOLARIZADA O MIXTA", isList: false },
|
||||
{ key: "programa_investigacion", title: "PROGRAMA DE INVESTIGACIÓN", isList: false },
|
||||
{ key: "curso_propedeutico", title: "CURSO PROPEDÉUTICO", isList: false },
|
||||
{ key: "propuesta_evaluacion", title: "PROPUESTA DE EVALUACIÓN PERIÓDICA DEL PLAN DE ESTUDIOS", isList: false },
|
||||
]
|
||||
|
||||
// Contar el número total de secciones con contenido
|
||||
totalSections = SECTIONS.length;
|
||||
|
||||
for (const s of SECTIONS) {
|
||||
drawnSections++; // Incrementar contador de secciones dibujadas
|
||||
|
||||
// Obtenemos el valor (que puede ser string, number, object, null, o undefined)
|
||||
let value = plan[s.key]
|
||||
|
||||
// Mapeo especial para el objetivo general institucional (si existe)
|
||||
if (s.key === "fin_aprendizaje" && (value === null || value === undefined)) {
|
||||
value = plan["objetivo_general"]
|
||||
}
|
||||
|
||||
// Inicializar content como string para pasarlo a la función de dibujo
|
||||
let content: string | null = null;
|
||||
|
||||
// Si el valor no es nulo/undefined, convertir a string
|
||||
if (value !== null && value !== undefined) {
|
||||
// Si es un objeto, lo ignoramos y usamos un string vacío.
|
||||
// Esto es clave para 'carreras', que no tiene un formato textual.
|
||||
if (typeof value === 'object' && !Array.isArray(value)) {
|
||||
content = "";
|
||||
} else {
|
||||
content = String(value);
|
||||
}
|
||||
}
|
||||
|
||||
// Si el contenido es nulo o vacío, usamos un placeholder común en el documento institucional
|
||||
if (!content || content.trim() === "") {
|
||||
// Para las secciones del plan generado, si no hay contenido, usar "Sin información."
|
||||
if (["objetivo_general", "perfil_ingreso", "perfil_egreso", "competencias_genericas", "competencias_especificas", "indicadores_desempeno", "sistema_evaluacion", "pertinencia"].includes(s.key)) {
|
||||
content = "Sin información."
|
||||
} else {
|
||||
// Para las secciones auxiliares del formato institucional
|
||||
if (s.key === "administracion" || s.key === "sustento_teorico" || s.key === "justificacion_curricular" || s.key === "programa_investigacion") {
|
||||
content = "No aplica"
|
||||
} else if (s.key === "curso_propedeutico") {
|
||||
content = "No tiene"
|
||||
} else if (s.key === "propuesta_evaluacion") {
|
||||
// Texto de la Propuesta de Evaluación (última página)
|
||||
content = "La Universidad La Salle aplica una metodología para la evaluación y modificación de los programas académicos de licenciatura o posgrado que imparte. Los principales niveles, estudios, acciones y plazos que comprende dicha metodología son los siguientes:\n\nNIVEL DE EVALUACIÓN CURRICULAR INTERNA: DIAGNÓSTICO DE ESTRUCTURA Y OPERACIÓN.\n1. Análisis técnico-pedagógico del planteamiento curricular vigente.\n2. Estudio con directivos del área académica correspondiente, para analizar y valorar las problemáticas en la estructura y gestión del programa académico durante el periodo en que se ha desarrollado.\n3. Consulta a profesores sobre: a) problemáticas percibidas en la formación académica, profesional y actitudinal de los estudiantes, b) problemáticas en la operación, c) necesidades sociales, avances disciplinarios y/o tecnológicos detectados en su propio ejercicio profesional, que consideran importante incluir en el planteamiento curricular.\n4. Estudio de opinión de estudiantes sobre las problemáticas que aprecian en la formación que reciben respecto a la operación y estructura del programa académico.\n\nNIVEL DE EVALUACIÓN CURRICULAR EXTERNA: DIAGNÓSTICO DE IMPACTO Y PRÁCTICAS PROFESIONALES.\n5. Estudio sobre el estado del conocimiento en que se encuentran el o los campos disciplinarios vinculados con el programa académico, en México y, de ser posible, en otros países.\n6. Análisis de la oferta y la evolución que, en términos estadísticos, han tenido programas académicos similares en el ámbito de influencia y/o en el país.\n7. Estudio sobre requerimientos y tendencias en la formación, a partir del análisis de criterios, perfiles, estándares y parámetros de organismos evaluadores o acreditadores de programas académicos (si existen para el campo profesional), así como de la comparación general del programa en evaluación con otros similares y prestigiosos, de IES nacionales y, de ser posible, extranjeras.\n8. Estudio con egresados del programa académico para conocer su opinión sobre: a) el mismo programa; b) formación recibida; c) sitios de inserción laboral y características de sus prácticas profesionales, y d) aspectos disciplinarios, tecnológicos y/o actitudinales que, a la luz de su experiencia, consideren necesario incluir como parte de la formación.\n9. Estudio con empleadores para conocer su valoración sobre las prácticas profesionales de los egresados del programa académico, y su apreciación sobre nuevos requerimientos en el campo."
|
||||
} else {
|
||||
continue; // Si sigue siendo nulo, saltar la sección
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Determinar si es la última sección que se dibujará
|
||||
const isLastSection = drawnSections === totalSections;
|
||||
|
||||
// Dibuja el recuadro del título de la sección
|
||||
drawHeadingBox(s.title, sectionGap)
|
||||
|
||||
// Dibuja el contenido de la sección dentro de su recuadro.
|
||||
// Pasamos isLastSection para que drawContentBox sepa si debe forzar el cierre.
|
||||
drawContentBox(content, s.isList, isLastSection)
|
||||
}
|
||||
|
||||
// Finalizar y dibujar encabezados/pies en todas las páginas (se dibuja en el paso final)
|
||||
drawHeaderAndFooter()
|
||||
|
||||
// Guardar el documento
|
||||
const name = (plan["prompt"] ? `Plan_${plan["prompt"]}` : `Plan_de_estudios`).replace(/\s+/g, "_")
|
||||
doc.save(`${name}_Institucional.pdf`)
|
||||
await fetchBinaryFallback();
|
||||
return;
|
||||
} catch (error) {
|
||||
console.error("Error al obtener PDF:", error);
|
||||
}
|
||||
|
||||
return (
|
||||
<Button variant="outline" className="flex items-center gap-2 " onClick={generatePDF}>
|
||||
Descargar PDF
|
||||
<Download className="w-4 h-4" />
|
||||
}
|
||||
|
||||
return (
|
||||
<Button
|
||||
variant="outline"
|
||||
className="flex items-center gap-2"
|
||||
onClick={() => void fetchPDF()}
|
||||
>
|
||||
Descargar PDF
|
||||
<Download className="w-4 h-4" />
|
||||
</Button>
|
||||
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
export default DownloadPlanPDF
|
||||
export default DownloadPlanPDF;
|
||||
|
||||
@@ -1,16 +1,26 @@
|
||||
import * as Icons from "lucide-react"
|
||||
import { useMemo, useState } from "react"
|
||||
import { useSuspenseQuery, useMutation, useQueryClient, queryOptions } from "@tanstack/react-query"
|
||||
import { Button } from "@/components/ui/button"
|
||||
import { Dialog, DialogContent, DialogHeader, DialogTitle, DialogFooter } from "@/components/ui/dialog"
|
||||
import { Textarea } from "@/components/ui/textarea"
|
||||
import { supabase,useSupabaseAuth } from "@/auth/supabase"
|
||||
import { toast } from "sonner"
|
||||
import ReactMarkdown from 'react-markdown'
|
||||
import { HistorialCambiosModal } from "../historico/HistorialCambiosModal"
|
||||
import * as Icons from "lucide-react";
|
||||
import { useMemo, useState } from "react";
|
||||
import {
|
||||
useSuspenseQuery,
|
||||
useMutation,
|
||||
useQueryClient,
|
||||
queryOptions,
|
||||
} from "@tanstack/react-query";
|
||||
import { Button } from "@/components/ui/button";
|
||||
import {
|
||||
Dialog,
|
||||
DialogContent,
|
||||
DialogHeader,
|
||||
DialogTitle,
|
||||
DialogFooter,
|
||||
} from "@/components/ui/dialog";
|
||||
import { Textarea } from "@/components/ui/textarea";
|
||||
import { supabase, useSupabaseAuth } from "@/auth/supabase";
|
||||
import { toast } from "sonner";
|
||||
import ReactMarkdown from "react-markdown";
|
||||
import { HistorialCambiosModal } from "../historico/HistorialCambiosModal";
|
||||
// @ts-ignore
|
||||
import AIChatModal from "../ai/AIChatModal"
|
||||
|
||||
import AIChatModal from "../ai/AIChatModal";
|
||||
|
||||
/* =====================================================
|
||||
Query keys & fetcher
|
||||
@@ -18,33 +28,29 @@ import AIChatModal from "../ai/AIChatModal"
|
||||
export const planKeys = {
|
||||
root: ["plan"] as const,
|
||||
byId: (id: string) => [...planKeys.root, id] as const,
|
||||
}
|
||||
};
|
||||
|
||||
export type PlanTextFields = {
|
||||
objetivo_general?: string | string[] | null
|
||||
sistema_evaluacion?: string | string[] | null
|
||||
perfil_ingreso?: string | string[] | null
|
||||
perfil_egreso?: string | string[] | null
|
||||
competencias_genericas?: string | string[] | null
|
||||
competencias_especificas?: string | string[] | null
|
||||
indicadores_desempeno?: string | string[] | null
|
||||
pertinencia?: string | string[] | null
|
||||
prompt?: string | null
|
||||
historico?: string | null
|
||||
}
|
||||
objetivo_general?: string | string[] | null;
|
||||
sistema_evaluacion?: string | string[] | null;
|
||||
perfil_ingreso?: string | string[] | null;
|
||||
perfil_egreso?: string | string[] | null;
|
||||
competencias_genericas?: string | string[] | null;
|
||||
competencias_especificas?: string | string[] | null;
|
||||
indicadores_desempeno?: string | string[] | null;
|
||||
pertinencia?: string | string[] | null;
|
||||
prompt?: string | null;
|
||||
historico?: string | null;
|
||||
};
|
||||
|
||||
async function fetchPlanText(planId: string): Promise<PlanTextFields> {
|
||||
const { data, error } = await supabase
|
||||
.from("plan_estudios")
|
||||
.select(
|
||||
`objetivo_general, sistema_evaluacion, perfil_ingreso, perfil_egreso,
|
||||
competencias_genericas, competencias_especificas, indicadores_desempeno,
|
||||
pertinencia, prompt`
|
||||
)
|
||||
.select(`*`)
|
||||
.eq("id", planId)
|
||||
.single()
|
||||
if (error) throw error
|
||||
return (data ?? {}) as PlanTextFields
|
||||
.single();
|
||||
if (error) throw error;
|
||||
return (data ?? {}) as PlanTextFields;
|
||||
}
|
||||
|
||||
export const planTextOptions = (planId: string) =>
|
||||
@@ -52,283 +58,503 @@ export const planTextOptions = (planId: string) =>
|
||||
queryKey: planKeys.byId(planId),
|
||||
queryFn: () => fetchPlanText(planId),
|
||||
staleTime: 60_000,
|
||||
})
|
||||
});
|
||||
|
||||
/* =====================================================
|
||||
Color helpers
|
||||
===================================================== */
|
||||
function hexToRgb(hex?: string | null): [number, number, number] {
|
||||
if (!hex) return [37, 99, 235]
|
||||
const h = hex.replace("#", "")
|
||||
const v = h.length === 3 ? h.split("").map((c) => c + c).join("") : h
|
||||
const n = parseInt(v, 16)
|
||||
return [(n >> 16) & 255, (n >> 8) & 255, n & 255]
|
||||
if (!hex) return [37, 99, 235];
|
||||
const h = hex.replace("#", "");
|
||||
const v =
|
||||
h.length === 3
|
||||
? h
|
||||
.split("")
|
||||
.map((c) => c + c)
|
||||
.join("")
|
||||
: h;
|
||||
const n = parseInt(v, 16);
|
||||
return [(n >> 16) & 255, (n >> 8) & 255, n & 255];
|
||||
}
|
||||
const rgba = (rgb: [number, number, number], a: number) => `rgba(${rgb[0]},${rgb[1]},${rgb[2]},${a})`
|
||||
const rgba = (rgb: [number, number, number], a: number) =>
|
||||
`rgba(${rgb[0]},${rgb[1]},${rgb[2]},${a})`;
|
||||
|
||||
/* =====================================================
|
||||
Expandable text
|
||||
===================================================== */
|
||||
function ExpandableText({ text, mono = false }: { text?: string | string[] | null; mono?: boolean }) {
|
||||
const [open, setOpen] = useState(false)
|
||||
function ExpandableText({
|
||||
text,
|
||||
mono = false,
|
||||
}: {
|
||||
text?: string | string[] | null;
|
||||
mono?: boolean;
|
||||
}) {
|
||||
const [open, setOpen] = useState(false);
|
||||
if (!text || (Array.isArray(text) && text.length === 0)) {
|
||||
return <span className="text-neutral-400">—</span>
|
||||
return <span className="text-neutral-400">—</span>;
|
||||
}
|
||||
const content = Array.isArray(text) ? text.join("\n• ") : text
|
||||
const rendered = Array.isArray(text) ? `• ${content}` : content
|
||||
const content = Array.isArray(text) ? text.join("\n• ") : text;
|
||||
const rendered = Array.isArray(text) ? `• ${content}` : content;
|
||||
return (
|
||||
<div>
|
||||
<ReactMarkdown>{rendered}</ReactMarkdown>
|
||||
{String(rendered).length > 220 && (
|
||||
<button onClick={() => setOpen((v) => !v)} className="mt-2 text-xs font-medium text-neutral-600 hover:underline">
|
||||
<button
|
||||
onClick={() => setOpen((v) => !v)}
|
||||
className="mt-2 text-xs font-medium text-neutral-600 hover:underline"
|
||||
>
|
||||
{open ? "Ver menos" : "Ver más"}
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/* =====================================================
|
||||
Section panel
|
||||
===================================================== */
|
||||
function SectionPanel({ title, icon: Icon, color, children, id }: { title: string; icon: any; color?: string | null; children: React.ReactNode; id: string }) {
|
||||
const rgb = hexToRgb(color)
|
||||
function SectionPanel({
|
||||
title,
|
||||
icon: Icon,
|
||||
color,
|
||||
children,
|
||||
id,
|
||||
}: {
|
||||
title: string;
|
||||
icon: any;
|
||||
color?: string | null;
|
||||
children: React.ReactNode;
|
||||
id: string;
|
||||
}) {
|
||||
const rgb = hexToRgb(color);
|
||||
return (
|
||||
<section id={id} className="relative rounded-3xl border overflow-hidden shadow-sm bg-white/70 dark:bg-neutral-900/60">
|
||||
<section
|
||||
id={id}
|
||||
className="relative rounded-3xl border overflow-hidden shadow-sm bg-white/70 dark:bg-neutral-900/60"
|
||||
>
|
||||
<div className="pointer-events-none absolute inset-0 -z-0">
|
||||
<div className="absolute -top-20 -left-20 w-[28rem] h-[28rem] rounded-full blur-3xl" style={{ background: `radial-gradient(circle, ${rgba(rgb, 0.2)}, transparent 60%)` }} />
|
||||
<div className="absolute -bottom-24 -right-16 w-[24rem] h-[24rem] rounded-full blur-[60px]" style={{ background: `radial-gradient(circle, ${rgba(rgb, 0.14)}, transparent 60%)` }} />
|
||||
<div
|
||||
className="absolute -top-20 -left-20 w-[28rem] h-[28rem] rounded-full blur-3xl"
|
||||
style={{
|
||||
background: `radial-gradient(circle, ${rgba(rgb, 0.2)}, transparent 60%)`,
|
||||
}}
|
||||
/>
|
||||
<div
|
||||
className="absolute -bottom-24 -right-16 w-[24rem] h-[24rem] rounded-full blur-[60px]"
|
||||
style={{
|
||||
background: `radial-gradient(circle, ${rgba(rgb, 0.14)}, transparent 60%)`,
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
<div className="relative z-10 px-4 py-3 flex items-center gap-2 border-b" style={{ background: `linear-gradient(180deg, ${rgba(rgb, 0.1)}, transparent)` }}>
|
||||
<span className="inline-flex items-center justify-center rounded-xl border px-2.5 py-2 bg-white/80" style={{ borderColor: rgba(rgb, 0.25) }}>
|
||||
<div
|
||||
className="relative z-10 px-4 py-3 flex items-center gap-2 border-b"
|
||||
style={{
|
||||
background: `linear-gradient(180deg, ${rgba(rgb, 0.1)}, transparent)`,
|
||||
}}
|
||||
>
|
||||
<span
|
||||
className="inline-flex items-center justify-center rounded-xl border px-2.5 py-2 bg-white/80"
|
||||
style={{ borderColor: rgba(rgb, 0.25) }}
|
||||
>
|
||||
<Icon className="w-4 h-4" />
|
||||
</span>
|
||||
<h3 className="font-semibold">{title}</h3>
|
||||
</div>
|
||||
<div className="relative z-10 p-5">{children}</div>
|
||||
</section>
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
/* =====================================================
|
||||
AcademicSections (con React Query)
|
||||
===================================================== */
|
||||
export function AcademicSections({ planId, color }: { planId: string; color?: string | null }) {
|
||||
const qc = useQueryClient()
|
||||
const auth = useSupabaseAuth()
|
||||
const [openHistorial, setOpenHistorial] = useState(false)
|
||||
const [openModalIa, setopenModalIa] = useState(false)
|
||||
if(!planId) return <div>Cargando…</div>
|
||||
const { data: plan } = useSuspenseQuery(planTextOptions(planId))
|
||||
export function AcademicSections({
|
||||
planId,
|
||||
color,
|
||||
}: {
|
||||
planId: string;
|
||||
color?: string | null;
|
||||
}) {
|
||||
const qc = useQueryClient();
|
||||
const auth = useSupabaseAuth();
|
||||
const [openHistorial, setOpenHistorial] = useState(false);
|
||||
const [openModalIa, setopenModalIa] = useState(false);
|
||||
if (!planId) return <div>Cargando…</div>;
|
||||
const { data: plan } = useSuspenseQuery(planTextOptions(planId));
|
||||
|
||||
const [editing, setEditing] = useState<null | { key: keyof PlanTextFields; title: string }>(null)
|
||||
const [draft, setDraft] = useState("")
|
||||
const [editing, setEditing] = useState<null | {
|
||||
key: keyof PlanTextFields;
|
||||
title: string;
|
||||
}>(null);
|
||||
const [draft, setDraft] = useState("");
|
||||
|
||||
// --- mutation con actualización optimista ---
|
||||
const updateField = useMutation({
|
||||
mutationFn: async ({ key, value }: { key: keyof PlanTextFields; value: string | string[] | null }) => {
|
||||
const payload: Record<string, any> = { [key]: value }
|
||||
const { error } = await supabase.from("plan_estudios").update(payload).eq("id", planId)
|
||||
if (error) throw error
|
||||
return payload
|
||||
mutationFn: async ({
|
||||
key,
|
||||
value,
|
||||
}: {
|
||||
key: keyof PlanTextFields;
|
||||
value: string | string[] | null;
|
||||
}) => {
|
||||
const payload: Record<string, any> = { [key]: value };
|
||||
const { error } = await supabase
|
||||
.from("plan_estudios")
|
||||
.update(payload)
|
||||
.eq("id", planId);
|
||||
if (error) throw error;
|
||||
return payload;
|
||||
},
|
||||
onMutate: async ({ key, value }) => {
|
||||
await qc.cancelQueries({ queryKey: planKeys.byId(planId) })
|
||||
const prev = qc.getQueryData<PlanTextFields>(planKeys.byId(planId))
|
||||
qc.setQueryData<PlanTextFields>(planKeys.byId(planId), (old) => ({ ...(old ?? {}), [key]: value }))
|
||||
return { prev }
|
||||
await qc.cancelQueries({ queryKey: planKeys.byId(planId) });
|
||||
const prev = qc.getQueryData<PlanTextFields>(planKeys.byId(planId));
|
||||
qc.setQueryData<PlanTextFields>(planKeys.byId(planId), (old) => ({
|
||||
...(old ?? {}),
|
||||
[key]: value,
|
||||
}));
|
||||
return { prev };
|
||||
},
|
||||
onError: (e, _vars, ctx) => {
|
||||
if (ctx?.prev) qc.setQueryData(planKeys.byId(planId), ctx.prev)
|
||||
toast.error((e as any)?.message || "No se pudo guardar 😓")
|
||||
if (ctx?.prev) qc.setQueryData(planKeys.byId(planId), ctx.prev);
|
||||
toast.error((e as any)?.message || "No se pudo guardar 😓");
|
||||
},
|
||||
onSuccess: () => {
|
||||
toast.success("Guardado ✅")
|
||||
toast.success("Guardado ✅");
|
||||
},
|
||||
onSettled: async () => {
|
||||
await qc.invalidateQueries({ queryKey: planKeys.byId(planId) })
|
||||
await qc.invalidateQueries({ queryKey: planKeys.byId(planId) });
|
||||
},
|
||||
})
|
||||
});
|
||||
|
||||
const sections = useMemo(
|
||||
() => [
|
||||
{ id: "sec-obj", title: "Objetivo general", icon: Icons.Target, key: "objetivo_general" as const, mono: false },
|
||||
{ id: "sec-eval", title: "Sistema de evaluación", icon: Icons.CheckCircle2, key: "sistema_evaluacion" as const, mono: false },
|
||||
{ id: "sec-ing", title: "Perfil de ingreso", icon: Icons.UserPlus, key: "perfil_ingreso" as const, mono: false },
|
||||
{ id: "sec-egr", title: "Perfil de egreso", icon: Icons.GraduationCap, key: "perfil_egreso" as const, mono: false },
|
||||
{ id: "sec-cg", title: "Competencias genéricas", icon: Icons.Sparkles, key: "competencias_genericas" as const, mono: false },
|
||||
{ id: "sec-ce", title: "Competencias específicas", icon: Icons.Cog, key: "competencias_especificas" 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-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 }
|
||||
{
|
||||
id: "sec-clave",
|
||||
title: "Clave del plan",
|
||||
icon: Icons.Key,
|
||||
key: "clave_del_plan_de_estudios" as const,
|
||||
mono: true,
|
||||
},
|
||||
{
|
||||
id: "sec-area",
|
||||
title: "Área de estudio",
|
||||
icon: Icons.Library,
|
||||
key: "area_de_estudio" as const,
|
||||
mono: false,
|
||||
},
|
||||
|
||||
// --- Estructura Temporal ---
|
||||
{
|
||||
id: "sec-ciclos",
|
||||
title: "Total de ciclos",
|
||||
icon: Icons.CalendarRange,
|
||||
key: "total_de_ciclos_del_plan_de_estudios" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-duracion-ciclo",
|
||||
title: "Duración del ciclo (semanas)",
|
||||
icon: Icons.CalendarDays,
|
||||
key: "duracion_del_ciclo_escolar" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-carga",
|
||||
title: "Carga horaria semanal",
|
||||
icon: Icons.Clock,
|
||||
key: "carga_horaria_a_la_semana" as const,
|
||||
mono: false,
|
||||
},
|
||||
|
||||
// --- Perfiles y Fines ---
|
||||
{
|
||||
id: "sec-antecedente",
|
||||
title: "Antecedente académico",
|
||||
icon: Icons.BookOpen,
|
||||
key: "antecedente_academico" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-ingreso",
|
||||
title: "Perfil de ingreso",
|
||||
icon: Icons.UserPlus,
|
||||
key: "perfil_de_ingreso" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-fines",
|
||||
title: "Fines de aprendizaje",
|
||||
icon: Icons.Target,
|
||||
key: "fines_de_aprendizaje_o_formacion" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-egreso",
|
||||
title: "Perfil de egreso",
|
||||
icon: Icons.UserCheck,
|
||||
key: "perfil_de_egreso" as const,
|
||||
mono: false,
|
||||
},
|
||||
|
||||
// --- Operatividad y Modelo ---
|
||||
{
|
||||
id: "sec-admin",
|
||||
title: "Administración y operatividad",
|
||||
icon: Icons.Briefcase,
|
||||
key: "administracion_y_operatividad_del_plan_de_estudios" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-sustento",
|
||||
title: "Sustento teórico",
|
||||
icon: Icons.Book,
|
||||
key: "sustento_teorico_del_modelo_curricular" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-justificacion",
|
||||
title: "Justificación curricular",
|
||||
icon: Icons.MessageSquareText,
|
||||
key: "justificacion_de_la_propuesta_curricular" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-evaluacion",
|
||||
title: "Evaluación periódica",
|
||||
icon: Icons.CheckCircle2,
|
||||
key: "propuesta_de_evaluacion_periodica_del_plan_de_estudios" as const,
|
||||
mono: false,
|
||||
},
|
||||
|
||||
// --- Específicos / Opcionales ---
|
||||
{
|
||||
id: "sec-investigacion",
|
||||
title: "Programa de investigación",
|
||||
icon: Icons.Microscope,
|
||||
key: "programa_de_investigacion" as const,
|
||||
mono: false,
|
||||
},
|
||||
{
|
||||
id: "sec-propedeutico",
|
||||
title: "Curso propedéutico",
|
||||
icon: Icons.School,
|
||||
key: "curso_propedeutico" as const,
|
||||
mono: false,
|
||||
},
|
||||
|
||||
// --- Meta / Sistema ---
|
||||
{
|
||||
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)
|
||||
|
||||
);
|
||||
const [iaContext, setIaContext] = useState<{
|
||||
key: keyof PlanTextFields;
|
||||
title: string;
|
||||
content: string;
|
||||
} | null>(null);
|
||||
|
||||
return (
|
||||
<>
|
||||
<div className="grid gap-5 md:grid-cols-2">
|
||||
{sections.map((s) => {
|
||||
const text = plan[s.key] ?? null
|
||||
return (
|
||||
<SectionPanel key={s.id} id={s.id} title={s.title} icon={s.icon} color={color}>
|
||||
{s.key === "historico" ? (
|
||||
<>
|
||||
<Button variant="outline" size="sm" onClick={() => setOpenHistorial(true)}>
|
||||
Ver historial
|
||||
</Button>
|
||||
<Button variant="outline" size="sm" onClick={() => setopenModalIa(true)}>
|
||||
Promt
|
||||
</Button>
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<ExpandableText text={text} mono={s.mono} />
|
||||
<div className="mt-4 flex flex-wrap gap-2">
|
||||
const text = String(plan[s.key]) ?? null;
|
||||
return (
|
||||
<SectionPanel
|
||||
key={s.id}
|
||||
id={s.id}
|
||||
title={s.title}
|
||||
icon={s.icon}
|
||||
color={color}
|
||||
>
|
||||
{s.key === "historico" ? (
|
||||
<>
|
||||
<Button
|
||||
variant="outline"
|
||||
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)
|
||||
}}
|
||||
onClick={() => setOpenHistorial(true)}
|
||||
>
|
||||
Copiar
|
||||
Ver historial
|
||||
</Button>
|
||||
{s.key !== "prompt" && (
|
||||
<Button
|
||||
variant="ghost"
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
onClick={() => setopenModalIa(true)}
|
||||
>
|
||||
Promt
|
||||
</Button>
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<ExpandableText text={text} mono={s.mono} />
|
||||
<div className="mt-4 flex flex-wrap gap-2">
|
||||
<Button
|
||||
variant="outline"
|
||||
size="sm"
|
||||
disabled={
|
||||
!text || (Array.isArray(text) && text.length === 0)
|
||||
}
|
||||
onClick={() => {
|
||||
const current = Array.isArray(text) ? text.join("\n") : (text ?? "")
|
||||
setEditing({ key: s.key, title: s.title })
|
||||
setDraft(current)
|
||||
const toCopy = Array.isArray(text)
|
||||
? text.join("\n")
|
||||
: (text ?? "");
|
||||
if (toCopy) navigator.clipboard.writeText(toCopy);
|
||||
}}
|
||||
>
|
||||
Editar
|
||||
</Button>
|
||||
)}
|
||||
</div>
|
||||
</>
|
||||
)}
|
||||
</SectionPanel>
|
||||
)
|
||||
})}
|
||||
|
||||
Copiar
|
||||
</Button>
|
||||
{s.key !== "prompt" && (
|
||||
<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>
|
||||
|
||||
{/* Diálogo de edición */}
|
||||
<Dialog open={!!editing} onOpenChange={(o) => { if (!o) setEditing(null) }}>
|
||||
<DialogContent className="max-w-2xl">
|
||||
<DialogHeader>
|
||||
<DialogTitle className="font-mono">
|
||||
{editing ? `Editar: ${sections.find((x) => x.key === editing.key)?.title}` : ""}
|
||||
</DialogTitle>
|
||||
</DialogHeader>
|
||||
{/* Diálogo de edición */}
|
||||
<Dialog
|
||||
open={!!editing}
|
||||
onOpenChange={(o) => {
|
||||
if (!o) setEditing(null);
|
||||
}}
|
||||
>
|
||||
<DialogContent className="max-w-2xl">
|
||||
<DialogHeader>
|
||||
<DialogTitle className="font-mono">
|
||||
{editing
|
||||
? `Editar: ${sections.find((x) => x.key === editing.key)?.title}`
|
||||
: ""}
|
||||
</DialogTitle>
|
||||
</DialogHeader>
|
||||
|
||||
<Textarea
|
||||
value={draft}
|
||||
onChange={(e) => setDraft(e.target.value)}
|
||||
className={`min-h-[260px] ${editing?.key === "prompt" ? "font-mono" : ""}`}
|
||||
placeholder="Escribe aquí…"
|
||||
/>
|
||||
<Textarea
|
||||
value={draft}
|
||||
onChange={(e) => setDraft(e.target.value)}
|
||||
className={`min-h-[260px] ${editing?.key === "prompt" ? "font-mono" : ""}`}
|
||||
placeholder="Escribe aquí…"
|
||||
/>
|
||||
|
||||
<DialogFooter>
|
||||
<Button variant="outline" onClick={() => setEditing(null)}>Cancelar</Button>
|
||||
<Button
|
||||
onClick={async () => {
|
||||
if (!editing) return
|
||||
<DialogFooter>
|
||||
<Button variant="outline" onClick={() => setEditing(null)}>
|
||||
Cancelar
|
||||
</Button>
|
||||
<Button
|
||||
onClick={async () => {
|
||||
if (!editing) return;
|
||||
|
||||
// 1️⃣ Obtener el valor anterior (por ejemplo, desde 'plan' o 'section')
|
||||
const oldValue = (plan as any)[editing.key]
|
||||
// 1️⃣ Obtener el valor anterior (por ejemplo, desde 'plan' o 'section')
|
||||
const oldValue = (plan as any)[editing.key];
|
||||
|
||||
// 2️⃣ Crear un diff tipo JSON Patch
|
||||
const diff = [{
|
||||
op: "replace",
|
||||
path: `/${editing.key}`,
|
||||
from: oldValue,
|
||||
value: draft
|
||||
}]
|
||||
// 2️⃣ Crear un diff tipo JSON Patch
|
||||
const diff = [
|
||||
{
|
||||
op: "replace",
|
||||
path: `/${editing.key}`,
|
||||
from: oldValue,
|
||||
value: draft,
|
||||
},
|
||||
];
|
||||
|
||||
// 3️⃣ Guardar respaldo antes de actualizar
|
||||
const { error: backupError } = await supabase.from("historico_cambios").insert({
|
||||
id_plan_estudios: planId, // asegúrate de tener `plan.id` disponible en este contexto
|
||||
json_cambios: diff,
|
||||
user_id:auth.user?.id,
|
||||
created_at: new Date().toISOString()
|
||||
})
|
||||
// 3️⃣ Guardar respaldo antes de actualizar
|
||||
const { error: backupError } = await supabase
|
||||
.from("historico_cambios")
|
||||
.insert({
|
||||
id_plan_estudios: planId, // asegúrate de tener `plan.id` disponible en este contexto
|
||||
json_cambios: diff,
|
||||
user_id: auth.user?.id,
|
||||
created_at: new Date().toISOString(),
|
||||
});
|
||||
|
||||
if (backupError) {
|
||||
console.error("Error al guardar respaldo:", backupError)
|
||||
alert("No se pudo guardar el respaldo de los cambios")
|
||||
return
|
||||
}
|
||||
if (backupError) {
|
||||
console.error("Error al guardar respaldo:", backupError);
|
||||
alert("No se pudo guardar el respaldo de los cambios");
|
||||
return;
|
||||
}
|
||||
|
||||
// 4️⃣ Ejecutar la mutación original
|
||||
updateField.mutate({ key: editing.key, value: draft })
|
||||
// 4️⃣ Ejecutar la mutación original
|
||||
updateField.mutate({ key: editing.key, value: draft });
|
||||
|
||||
// 5️⃣ Cerrar el diálogo
|
||||
setEditing(null)
|
||||
}}
|
||||
disabled={updateField.isPending}
|
||||
>
|
||||
{updateField.isPending ? "Guardando…" : "Guardar"}
|
||||
</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>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
// 5️⃣ Cerrar el diálogo
|
||||
setEditing(null);
|
||||
}}
|
||||
disabled={updateField.isPending}
|
||||
>
|
||||
{updateField.isPending ? "Guardando…" : "Guardar"}
|
||||
</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>
|
||||
</DialogContent>
|
||||
</Dialog>
|
||||
<HistorialCambiosModal
|
||||
open={openHistorial}
|
||||
onClose={() => setOpenHistorial(false)}
|
||||
planId={planId}
|
||||
onRestore={async (key, value) => {
|
||||
updateField.mutate({ key, value })
|
||||
}}
|
||||
/>
|
||||
|
||||
<AIChatModal
|
||||
open={openModalIa}
|
||||
onClose={() => setopenModalIa(false)}
|
||||
context={{
|
||||
section: iaContext?.title,
|
||||
fieldKey: iaContext?.key,
|
||||
originalText: iaContext?.content,
|
||||
cont_conversation: `Eres un experto en craer planes de estudios basate en el id del plan ${planId} que se encuentra en la tabla plan_estudios con el mcp para realizar los cambios que se te soliciten Responde únicamente con la información solicitada.
|
||||
open={openHistorial}
|
||||
onClose={() => setOpenHistorial(false)}
|
||||
planId={planId}
|
||||
onRestore={async (key, value) => {
|
||||
updateField.mutate({ key, value });
|
||||
}}
|
||||
/>
|
||||
|
||||
<AIChatModal
|
||||
open={openModalIa}
|
||||
onClose={() => setopenModalIa(false)}
|
||||
context={{
|
||||
section: iaContext?.title,
|
||||
fieldKey: iaContext?.key,
|
||||
originalText: iaContext?.content,
|
||||
cont_conversation: `Eres un experto en craer planes de estudios basate en el id del plan ${planId} que se encuentra en la tabla plan_estudios con el mcp para realizar los cambios que se te soliciten Responde únicamente con la información solicitada.
|
||||
No uses frases como “claro”, “por supuesto”, “aquí tienes”, “con gusto”, “hola”, “perfecto”.
|
||||
No uses introducciones, despedidas ni texto de relleno.
|
||||
Entrega solo el contenido útil.`,
|
||||
}}
|
||||
onAccept={(newText: string) => {
|
||||
if (iaContext) {
|
||||
updateField.mutate({ key: iaContext.key, value: newText })
|
||||
setIaContext(null)
|
||||
}
|
||||
}}
|
||||
/>
|
||||
|
||||
|
||||
}}
|
||||
onAccept={(newText: string) => {
|
||||
if (iaContext) {
|
||||
updateField.mutate({ key: iaContext.key, value: newText });
|
||||
setIaContext(null);
|
||||
}
|
||||
}}
|
||||
/>
|
||||
</>
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
@@ -26,15 +26,17 @@ export function planByIdOptions(planId: string) {
|
||||
queryKey: planKeys.byId(planId),
|
||||
queryFn: async (): Promise<PlanFull> => {
|
||||
const { data, error } = await supabase
|
||||
.from("plan_estudios")
|
||||
.select(`
|
||||
id, nombre, nivel, objetivo_general, perfil_ingreso, perfil_egreso, duracion, total_creditos,
|
||||
competencias_genericas, competencias_especificas, sistema_evaluacion, indicadores_desempeno,
|
||||
pertinencia, prompt, estado, fecha_creacion,
|
||||
carreras ( id, nombre, facultades:facultades ( id, nombre, color, icon ) )
|
||||
`)
|
||||
.eq("id", planId)
|
||||
.maybeSingle()
|
||||
.from("plan_estudios")
|
||||
.select(`
|
||||
*,
|
||||
carreras (
|
||||
id,
|
||||
nombre,
|
||||
facultades ( id, nombre, color, icon )
|
||||
)
|
||||
`)
|
||||
.eq("id", planId)
|
||||
.maybeSingle();
|
||||
if (error || !data) throw error ?? new Error("Plan no encontrado")
|
||||
return data as unknown as PlanFull
|
||||
},
|
||||
|
||||
@@ -49,7 +49,7 @@ async function fetchDashboard(): Promise<LoaderData> {
|
||||
supabase
|
||||
.from('plan_estudios')
|
||||
.select(
|
||||
'id, nombre, fecha_creacion, objetivo_general, perfil_ingreso, perfil_egreso, sistema_evaluacion, total_creditos'
|
||||
'*'
|
||||
),
|
||||
supabase
|
||||
.from('asignaturas')
|
||||
|
||||
@@ -116,8 +116,9 @@ function RouteComponent() {
|
||||
<CardContent ref={statsRef}>
|
||||
<div className="academics relative z-10 grid gap-3 [grid-template-columns:repeat(auto-fit,minmax(180px,1fr))]">
|
||||
<StatCard label="Nivel" value={plan.nivel ?? "—"} Icon={Icons.GraduationCap} accent={facColor} />
|
||||
<StatCard label="Duración" value={plan.duracion ?? "—"} Icon={Icons.Clock} accent={facColor} />
|
||||
<StatCard label="Créditos" value={fmt(plan.total_creditos)} Icon={Icons.Coins} accent={facColor} />
|
||||
<StatCard label="Duración" value={plan.total_de_ciclos_del_plan_de_estudios ?? "—"} Icon={Icons.Clock} accent={facColor} />
|
||||
<StatCard label="Modalidad educativa" value={plan.modalidad_educativa ?? "—"} Icon={Icons.Layers} accent={facColor} />
|
||||
<StatCard label="Diseño curricular" value={plan.diseno_curricular ?? "—"} Icon={Icons.Layout} accent={facColor} />
|
||||
<StatCard label="Creado" value={plan.fecha_creacion ? new Date(plan.fecha_creacion).toLocaleDateString() : "—"} Icon={Icons.CalendarDays} accent={facColor} />
|
||||
</div>
|
||||
</CardContent>
|
||||
|
||||
Reference in New Issue
Block a user