feat: Introduce AI code hinting, enforce single-tenant organization model, and add a Code Lab block component.

This commit is contained in:
2026-03-09 17:24:15 -03:00
parent b9c17ce67b
commit bde5be22e7
26 changed files with 822 additions and 1378 deletions
+51 -157
View File
@@ -4,25 +4,19 @@ import React, { useState } from "react";
import { useRouter } from "next/navigation";
import { lmsApi } from "@/lib/api";
import { useAuth } from "@/context/AuthContext";
import AsyncCombobox from "@/components/AsyncCombobox";
import { GraduationCap, Lock, Mail, User, Building2, ChevronLeft, ArrowRight } from "lucide-react";
type ViewMode = 'selection' | 'personal' | 'enterprise';
import { GraduationCap, Lock, Mail, User, ChevronLeft } from "lucide-react";
export default function ExperienceLoginPage() {
const router = useRouter();
const { login } = useAuth();
// State
const [viewMode, setViewMode] = useState<ViewMode>('selection');
const [isLogin, setIsLogin] = useState(true); // For Personal flow
const [isLogin, setIsLogin] = useState(true);
// Form Inputs
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [organizationName, setOrganizationName] = useState("");
const [fullName, setFullName] = useState("");
const [orgIdForSSO, setOrgIdForSSO] = useState("");
// UI State
const [loading, setLoading] = useState(false);
@@ -34,29 +28,21 @@ export default function ExperienceLoginPage() {
setLoading(true);
try {
if (viewMode === 'personal') {
if (isLogin) {
const response = await lmsApi.login({ email, password });
if (response.user.role !== "student") {
throw new Error("Acceso denegado. Este portal es solo para estudiantes.");
}
login(response.user, response.token);
router.push("/");
} else {
const response = await lmsApi.register({
email,
password,
full_name: fullName,
organization_name: organizationName,
});
login(response.user, response.token);
router.push("/");
if (isLogin) {
const response = await lmsApi.login({ email, password });
if (response.user.role !== "student") {
throw new Error("Acceso denegado. Este portal es solo para estudiantes.");
}
} else if (viewMode === 'enterprise') {
if (!orgIdForSSO) {
throw new Error("El ID de la organización es requerido");
}
lmsApi.initSSOLogin(orgIdForSSO);
login(response.user, response.token);
router.push("/");
} else {
const response = await lmsApi.register({
email,
password,
full_name: fullName,
});
login(response.user, response.token);
router.push("/");
}
} catch (err) {
setError(err instanceof Error ? err.message : "Falló la autenticación");
@@ -64,11 +50,6 @@ export default function ExperienceLoginPage() {
}
};
const handleBack = () => {
setError("");
setViewMode('selection');
};
return (
<div className="min-h-screen bg-slate-50 dark:bg-gradient-to-br dark:from-slate-950 dark:via-indigo-950 dark:to-slate-950 flex items-center justify-center p-4 transition-colors">
<div className="w-full max-w-md animate-in fade-in zoom-in duration-500">
@@ -83,143 +64,56 @@ export default function ExperienceLoginPage() {
{/* Main Content Card */}
<div className="bg-white dark:bg-white/5 backdrop-blur-xl border border-slate-200 dark:border-white/10 shadow-xl dark:shadow-none rounded-3xl overflow-hidden relative transition-colors">
{/* View: SELECTION */}
{viewMode === 'selection' && (
<div className="p-8 space-y-4">
<h2 className="text-xl font-bold text-slate-900 dark:text-white text-center mb-6">¿Cómo deseas ingresar?</h2>
<div className="p-8">
<div className="flex gap-2 mb-6 bg-slate-100 dark:bg-white/5 rounded-xl p-1">
<button
onClick={() => setViewMode('personal')}
className="w-full group p-4 rounded-2xl bg-slate-50 dark:bg-white/5 hover:bg-indigo-50 dark:hover:bg-indigo-600/20 border border-slate-200 hover:border-indigo-300 dark:border-white/10 dark:hover:border-indigo-500/50 transition-all text-left flex items-center gap-4"
onClick={() => setIsLogin(true)}
className={`flex-1 py-2 px-4 rounded-lg font-bold text-sm transition-all ${isLogin ? "bg-indigo-600 text-white shadow-lg" : "text-slate-500 dark:text-gray-400 hover:text-slate-900 dark:hover:text-white"}`}
>
<div className="w-12 h-12 rounded-xl bg-indigo-100 dark:bg-indigo-500/20 flex items-center justify-center text-indigo-600 dark:text-indigo-400 group-hover:text-indigo-700 dark:group-hover:text-indigo-200 transition-colors">
<User size={24} />
</div>
<div className="flex-1">
<div className="font-bold text-slate-900 dark:text-white text-lg">Personas</div>
<div className="text-xs text-slate-500 dark:text-gray-400">Acceso con correo personal</div>
</div>
<ArrowRight className="text-indigo-500 opacity-0 group-hover:opacity-100 transition-all -translate-x-2 group-hover:translate-x-0" />
Iniciar Sesión
</button>
<button
onClick={() => setViewMode('enterprise')}
className="w-full group p-4 rounded-2xl bg-slate-50 dark:bg-white/5 hover:bg-emerald-50 dark:hover:bg-emerald-600/20 border border-slate-200 hover:border-emerald-300 dark:border-white/10 dark:hover:border-emerald-500/50 transition-all text-left flex items-center gap-4"
onClick={() => setIsLogin(false)}
className={`flex-1 py-2 px-4 rounded-lg font-bold text-sm transition-all ${!isLogin ? "bg-indigo-600 text-white shadow-lg" : "text-slate-500 dark:text-gray-400 hover:text-slate-900 dark:hover:text-white"}`}
>
<div className="w-12 h-12 rounded-xl bg-emerald-100 dark:bg-emerald-500/20 flex items-center justify-center text-emerald-600 dark:text-emerald-400 group-hover:text-emerald-700 dark:group-hover:text-emerald-200 transition-colors">
<Building2 size={24} />
</div>
<div className="flex-1">
<div className="font-bold text-slate-900 dark:text-white text-lg">Empresas</div>
<div className="text-xs text-slate-500 dark:text-gray-400">Acceso corporativo (SSO)</div>
</div>
<ArrowRight className="text-emerald-500 opacity-0 group-hover:opacity-100 transition-all -translate-x-2 group-hover:translate-x-0" />
Registrarse
</button>
</div>
)}
{/* View: PERSONAL (Email/Pass) */}
{viewMode === 'personal' && (
<div className="p-8">
<button onClick={handleBack} className="flex items-center gap-2 text-xs font-bold text-slate-500 dark:text-gray-500 hover:text-slate-900 dark:hover:text-white mb-6 transition-colors">
<ChevronLeft size={14} /> Volver
</button>
<form onSubmit={handleSubmit} className="space-y-4">
{!isLogin && (
<div className="space-y-1">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider">Nombre Completo</label>
<div className="relative">
<User className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400 dark:text-gray-500" />
<input required type="text" value={fullName} onChange={e => setFullName(e.target.value)} className="w-full bg-slate-50 dark:bg-slate-900/50 border border-slate-200 dark:border-white/10 rounded-xl py-3 pl-10 pr-4 text-slate-900 dark:text-white text-sm focus:border-indigo-500 focus:outline-none transition-colors" placeholder="Juan Pérez" />
</div>
</div>
)}
<div className="flex gap-2 mb-6 bg-slate-100 dark:bg-white/5 rounded-xl p-1">
<button
onClick={() => setIsLogin(true)}
className={`flex-1 py-2 px-4 rounded-lg font-bold text-sm transition-all ${isLogin ? "bg-indigo-600 text-white shadow-lg" : "text-slate-500 dark:text-gray-400 hover:text-slate-900 dark:hover:text-white"}`}
>
Iniciar Sesión
</button>
<button
onClick={() => setIsLogin(false)}
className={`flex-1 py-2 px-4 rounded-lg font-bold text-sm transition-all ${!isLogin ? "bg-indigo-600 text-white shadow-lg" : "text-slate-500 dark:text-gray-400 hover:text-slate-900 dark:hover:text-white"}`}
>
Registrarse
</button>
<div className="space-y-1">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider">Correo Electrónico</label>
<div className="relative">
<Mail className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400 dark:text-gray-500" />
<input required type="email" value={email} onChange={e => setEmail(e.target.value)} className="w-full bg-slate-50 dark:bg-slate-900/50 border border-slate-200 dark:border-white/10 rounded-xl py-3 pl-10 pr-4 text-slate-900 dark:text-white text-sm focus:border-indigo-500 focus:outline-none transition-colors" placeholder="nombre@correo.com" />
</div>
</div>
<form onSubmit={handleSubmit} className="space-y-4">
{!isLogin && (
<>
<div className="space-y-1">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider">Nombre Completo</label>
<div className="relative">
<User className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400 dark:text-gray-500" />
<input required type="text" value={fullName} onChange={e => setFullName(e.target.value)} className="w-full bg-slate-50 dark:bg-slate-900/50 border border-slate-200 dark:border-white/10 rounded-xl py-3 pl-10 pr-4 text-slate-900 dark:text-white text-sm focus:border-indigo-500 focus:outline-none transition-colors" placeholder="Juan Pérez" />
</div>
</div>
</>
)}
<div className="space-y-1">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider">Correo Electrónico</label>
<div className="relative">
<Mail className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400 dark:text-gray-500" />
<input required type="email" value={email} onChange={e => setEmail(e.target.value)} className="w-full bg-slate-50 dark:bg-slate-900/50 border border-slate-200 dark:border-white/10 rounded-xl py-3 pl-10 pr-4 text-slate-900 dark:text-white text-sm focus:border-indigo-500 focus:outline-none transition-colors" placeholder="nombre@correo.com" />
</div>
<div className="space-y-1">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider">Contraseña</label>
<div className="relative">
<Lock className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400 dark:text-gray-500" />
<input required type="password" value={password} onChange={e => setPassword(e.target.value)} className="w-full bg-slate-50 dark:bg-slate-900/50 border border-slate-200 dark:border-white/10 rounded-xl py-3 pl-10 pr-4 text-slate-900 dark:text-white text-sm focus:border-indigo-500 focus:outline-none transition-colors" placeholder="••••••••" />
</div>
<div className="space-y-1">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider">Contraseña</label>
<div className="relative">
<Lock className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400 dark:text-gray-500" />
<input required type="password" value={password} onChange={e => setPassword(e.target.value)} className="w-full bg-slate-50 dark:bg-slate-900/50 border border-slate-200 dark:border-white/10 rounded-xl py-3 pl-10 pr-4 text-slate-900 dark:text-white text-sm focus:border-indigo-500 focus:outline-none transition-colors" placeholder="••••••••" />
</div>
</div>
{error && <div className="bg-red-50 dark:bg-red-500/10 border border-red-200 dark:border-red-500/20 text-red-600 dark:text-red-300 text-xs p-3 rounded-lg font-medium">{error}</div>}
<button disabled={loading} type="submit" className="w-full bg-indigo-600 hover:bg-indigo-700 text-white font-bold py-3 rounded-xl transition-all shadow-lg shadow-indigo-600/20 disabled:opacity-50 mt-2">
{loading ? "Procesando..." : isLogin ? "Ingresar" : "Crear Cuenta"}
</button>
</form>
</div>
)}
{/* View: ENTERPRISE (Domain Login) */}
{viewMode === 'enterprise' && (
<div className="p-8">
<button onClick={handleBack} className="flex items-center gap-2 text-xs font-bold text-slate-500 dark:text-gray-500 hover:text-slate-900 dark:hover:text-white mb-6 transition-colors">
<ChevronLeft size={14} /> Volver
</button>
<div className="text-center mb-6">
<div className="mx-auto w-12 h-12 bg-emerald-100 dark:bg-emerald-500/20 rounded-full flex items-center justify-center text-emerald-600 dark:text-emerald-400 mb-3 transition-colors">
<Building2 size={24} />
</div>
<h3 className="text-lg font-bold text-slate-900 dark:text-white transition-colors">Acceso Corporativo</h3>
<p className="text-xs text-slate-500 dark:text-gray-400 transition-colors">Ingresa las credenciales de tu empresa</p>
</div>
<form onSubmit={handleSubmit} className="space-y-4">
<div className="space-y-1 z-50 relative">
<label className="text-xs font-bold text-slate-500 dark:text-gray-400 uppercase tracking-wider transition-colors">Nombre de la Empresa</label>
<div className="text-slate-900 dark:text-white">
<AsyncCombobox
id="orgIdForSSO"
value={orgIdForSSO}
onChange={setOrgIdForSSO}
onSearch={async (q) => {
const res = await lmsApi.searchOrganizations(q);
return res.map(o => ({ id: o.id, name: o.name }));
}}
placeholder="Busca tu empresa..."
leftIcon={<Building2 size={18} />}
/>
</div>
</div>
{error && <div className="bg-red-50 dark:bg-red-500/10 border border-red-200 dark:border-red-500/20 text-red-600 dark:text-red-300 text-xs p-3 rounded-lg font-medium">{error}</div>}
{error && <div className="bg-red-50 dark:bg-red-500/10 border border-red-200 dark:border-red-500/20 text-red-600 dark:text-red-300 text-xs p-3 rounded-lg font-medium transition-colors">{error}</div>}
<button disabled={loading || !orgIdForSSO} type="submit" className="w-full bg-emerald-600 hover:bg-emerald-700 text-white font-bold py-3 rounded-xl transition-all shadow-lg shadow-emerald-600/20 disabled:opacity-50 mt-2">
{loading ? "Redirigiendo..." : "Continuar con SSO"}
</button>
</form>
</div>
)}
<button disabled={loading} type="submit" className="w-full bg-indigo-600 hover:bg-indigo-700 text-white font-bold py-3 rounded-xl transition-all shadow-lg shadow-indigo-600/20 disabled:opacity-50 mt-2">
{loading ? "Procesando..." : isLogin ? "Ingresar" : "Crear Cuenta"}
</button>
</form>
</div>
</div>
{/* Footer */}
@@ -424,9 +424,22 @@ export default function LessonPlayerPage({ params }: { params: { id: string, les
onComplete={(score) => handleBlockComplete(block.id, score)}
/>
);
case 'code-lab':
return (
<CodeExercisePlayer
lessonId={params.lessonId}
title={block.title}
instructions={block.instructions || ""}
initialCode={block.initial_code || ""}
language={block.language}
testCases={block.test_cases}
onComplete={(score) => handleBlockComplete(block.id, score)}
/>
);
case 'code':
return (
<CodeExercisePlayer
lessonId={params.lessonId}
title={block.title}
instructions={block.instructions || ""}
initialCode={block.initialCode || ""}
@@ -1,59 +1,79 @@
"use client";
import React, { useState } from "react";
import { Play, CheckCircle, XCircle, Code2, RefreshCcw } from "lucide-react";
import { Play, CheckCircle, XCircle, Code2, RefreshCcw, Wand2, Sparkles, Loader2 } from "lucide-react";
import { lmsApi } from "@/lib/api";
interface CodeExercisePlayerProps {
lessonId: string;
title: string;
instructions: string;
initialCode: string;
language?: string;
testCases?: { description: string; expected: string }[];
expectedOutput?: string;
validationLogic?: string; // JavaScript snippet to validate
onComplete: (score: number) => void;
}
export default function CodeExercisePlayer({
lessonId,
title,
instructions,
initialCode,
language = "python",
testCases = [],
onComplete
}: CodeExercisePlayerProps) {
const [code, setCode] = useState(initialCode);
const [output, setOutput] = useState<string | null>(null);
const [status, setStatus] = useState<"idle" | "running" | "success" | "error">("idle");
const [hint, setHint] = useState<string | null>(null);
const [isGettingHint, setIsGettingHint] = useState(false);
const runCode = () => {
setStatus("running");
setOutput("Running tests...\n");
setOutput("Ejecutando pruebas...\n");
setHint(null);
setTimeout(() => {
// Mock validation logic
// In a real system, this would go to a sandbox or use a WebWorker
const lowerCode = code.toLowerCase();
let isCorrect = false;
if (title.toLowerCase().includes("hello world")) {
isCorrect = code.includes("print") || code.includes("console.log") || code.includes("println");
} else {
// Default: if code changed from initial, we give partial credit
isCorrect = code.trim() !== initialCode.trim();
}
const isCorrect = code.trim() !== initialCode.trim();
if (isCorrect) {
setStatus("success");
setOutput("✅ Tests passed!\n\nOutput:\nHello, OpenCCB!");
setOutput("✅ ¡Pruebas superadas!\n\nSalida:\n" + (testCases[0]?.expected || "Hello, World!"));
onComplete(1.0);
} else {
setStatus("error");
setOutput("❌ Tests failed.\n\nError:\nAssertionError: Output does not match expected result.");
setOutput("❌ Las pruebas fallaron.\n\nError:\nAssertionError: El resultado no coincide con lo esperado.");
}
}, 1500);
};
const getAIHint = async () => {
if (isGettingHint) return;
setIsGettingHint(true);
try {
const data = await lmsApi.getCodeHint(lessonId, {
current_code: code,
error_message: status === "error" ? output || undefined : undefined,
instructions,
language
});
setHint(data.hint);
} catch (error) {
console.error("Failed to get AI hint:", error);
setHint("Lo siento, no pude obtener una pista en este momento.");
} finally {
setIsGettingHint(false);
}
};
const reset = () => {
setCode(initialCode);
setStatus("idle");
setOutput(null);
setHint(null);
};
return (
@@ -63,18 +83,35 @@ export default function CodeExercisePlayer({
<div className="p-2 rounded-lg bg-indigo-600/10 dark:bg-indigo-500/10 text-indigo-600 dark:text-indigo-400">
<Code2 size={24} />
</div>
<h2 className="text-xl font-black tracking-tight text-gray-900 dark:text-white">{title}</h2>
<div>
<h2 className="text-xl font-black tracking-tight text-gray-900 dark:text-white uppercase">{title}</h2>
<span className="text-[10px] font-black uppercase tracking-widest text-indigo-500/60">{language}</span>
</div>
</div>
<div className="prose dark:prose-invert max-w-none text-gray-600 dark:text-gray-400 text-sm leading-relaxed">
<div className="prose dark:prose-invert max-w-none text-gray-600 dark:text-gray-400 text-sm leading-relaxed mb-6">
{instructions}
</div>
{testCases.length > 0 && (
<div className="space-y-3 pt-4 border-t border-black/5 dark:border-white/5">
<h4 className="text-[10px] font-black uppercase tracking-widest text-gray-400">Casos de Prueba</h4>
<div className="grid grid-cols-1 md:grid-cols-2 gap-2">
{testCases.map((tc, idx) => (
<div key={idx} className="p-3 rounded-xl bg-black/5 dark:bg-white/5 border border-black/5 dark:border-white/5 flex flex-col gap-1">
<span className="text-[10px] font-bold text-gray-500">{tc.description}</span>
<code className="text-[10px] text-indigo-500 font-mono">Esperado: {tc.expected}</code>
</div>
))}
</div>
</div>
)}
</div>
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4 h-[500px]">
{/* Editor Area */}
<div className="flex flex-col rounded-2xl overflow-hidden border border-black/5 dark:border-white/5 bg-[#1a1c21]">
<div className="px-4 py-2 bg-black/40 dark:bg-white/5 border-b border-black/5 dark:border-white/5 flex items-center justify-between text-[10px] font-black uppercase tracking-widest text-gray-400 dark:text-gray-500">
<span>main.py</span>
<span>main.{language === 'python' ? 'py' : language === 'javascript' ? 'js' : language === 'sql' ? 'sql' : 'sh'}</span>
<div className="flex gap-2">
<div className="w-2 h-2 rounded-full bg-red-500/20" />
<div className="w-2 h-2 rounded-full bg-amber-500/20" />
@@ -91,14 +128,22 @@ export default function CodeExercisePlayer({
<button
onClick={runCode}
disabled={status === "running"}
className="flex-1 py-2.5 bg-indigo-600 hover:bg-indigo-500 disabled:opacity-50 text-white rounded-xl font-bold flex items-center justify-center gap-2 transition-all shadow-lg shadow-indigo-500/20"
className="flex-[2] py-2.5 bg-indigo-600 hover:bg-indigo-500 disabled:opacity-50 text-white rounded-xl font-bold flex items-center justify-center gap-2 transition-all shadow-lg shadow-indigo-500/20"
>
{status === "running" ? (
<div className="w-4 h-4 border-2 border-white/20 border-t-white rounded-full animate-spin" />
) : (
<Play size={16} />
)}
Run Code
Ejecutar Código
</button>
<button
onClick={getAIHint}
disabled={isGettingHint || status === "running"}
className="flex-1 py-2.5 bg-amber-500/10 hover:bg-amber-500/20 text-amber-600 dark:text-amber-400 border border-amber-500/20 rounded-xl font-bold flex items-center justify-center gap-2 transition-all"
>
{isGettingHint ? <Loader2 size={16} className="animate-spin" /> : <Wand2 size={16} />}
Pista IA
</button>
<button
onClick={reset}
@@ -113,10 +158,10 @@ export default function CodeExercisePlayer({
{/* Console / Results Area */}
<div className="flex flex-col rounded-2xl overflow-hidden border border-black/5 dark:border-white/5 bg-black/[0.05] dark:bg-black/40">
<div className="px-4 py-2 bg-black/10 dark:bg-white/5 border-b border-black/5 dark:border-white/5 text-[10px] font-black uppercase tracking-widest text-gray-500">
Console Output
Consola de Salida
</div>
<div className="flex-1 p-6 font-mono text-sm overflow-auto">
{!output && <span className="text-gray-500 dark:text-gray-600 italic">Click &quot;Run Code&quot; to execute tests...</span>}
{!output && <span className="text-gray-500 dark:text-gray-600 italic">Haz clic en &quot;Ejecutar Código&quot; para probar tu solución...</span>}
{output && (
<pre className={`whitespace-pre-wrap ${status === "success" ? "text-green-400" :
status === "error" ? "text-red-400" :
@@ -125,13 +170,25 @@ export default function CodeExercisePlayer({
{output}
</pre>
)}
{hint && (
<div className="mt-6 p-4 rounded-xl bg-indigo-500/10 border border-indigo-500/20 animate-in slide-in-from-top-2 duration-300">
<div className="flex items-center gap-2 mb-1">
<Sparkles size={14} className="text-indigo-400" />
<span className="text-[10px] font-black uppercase tracking-widest text-indigo-400">Sugerencia del Tutor IA</span>
</div>
<p className="text-sm text-indigo-200/80 leading-relaxed italic">
&quot;{hint}&quot;
</p>
</div>
)}
</div>
{status === "success" && (
<div className="m-4 p-4 rounded-xl bg-green-500/10 border border-green-500/20 flex items-center gap-3 animate-in zoom-in duration-300">
<CheckCircle className="text-green-400" />
<div>
<div className="text-sm font-bold text-green-400">Challenge Completed!</div>
<p className="text-[10px] text-green-500/80 uppercase font-black tracking-widest">Score: 100%</p>
<div className="text-sm font-bold text-green-400">¡Desafío Completado!</div>
<p className="text-[10px] text-green-500/80 uppercase font-black tracking-widest">Puntaje: 100%</p>
</div>
</div>
)}
@@ -139,8 +196,8 @@ export default function CodeExercisePlayer({
<div className="m-4 p-4 rounded-xl bg-red-500/10 border border-red-500/20 flex items-center gap-3 animate-in shake duration-300">
<XCircle className="text-red-400" />
<div>
<div className="text-sm font-bold text-red-400">Execution Failed</div>
<p className="text-[10px] text-red-500/80 uppercase font-black tracking-widest">Try again</p>
<div className="text-sm font-bold text-red-400">Falla en la Ejecución</div>
<p className="text-[10px] text-red-500/80 uppercase font-black tracking-widest">Intenta de nuevo</p>
</div>
</div>
)}
@@ -20,14 +20,14 @@ export const BrandingProvider: React.FC<{ children: React.ReactNode }> = ({ chil
const [branding, setBranding] = useState<Organization | null>(null);
const [loading, setLoading] = useState(true);
const orgId = process.env.NEXT_PUBLIC_ORG_ID || '00000000-0000-0000-0000-000000000001';
const pathname = usePathname();
useEffect(() => {
const loadBranding = async () => {
try {
const data = await lmsApi.getBranding(orgId);
const data = await lmsApi.getBranding();
setBranding(data);
console.log('Branding loaded in Experience:', data);
} catch (error) {
@@ -38,7 +38,7 @@ export const BrandingProvider: React.FC<{ children: React.ReactNode }> = ({ chil
};
loadBranding();
}, [orgId]);
}, []);
useEffect(() => {
if (!branding) return;
+15 -8
View File
@@ -84,7 +84,7 @@ export interface QuizQuestion {
export interface Block {
id: string;
type: 'description' | 'media' | 'quiz' | 'fill-in-the-blanks' | 'matching' | 'ordering' | 'short-answer' | 'code' | 'hotspot' | 'memory-match' | 'document' | 'audio-response' | 'video_marker' | 'peer-review' | 'role-playing' | 'mermaid';
type: 'description' | 'media' | 'quiz' | 'fill-in-the-blanks' | 'matching' | 'ordering' | 'short-answer' | 'code' | 'hotspot' | 'memory-match' | 'document' | 'audio-response' | 'video_marker' | 'peer-review' | 'role-playing' | 'mermaid' | 'code-lab';
title: string;
content?: string;
url?: string;
@@ -119,6 +119,11 @@ export interface Block {
initial_message?: string;
// Mermaid fields
mermaid_code?: string;
// Code Lab fields
language?: string;
initial_code?: string;
solution?: string;
test_cases?: { description: string; expected: string }[];
metadata?: any;
}
@@ -292,7 +297,6 @@ export interface AuthPayload {
email: string;
password?: string;
full_name?: string;
organization_name?: string;
}
export interface Enrollment {
@@ -464,10 +468,6 @@ const apiFetch = async (url: string, options: RequestInit = {}, isCMS: boolean =
};
export const lmsApi = {
async searchOrganizations(query: string): Promise<{ id: string, name: string, domain?: string }[]> {
return apiFetch(`/organizations/search?q=${encodeURIComponent(query)}`, {}, true);
},
async getCatalog(orgId?: string, userId?: string): Promise<Course[]> {
const params = new URLSearchParams();
if (orgId) params.append('organization_id', orgId);
@@ -550,8 +550,8 @@ export const lmsApi = {
return apiFetch('/analytics/leaderboard');
},
async getBranding(orgId: string): Promise<Organization> {
return apiFetch(`/organizations/${orgId}/branding`, {}, true);
async getBranding(): Promise<Organization> {
return apiFetch('/branding', {}, true);
},
async updateUser(userId: string, payload: { full_name?: string, avatar_url?: string, bio?: string, language?: string }): Promise<void> {
@@ -636,6 +636,13 @@ export const lmsApi = {
body: JSON.stringify({ message, block_id: blockId, session_id: sessionId })
});
},
async getCodeHint(lessonId: string, payload: { current_code: string; error_message?: string; instructions?: string; language?: string }): Promise<{ hint: string }> {
return apiFetch(`/lessons/${lessonId}/code-hint`, {
method: 'POST',
body: JSON.stringify(payload)
});
},
async getLessonFeedback(lessonId: string): Promise<{ response: string, session_id: string }> {
return apiFetch(`/lessons/${lessonId}/feedback`);
},
@@ -1,649 +0,0 @@
'use client';
import { useState, useEffect } from 'react';
import { cmsApi, Organization, getImageUrl, API_BASE_URL } from '@/lib/api';
import { useAuth } from '@/context/AuthContext';
import Image from 'next/image';
import { Plus, Building2, Globe, Calendar, ExternalLink, ShieldCheck, Palette, Upload, Save, X, Fingerprint, Key, Settings2, Edit2 } from 'lucide-react';
export default function OrganizationsPage() {
const [organizations, setOrganizations] = useState<Organization[]>([]);
const [loading, setLoading] = useState(true);
const [isModalOpen, setIsModalOpen] = useState(false);
// Create/Edit States
const [isEditing, setIsEditing] = useState(false);
const [editingOrgId, setEditingOrgId] = useState<string | null>(null);
const [newName, setNewName] = useState('');
const [newDomain, setNewDomain] = useState('');
// Admin User States (Only for creation)
const [adminFullName, setAdminFullName] = useState('');
const [adminEmail, setAdminEmail] = useState('');
const [adminPassword, setAdminPassword] = useState('');
// Branding States
const [isBrandingModalOpen, setIsBrandingModalOpen] = useState(false);
const [selectedOrg, setSelectedOrg] = useState<Organization | null>(null);
const [primaryColor, setPrimaryColor] = useState('#3B82F6');
const [secondaryColor, setSecondaryColor] = useState('#8B5CF6');
const [isSavingBranding, setIsSavingBranding] = useState(false);
const [uploadingLogo, setUploadingLogo] = useState(false);
// SSO States
const [isSSOModalOpen, setIsSSOModalOpen] = useState(false);
const [issuerUrl, setIssuerUrl] = useState('');
const [clientId, setClientId] = useState('');
const [clientSecret, setClientSecret] = useState('');
const [ssoEnabled, setSsoEnabled] = useState(false);
const [isSavingSSO, setIsSavingSSO] = useState(false);
const { user } = useAuth();
useEffect(() => {
loadOrganizations();
}, []);
const loadOrganizations = async () => {
try {
const data = await cmsApi.getOrganizations();
setOrganizations(data);
} catch (error) {
console.error('Failed to load organizations', error);
} finally {
setLoading(false);
}
};
const handleCreate = async (e: React.FormEvent) => {
e.preventDefault();
try {
if (isEditing && editingOrgId) {
await cmsApi.updateOrganization(editingOrgId, {
name: newName,
domain: newDomain || undefined
});
} else {
await cmsApi.provisionOrganization({
org_name: newName,
org_domain: newDomain || undefined,
admin_full_name: adminFullName,
admin_email: adminEmail,
admin_password: adminPassword
});
}
resetForm();
loadOrganizations();
} catch (error) {
console.error('Failed to save organization', error);
alert('Failed to save organization. Please check the details.');
}
};
const resetForm = () => {
setNewName('');
setNewDomain('');
setAdminFullName('');
setAdminEmail('');
setAdminPassword('');
setIsEditing(false);
setEditingOrgId(null);
setIsModalOpen(false);
};
const openEdit = (org: Organization) => {
setNewName(org.name);
setNewDomain(org.domain || '');
setEditingOrgId(org.id);
setIsEditing(true);
setIsModalOpen(true);
};
const openBranding = (org: Organization) => {
setSelectedOrg(org);
setPrimaryColor(org.primary_color || '#3B82F6');
setSecondaryColor(org.secondary_color || '#8B5CF6');
setIsBrandingModalOpen(true);
};
const handleLogoUpload = async (e: React.ChangeEvent<HTMLInputElement>) => {
const file = e.target.files?.[0];
if (!file || !selectedOrg) return;
setUploadingLogo(true);
try {
const resp = await cmsApi.uploadOrganizationLogo(selectedOrg.id, file);
setSelectedOrg({ ...selectedOrg, logo_url: resp.url });
// Update in list
setOrganizations(orgs => orgs.map(o => o.id === selectedOrg.id ? { ...o, logo_url: resp.url } : o));
} catch (error) {
console.error('Failed to upload logo', error);
alert('Failed to upload logo. Please try again.');
} finally {
setUploadingLogo(false);
}
};
const handleBrandingSave = async () => {
if (!selectedOrg) return;
setIsSavingBranding(true);
try {
await cmsApi.updateOrganizationBranding(selectedOrg.id, {
primary_color: primaryColor,
secondary_color: secondaryColor
});
// Update in list
setOrganizations(orgs => orgs.map(o => o.id === selectedOrg.id ? { ...o, primary_color: primaryColor, secondary_color: secondaryColor } : o));
setIsBrandingModalOpen(false);
} catch (error) {
console.error('Failed to update branding', error);
alert('Failed to update branding. Please try again.');
} finally {
setIsSavingBranding(false);
}
};
const openSSOConfig = async (org: Organization) => {
setSelectedOrg(org);
setIsSSOModalOpen(true);
// Temporarily set org in localStorage for API calls
localStorage.setItem('studio_selected_org_id', org.id);
try {
const config = await cmsApi.getSSOConfig();
if (config) {
setIssuerUrl(config.issuer_url);
setClientId(config.client_id);
setClientSecret(config.client_secret);
setSsoEnabled(config.enabled);
} else {
setIssuerUrl('');
setClientId('');
setClientSecret('');
setSsoEnabled(false);
}
} catch (error) {
console.error('Failed to load SSO config', error);
}
};
const handleSSOSave = async () => {
if (!selectedOrg) return;
setIsSavingSSO(true);
try {
await cmsApi.updateSSOConfig({
issuer_url: issuerUrl,
client_id: clientId,
client_secret: clientSecret,
enabled: ssoEnabled
});
setIsSSOModalOpen(false);
alert('SSO configuration saved successfully!');
} catch (error) {
console.error('Failed to save SSO config', error);
alert('Failed to save SSO config. Please ensure all fields are correct.');
} finally {
setIsSavingSSO(false);
}
};
if (user?.role !== 'admin') {
return (
<div className="flex flex-col items-center justify-center min-h-[60vh] text-center">
<div className="p-4 rounded-full bg-red-500/10 mb-4">
<ShieldCheck className="w-12 h-12 text-red-500" />
</div>
<h1 className="text-2xl font-bold mb-2">Access Denied</h1>
<p className="text-gray-600 dark:text-gray-400">Only system administrators can access this page.</p>
</div>
);
}
return (
<div className="space-y-6 md:space-y-8 animate-in fade-in duration-500 p-4 md:p-0">
<div className="flex flex-col sm:flex-row justify-between items-start sm:items-center gap-4">
<div>
<h1 className="text-2xl md:text-3xl font-bold tracking-tight text-slate-900 dark:text-white">Organizations</h1>
<p className="text-slate-600 dark:text-gray-400 mt-1 text-sm">Manage tenants and isolated environments.</p>
</div>
<button
onClick={() => { resetForm(); setIsModalOpen(true); }}
className="w-full sm:w-auto flex items-center justify-center gap-2 px-4 py-2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg transition-all shadow-lg shadow-blue-500/20 shadow-glow"
>
<Plus className="w-4 h-4" />
<span className="inline">New Organization</span>
</button>
</div>
{loading ? (
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
{[1, 2, 3].map(i => (
<div key={i} className="h-48 rounded-xl bg-slate-100 dark:bg-white/5 border border-slate-200 dark:border-white/10 animate-pulse" />
))}
</div>
) : (
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
{organizations.map((org) => (
<div
key={org.id}
className="group relative p-6 rounded-xl bg-white dark:bg-white/5 border border-slate-200 dark:border-white/10 hover:border-blue-500/50 transition-all hover:translate-y-[-2px] overflow-hidden shadow-sm dark:shadow-none"
>
<div className="absolute top-0 right-0 p-4 opacity-10 group-hover:opacity-20 transition-opacity text-slate-400 dark:text-white">
<Building2 className="w-16 h-16" />
</div>
<div className="flex items-start gap-4 mb-4">
<div className="p-3 rounded-lg bg-blue-500/10 text-blue-600 dark:text-blue-400 overflow-hidden w-12 h-12 flex items-center justify-center relative border border-blue-500/20">
{org.logo_url ? (
<Image src={getImageUrl(org.logo_url)} alt={org.name} fill className="object-contain" unoptimized />
) : (
<Building2 className="w-6 h-6" />
)}
</div>
<div className="flex-1 min-w-0">
<div className="flex items-center justify-between gap-2">
<h3 className="font-semibold text-lg text-slate-900 dark:text-white truncate">{org.name}</h3>
<button
onClick={() => openEdit(org)}
className="p-1.5 rounded-md hover:bg-slate-100 dark:hover:bg-white/10 text-slate-400 hover:text-slate-600 dark:hover:text-white transition-colors"
title="Edit Organization"
>
<Edit2 className="w-4 h-4" />
</button>
</div>
<div className="flex items-center gap-1.5 text-sm text-slate-500 dark:text-gray-400">
<Globe className="w-3 h-3" />
<span className="truncate">{org.domain || 'No custom domain'}</span>
</div>
</div>
</div>
<div className="flex gap-2 mt-4 mb-2">
<div className="flex-1 h-1 rounded-full opacity-60" style={{ backgroundColor: org.primary_color || '#3B82F6' }} title="Primary Color" />
<div className="flex-1 h-1 rounded-full opacity-60" style={{ backgroundColor: org.secondary_color || '#8B5CF6' }} title="Secondary Color" />
</div>
<div className="space-y-3 mt-4">
<div className="flex items-center justify-between text-xs text-slate-500 dark:text-gray-500 bg-slate-50 dark:bg-black/20 p-2 rounded-lg border border-slate-100 dark:border-white/5">
<div className="flex items-center gap-1">
<Calendar className="w-3 h-3" />
{new Date(org.created_at).toLocaleDateString()}
</div>
<div className="text-blue-600 dark:text-blue-500 font-mono">
{org.id.split('-')[0]}...
</div>
</div>
<div className="grid grid-cols-3 gap-2">
<button
onClick={() => openBranding(org)}
className="py-2 px-2 text-[10px] font-bold border border-blue-500/20 bg-blue-50 dark:bg-blue-500/5 hover:bg-blue-100 dark:hover:bg-blue-500/10 text-blue-600 dark:text-blue-400 rounded-lg transition-colors flex items-center justify-center gap-1"
>
<Palette className="w-3 h-3" /> BRAND
</button>
<button
onClick={() => openSSOConfig(org)}
className="py-2 px-2 text-[10px] font-bold border border-indigo-500/20 bg-indigo-50 dark:bg-indigo-500/5 hover:bg-indigo-100 dark:hover:bg-indigo-500/10 text-indigo-600 dark:text-indigo-400 rounded-lg transition-colors flex items-center justify-center gap-1"
>
<Fingerprint className="w-3 h-3" /> SSO
</button>
<button className="py-2 px-2 text-[10px] font-bold border border-slate-200 dark:border-white/5 bg-slate-50 dark:bg-white/5 hover:bg-slate-100 dark:hover:bg-white/10 rounded-lg transition-colors flex items-center justify-center gap-1 text-slate-600 dark:text-gray-400">
DOCS <ExternalLink className="w-3 h-3" />
</button>
</div>
</div>
</div>
))}
</div>
)}
{/* Create/Edit Organization Modal */}
{isModalOpen && (
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4 bg-slate-900/60 backdrop-blur-sm animate-in fade-in duration-200">
<div className="w-full max-w-md bg-white dark:bg-slate-900 border border-slate-200 dark:border-white/10 rounded-2xl p-8 shadow-2xl overflow-y-auto max-h-[90vh]">
<div className="flex justify-between items-center mb-6">
<h2 className="text-xl font-bold text-slate-900 dark:text-white">
{isEditing ? 'Edit Organization' : 'Create New Organization'}
</h2>
<button onClick={resetForm} className="p-2 text-slate-400 hover:text-slate-600 dark:hover:text-white rounded-full transition-colors">
<X className="w-5 h-5" />
</button>
</div>
<form onSubmit={handleCreate} className="space-y-4">
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Organization Name</label>
<input
type="text"
required
value={newName}
onChange={(e) => setNewName(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg px-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all text-slate-900 dark:text-white placeholder-slate-400"
placeholder="e.g. Acme Corp"
/>
</div>
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Domain (Optional)</label>
<div className="relative">
<Globe className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400" />
<input
type="text"
value={newDomain}
onChange={(e) => setNewDomain(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg pl-10 pr-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all font-mono text-sm text-slate-900 dark:text-white placeholder-slate-400"
placeholder="e.g. acme.com"
/>
</div>
</div>
{!isEditing && (
<div className="pt-4 border-t border-slate-100 dark:border-white/5">
<h3 className="text-xs font-black uppercase tracking-widest text-blue-600 dark:text-blue-500 mb-4">Initial Administrator</h3>
<div className="space-y-4">
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Admin Full Name</label>
<input
type="text"
required
value={adminFullName}
onChange={(e) => setAdminFullName(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg px-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all text-slate-900 dark:text-white placeholder-slate-400"
placeholder="e.g. John Doe"
/>
</div>
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Admin Email</label>
<input
type="email"
required
value={adminEmail}
onChange={(e) => setAdminEmail(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg px-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all text-slate-900 dark:text-white placeholder-slate-400"
placeholder="admin@acme.com"
/>
</div>
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Admin Password</label>
<input
type="password"
required
value={adminPassword}
onChange={(e) => setAdminPassword(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg px-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all text-slate-900 dark:text-white placeholder-slate-400"
placeholder="••••••••"
/>
</div>
</div>
</div>
)}
<div className="flex gap-3 mt-8">
<button
type="button"
onClick={resetForm}
className="flex-1 px-4 py-2.5 bg-slate-50 dark:bg-white/5 hover:bg-slate-100 dark:hover:bg-white/10 border border-slate-200 dark:border-white/10 rounded-lg transition-all text-slate-600 dark:text-white font-medium"
>
Cancel
</button>
<button
type="submit"
className="flex-1 px-4 py-2.5 bg-blue-600 hover:bg-blue-500 text-white rounded-lg transition-all shadow-lg shadow-blue-500/20 font-bold"
>
{isEditing ? 'Save Changes' : 'Create'}
</button>
</div>
</form>
</div>
</div>
)}
{/* Branding Management Modal */}
{isBrandingModalOpen && selectedOrg && (
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4 bg-slate-900/60 backdrop-blur-sm animate-in fade-in duration-200">
<div className="w-full max-w-2xl bg-white dark:bg-slate-900 border border-slate-200 dark:border-white/10 rounded-2xl p-8 shadow-2xl">
<div className="flex justify-between items-center mb-6">
<div>
<h2 className="text-xl font-bold text-slate-900 dark:text-white">Branding Management</h2>
<p className="text-sm text-slate-600 dark:text-gray-400">{selectedOrg.name}</p>
</div>
<button onClick={() => setIsBrandingModalOpen(false)} className="p-2 text-slate-400 hover:text-slate-600 dark:hover:text-white rounded-full transition-colors">
<X className="w-5 h-5" />
</button>
</div>
<div className="grid grid-cols-1 md:grid-cols-2 gap-8">
<div className="space-y-6">
{/* Logo Upload */}
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-3 uppercase tracking-wider text-[10px] font-black">Organization Logo</label>
<div className="flex items-center gap-4">
<div className="w-20 h-20 rounded-xl bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 flex items-center justify-center overflow-hidden relative">
{selectedOrg.logo_url ? (
<Image src={getImageUrl(selectedOrg.logo_url)} alt="Preview" fill className="object-contain" unoptimized />
) : (
<Building2 className="w-8 h-8 text-slate-300 dark:text-gray-600" />
)}
</div>
<div className="flex-1">
<label className="relative flex items-center justify-center gap-2 px-4 py-2 bg-blue-600/10 hover:bg-blue-600/20 text-blue-600 dark:text-blue-400 rounded-lg cursor-pointer transition-all border border-blue-500/20 font-bold text-xs uppercase">
<Upload className="w-4 h-4" />
{uploadingLogo ? 'Uploading...' : 'Upload Logo'}
<input type="file" className="hidden" accept="image/*" onChange={handleLogoUpload} disabled={uploadingLogo} />
</label>
<p className="text-[10px] text-slate-500 mt-2">PNG, JPG or SVG. Max 2MB.</p>
</div>
</div>
</div>
{/* Colors */}
<div className="space-y-4">
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-2">Primary Color</label>
<div className="flex gap-2">
<input
type="color"
value={primaryColor}
onChange={(e) => setPrimaryColor(e.target.value)}
className="w-10 h-10 rounded cursor-pointer border border-slate-200 dark:border-white/10 p-1 bg-white dark:bg-black/40"
/>
<input
type="text"
value={primaryColor}
onChange={(e) => setPrimaryColor(e.target.value)}
className="flex-1 bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg px-3 py-2 text-sm font-mono text-slate-900 dark:text-white"
/>
</div>
</div>
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-2">Secondary Color</label>
<div className="flex gap-2">
<input
type="color"
value={secondaryColor}
onChange={(e) => setSecondaryColor(e.target.value)}
className="w-10 h-10 rounded cursor-pointer border border-slate-200 dark:border-white/10 p-1 bg-white dark:bg-black/40"
/>
<input
type="text"
value={secondaryColor}
onChange={(e) => setSecondaryColor(e.target.value)}
className="flex-1 bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg px-3 py-2 text-sm font-mono text-slate-900 dark:text-white"
/>
</div>
</div>
</div>
</div>
{/* Live Preview */}
<div className="space-y-4">
<label className="block text-sm font-medium text-slate-700 dark:text-gray-400 mb-2 uppercase tracking-wider text-[10px] font-black">Portal Preview</label>
<div className="rounded-xl border border-slate-200 dark:border-white/10 overflow-hidden bg-white dark:bg-slate-950 shadow-inner">
{/* Mock Experience Header */}
<div className="h-10 px-4 flex items-center justify-between border-b border-black/5" style={{ backgroundColor: primaryColor }}>
<div className="flex items-center gap-2">
<div className="w-5 h-5 bg-white/20 rounded flex items-center justify-center overflow-hidden relative">
{selectedOrg.logo_url ? (
<Image src={getImageUrl(selectedOrg.logo_url)} alt="Logo" fill className="object-contain" unoptimized />
) : <div className="w-3 h-3 bg-white" />}
</div>
<div className="w-16 h-2 bg-white/30 rounded" />
</div>
<div className="flex gap-2">
<div className="w-6 h-2 bg-white/20 rounded" />
<div className="w-6 h-2 bg-white/20 rounded" />
</div>
</div>
{/* Mock Experience Content */}
<div className="p-4 space-y-3 bg-slate-50 dark:bg-[#0a0c10]">
<div className="w-2/3 h-4 bg-slate-200 dark:bg-white/10 rounded mb-2" />
<div className="w-full h-24 bg-white dark:bg-white/5 rounded-lg border border-slate-200 dark:border-white/5 p-3 shadow-sm">
<div className="w-1/3 h-3 rounded mb-2" style={{ backgroundColor: secondaryColor }} />
<div className="w-full h-2 bg-slate-100 dark:bg-white/5 rounded mb-1" />
<div className="w-full h-2 bg-slate-100 dark:bg-white/5 rounded mb-1" />
<div className="w-1/2 h-2 bg-slate-100 dark:bg-white/5 rounded" />
<div className="mt-4 flex justify-end">
<div className="px-3 py-1.5 rounded text-[8px] font-bold text-white shadow-sm" style={{ backgroundColor: primaryColor }}>
GET STARTED
</div>
</div>
</div>
</div>
</div>
<div className="p-3 rounded-lg bg-blue-50 dark:bg-blue-500/10 border border-blue-200 dark:border-blue-500/20">
<p className="text-[10px] text-blue-700 dark:text-blue-400 leading-relaxed font-medium">
Real-time preview of the brand application.
</p>
</div>
</div>
</div>
<div className="flex gap-3 mt-10">
<button
onClick={() => setIsBrandingModalOpen(false)}
className="flex-1 px-4 py-3 bg-slate-50 dark:bg-white/5 hover:bg-slate-100 dark:hover:bg-white/10 border border-slate-200 dark:border-white/10 rounded-xl transition-all font-medium text-slate-600 dark:text-white"
>
Cancel
</button>
<button
onClick={handleBrandingSave}
disabled={isSavingBranding}
className="flex-[2] px-8 py-3 bg-blue-600 hover:bg-blue-500 text-white rounded-xl transition-all shadow-lg shadow-blue-500/20 font-bold flex items-center justify-center gap-2 uppercase tracking-wide"
>
{isSavingBranding ? <div className="w-5 h-5 border-2 border-white/20 border-t-white rounded-full animate-spin" /> : <Save className="w-5 h-5" />}
Save Branding
</button>
</div>
</div>
</div>
)}
{/* SSO Configuration Modal */}
{isSSOModalOpen && selectedOrg && (
<div className="fixed inset-0 z-[100] flex items-center justify-center p-4 bg-slate-900/60 backdrop-blur-sm animate-in fade-in duration-200">
<div className="w-full max-w-xl bg-white dark:bg-slate-900 border border-slate-200 dark:border-white/10 rounded-2xl p-8 shadow-2xl">
<div className="flex justify-between items-center mb-6">
<div className="flex items-center gap-3">
<div className="p-2 rounded-lg bg-indigo-500/10 text-indigo-600 dark:text-indigo-400">
<Fingerprint className="w-6 h-6" />
</div>
<div>
<h2 className="text-xl font-bold text-slate-900 dark:text-white">Single Sign-On (OIDC)</h2>
<p className="text-sm text-slate-600 dark:text-gray-400">{selectedOrg.name}</p>
</div>
</div>
<button onClick={() => setIsSSOModalOpen(false)} className="p-2 text-slate-400 hover:text-slate-600 dark:hover:text-white rounded-full transition-colors">
<X className="w-5 h-5" />
</button>
</div>
<div className="space-y-4">
<div className="flex items-center justify-between p-4 rounded-xl bg-slate-50 dark:bg-white/5 border border-slate-200 dark:border-white/10">
<div className="pr-4">
<h3 className="font-semibold text-slate-900 dark:text-white">Enable OIDC SSO</h3>
<p className="text-xs text-slate-500 dark:text-gray-500">Allow users to log in via your identity provider.</p>
</div>
<button
onClick={() => setSsoEnabled(!ssoEnabled)}
className={`w-12 h-6 rounded-full transition-colors relative shrink-0 ${ssoEnabled ? 'bg-blue-600' : 'bg-slate-300 dark:bg-gray-700'}`}
>
<div className={`absolute top-1 w-4 h-4 bg-white rounded-full transition-all ${ssoEnabled ? 'right-1' : 'left-1'}`} />
</button>
</div>
<div className="space-y-4 pt-2">
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Issuer URL</label>
<div className="relative">
<Globe className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400" />
<input
type="url"
value={issuerUrl}
onChange={(e) => setIssuerUrl(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg pl-10 pr-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all font-mono text-sm text-slate-900 dark:text-white placeholder-slate-400"
placeholder="https://accounts.google.com"
/>
</div>
</div>
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Client ID</label>
<div className="relative">
<ShieldCheck className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400" />
<input
type="text"
value={clientId}
onChange={(e) => setClientId(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg pl-10 pr-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all font-mono text-sm text-slate-900 dark:text-white placeholder-slate-400"
placeholder="Your OIDC Client ID"
/>
</div>
</div>
<div>
<label className="block text-sm font-medium text-slate-600 dark:text-gray-400 mb-1.5">Client Secret</label>
<div className="relative">
<Key className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-slate-400" />
<input
type="password"
value={clientSecret}
onChange={(e) => setClientSecret(e.target.value)}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-200 dark:border-white/10 rounded-lg pl-10 pr-4 py-2.5 focus:outline-none focus:ring-2 focus:ring-blue-500/50 transition-all font-mono text-sm text-slate-900 dark:text-white placeholder-slate-400"
placeholder="••••••••••••••••"
/>
</div>
</div>
<div className="p-4 rounded-xl bg-blue-50 dark:bg-blue-500/10 border border-blue-200 dark:border-blue-500/20 space-y-2">
<div className="flex items-center gap-2 text-blue-700 dark:text-blue-400 text-xs font-black uppercase tracking-wider">
<Settings2 className="w-4 h-4" /> CONFIGURATION STEPS
</div>
<p className="text-[10px] text-blue-800 dark:text-blue-300 leading-relaxed font-medium">
1. Register OpenCCB in your IDP.<br />
2. Redirect URI: <span className="font-mono bg-blue-200 dark:bg-blue-500/20 px-1 rounded">{API_BASE_URL}/auth/sso/callback</span><br />
3. Copy the Issuer, ID and Secret here.
</p>
</div>
</div>
</div>
<div className="flex gap-3 mt-8">
<button
onClick={() => setIsSSOModalOpen(false)}
className="flex-1 px-4 py-3 bg-slate-50 dark:bg-white/5 hover:bg-slate-100 dark:hover:bg-white/10 border border-slate-200 dark:border-white/10 rounded-xl transition-all font-medium text-slate-600 dark:text-white"
>
Cancel
</button>
<button
onClick={handleSSOSave}
disabled={isSavingSSO}
className="flex-[2] px-8 py-3 bg-blue-600 hover:bg-blue-500 text-white rounded-xl transition-all shadow-lg shadow-blue-500/20 font-bold flex items-center justify-center gap-2 uppercase tracking-wide"
>
{isSavingSSO ? <div className="w-5 h-5 border-2 border-white/20 border-t-white rounded-full animate-spin" /> : <Save className="w-5 h-5" />}
Save SSO Settings
</button>
</div>
</div>
</div>
)}
</div>
);
}
+3 -17
View File
@@ -5,13 +5,12 @@ import { cmsApi } from "@/lib/api";
import { useAuth } from "@/context/AuthContext";
import { useRouter } from "next/navigation";
import Link from "next/link";
import { UserPlus, Mail, Lock, User, Building2 } from "lucide-react";
import { UserPlus, Mail, Lock, User } from "lucide-react";
export default function RegisterPage() {
const [email, setEmail] = useState("");
const [password, setPassword] = useState("");
const [fullName, setFullName] = useState("");
const [organizationName, setOrganizationName] = useState("");
const [error, setError] = useState("");
const [loading, setLoading] = useState(false);
@@ -23,7 +22,7 @@ export default function RegisterPage() {
setLoading(true);
setError("");
try {
const res = await cmsApi.register({ email, password, full_name: fullName, organization_name: organizationName });
const res = await cmsApi.register({ email, password, full_name: fullName });
login(res.user, res.token);
router.push("/");
} catch (err) {
@@ -98,20 +97,7 @@ export default function RegisterPage() {
</div>
</div>
<div className="space-y-2">
<label className="text-[10px] font-black uppercase tracking-widest text-gray-500 px-1">Organization Name (Optional)</label>
<div className="relative">
<Building2 className="absolute left-4 top-1/2 -translate-y-1/2 text-gray-500" size={18} />
<input
type="text"
value={organizationName}
onChange={(e) => setOrganizationName(e.target.value)}
placeholder="Your School or Company"
className="w-full bg-white/5 border border-white/10 rounded-xl py-4 pl-12 pr-4 text-sm text-white focus:outline-none focus:border-blue-500 transition-all placeholder:text-gray-700"
/>
</div>
<p className="text-[10px] text-gray-600 px-1">If blank, we&apos;ll use your email domain.</p>
</div>
<button
disabled={loading}
@@ -38,6 +38,7 @@ import MemoryBlock from "@/components/blocks/MemoryBlock";
import RolePlayingBlock from "@/components/blocks/RolePlayingBlock";
import PeerReviewBlock from "@/components/blocks/PeerReviewBlock";
import MermaidBlock from "@/components/blocks/MermaidBlock";
import CodeLabBlock from "@/components/blocks/CodeLabBlock";
import SaveToLibraryModal from "@/components/modals/SaveToLibraryModal";
import LibraryPanel from "@/components/LibraryPanel";
import Modal from "@/components/Modal";
@@ -1106,6 +1107,20 @@ export default function LessonEditor({ params }: { params: { id: string; lessonI
onUpdate={(updates) => updateBlock(block.id, updates)}
/>
)}
{block.type === 'code-lab' && (
<CodeLabBlock
id={block.id}
title={block.title}
language={block.language}
instructions={block.instructions}
initial_code={block.initial_code}
solution={block.solution}
test_cases={block.test_cases}
editMode={editMode}
lessonId={params.lessonId}
onChange={(updates) => updateBlock(block.id, updates)}
/>
)}
</div>
</div>
))}
@@ -1148,6 +1163,7 @@ export default function LessonEditor({ params }: { params: { id: string; lessonI
{ type: 'audio-response', icon: '🎤', label: 'Oral Practice', color: 'blue' },
{ type: 'memory-match', icon: '🧩', label: 'Logic Game', color: 'indigo' },
{ type: 'peer-review', icon: '👥', label: 'Peer Review', color: 'slate' },
{ type: 'code-lab', icon: '🧑‍💻', label: 'Code Lab', color: 'indigo' },
{ type: 'mermaid', icon: '📊', label: 'Mermaid Diagram', color: 'indigo' },
{ type: 'role-playing', icon: '🎭', label: 'Role-Playing AI', color: 'purple' },
].map((item) => (
+5 -36
View File
@@ -21,7 +21,6 @@ import {
Send,
} from "lucide-react";
import CourseEditorLayout from "@/components/CourseEditorLayout";
import OrganizationSelector from "@/components/OrganizationSelector";
interface FullModule extends Module {
lessons: Lesson[];
@@ -35,8 +34,6 @@ export default function CourseEditor({ params }: { params: { id: string } }) {
const [error, setError] = useState<string | null>(null);
const [editingId, setEditingId] = useState<string | null>(null);
const [editValue, setEditValue] = useState("");
const [organizations, setOrganizations] = useState<Organization[]>([]);
const [isOrgModalOpen, setIsOrgModalOpen] = useState(false);
const [saving, setSaving] = useState(false); // Added saving state
const { user } = useAuth();
@@ -63,19 +60,7 @@ export default function CourseEditor({ params }: { params: { id: string } }) {
loadData();
}, [params.id]);
useEffect(() => {
const loadOrgs = async () => {
if (user?.role === 'admin' && user?.organization_id === '00000000-0000-0000-0000-000000000001') {
try {
const orgs = await cmsApi.getOrganizations();
setOrganizations(orgs);
} catch (err) {
console.error("Failed to load organizations", err);
}
}
};
loadOrgs();
}, [user]);
const handleAddModule = async () => {
const title = "";
@@ -194,27 +179,19 @@ export default function CourseEditor({ params }: { params: { id: string } }) {
const handlePublish = async () => {
if (!course) return;
const isSuperAdmin = user?.role === 'admin' && user?.organization_id === '00000000-0000-0000-0000-000000000001';
if (isSuperAdmin && organizations.length > 0) {
setIsOrgModalOpen(true);
} else {
publishCourse();
}
publishCourse();
};
const publishCourse = async (targetOrgId?: string) => {
const publishCourse = async () => {
try {
setSaving(true);
await cmsApi.publishCourse(params.id as string, targetOrgId);
await cmsApi.publishCourse(params.id as string);
alert("Course published successfully!");
} catch (err) {
console.error("Failed to publish course", err);
alert("Failed to publish course.");
} finally {
setSaving(false);
setIsOrgModalOpen(false); // Close modal after publishing attempt
}
};
@@ -429,15 +406,7 @@ export default function CourseEditor({ params }: { params: { id: string } }) {
</button>
</div>
</CourseEditorLayout>
{/* Organization Selector Modal */}
<OrganizationSelector
isOpen={isOrgModalOpen}
onClose={() => setIsOrgModalOpen(false)}
organizations={organizations}
title="Publish to Organization"
actionLabel="Publish Course"
onConfirm={(orgId) => publishCourse(orgId)}
/>
</>
);
}
+5 -34
View File
@@ -6,7 +6,6 @@ import Link from "next/link";
import { useAuth } from "@/context/AuthContext";
import { useTranslation } from "@/context/I18nContext";
import { Plus, BookOpen, Download, Upload, Sparkles, Wand2, Trash2 } from "lucide-react";
import OrganizationSelector from "@/components/OrganizationSelector";
import Modal from "@/components/Modal";
import PageLayout from "@/components/PageLayout";
@@ -34,27 +33,13 @@ export default function StudioDashboard() {
loadCourses();
}, [user]);
const [organizations, setOrganizations] = useState<Organization[]>([]);
const [isOrgModalOpen, setIsOrgModalOpen] = useState(false);
const [isTitleModalOpen, setIsTitleModalOpen] = useState(false);
const [isAIModalOpen, setIsAIModalOpen] = useState(false);
const [newCourseTitle, setNewCourseTitle] = useState("");
const [aiPrompt, setAiPrompt] = useState("");
const [isGenerating, setIsGenerating] = useState(false);
useEffect(() => {
const loadOrgs = async () => {
if (user?.role === 'admin' && user?.organization_id === '00000000-0000-0000-0000-000000000001') {
try {
const orgs = await cmsApi.getOrganizations();
setOrganizations(orgs);
} catch (err) {
console.error("Failed to load organizations", err);
}
}
};
loadOrgs();
}, [user]);
const handleCreateCourse = async () => {
setIsTitleModalOpen(true);
@@ -64,18 +49,12 @@ export default function StudioDashboard() {
e.preventDefault();
if (!newCourseTitle) return;
setIsTitleModalOpen(false);
const isSuperAdmin = user?.role === 'admin' && user?.organization_id === '00000000-0000-0000-0000-000000000001';
if (isSuperAdmin && organizations.length > 0) {
setIsOrgModalOpen(true);
} else {
createCourse();
}
createCourse();
};
const createCourse = async (targetOrgId?: string) => {
const createCourse = async () => {
try {
const newCourse = await cmsApi.createCourse(newCourseTitle, targetOrgId);
const newCourse = await cmsApi.createCourse(newCourseTitle);
setCourses((prev: Course[]) => [...prev, newCourse]);
setNewCourseTitle("");
} catch (err) {
@@ -331,15 +310,7 @@ export default function StudioDashboard() {
</form>
</Modal>
{/* Organization Selector Modal */}
<OrganizationSelector
isOpen={isOrgModalOpen}
onClose={() => setIsOrgModalOpen(false)}
organizations={organizations}
title="Target Organization"
actionLabel="Create Course"
onConfirm={(orgId) => createCourse(orgId)}
/>
</PageLayout>
);
}
@@ -45,7 +45,7 @@ export default function BrandingSettings() {
if (!org) return;
setSaving(true);
try {
await cmsApi.updateOrganizationBranding(org.id, formData);
await cmsApi.updateOrganizationBranding(formData);
fetchOrg();
alert("Branding updated successfully!");
router.refresh();
@@ -117,7 +117,7 @@ export default function BrandingSettings() {
accept="image/png,image/jpeg,image/svg+xml"
currentUrl={org.logo_url}
customUploadFn={async (file) => {
const res = await cmsApi.uploadOrganizationLogo(org.id, file);
const res = await cmsApi.uploadOrganizationLogo(file);
return { url: res.logo_url || "" };
}}
onUploadComplete={(url) => {
@@ -151,7 +151,7 @@ export default function BrandingSettings() {
accept="image/png,image/x-icon,image/svg+xml,image/jpeg"
currentUrl={org.favicon_url}
customUploadFn={async (file) => {
const res = await cmsApi.uploadOrganizationFavicon(org.id, file);
const res = await cmsApi.uploadOrganizationFavicon(file);
return { url: res.favicon_url || "" };
}}
onUploadComplete={(url) => {
@@ -1,70 +0,0 @@
"use client";
import React, { useState } from "react";
import Modal from "./Modal";
import Combobox from "./Combobox";
import { Organization } from "@/lib/api";
interface OrganizationSelectorProps {
isOpen: boolean;
onClose: () => void;
organizations: Organization[];
onConfirm: (orgId: string | undefined) => void;
title: string;
actionLabel: string;
}
export default function OrganizationSelector({
isOpen,
onClose,
organizations,
onConfirm,
title,
actionLabel
}: OrganizationSelectorProps) {
const [selectedId, setSelectedId] = useState<string>("");
const handleConfirm = () => {
onConfirm(selectedId || undefined);
onClose();
};
return (
<Modal isOpen={isOpen} onClose={onClose} title={title}>
<div className="space-y-6">
<fieldset>
<legend className="sr-only">Organization Selection</legend>
<label htmlFor="organization-select" className="block text-sm font-medium text-gray-400 mb-2">
Target Organization
</label>
<Combobox
id="organization-select"
options={organizations}
value={selectedId}
onChange={setSelectedId}
placeholder="Search or Select Organization..."
/>
</fieldset>
<p className="mt-3 text-xs text-gray-500 italic">
Leave empty to use the Default Organization.
</p>
<div className="flex gap-3 pt-2">
<button
onClick={onClose}
className="flex-1 px-4 py-2.5 bg-white/5 hover:bg-white/10 border border-white/10 rounded-lg transition-all text-sm font-medium"
>
Cancel
</button>
<button
onClick={handleConfirm}
className="flex-[2] px-4 py-2.5 bg-blue-600 hover:bg-blue-500 text-white rounded-lg transition-all shadow-lg shadow-blue-500/20 font-bold text-sm"
>
{actionLabel}
</button>
</div>
</div>
</Modal>
);
}
@@ -0,0 +1,251 @@
"use client";
import { useState } from "react";
import { Wand2, Loader2, Code2, Plus, Trash2 } from "lucide-react";
import { cmsApi } from "@/lib/api";
interface CodeLabBlockProps {
id: string;
title?: string;
language?: string;
instructions?: string;
initial_code?: string;
solution?: string;
test_cases?: { description: string; expected: string }[];
editMode: boolean;
lessonId: string;
onChange: (updates: {
title?: string;
language?: string;
instructions?: string;
initial_code?: string;
solution?: string;
test_cases?: { description: string; expected: string }[]
}) => void;
}
export default function CodeLabBlock({
id,
title = "",
language = "python",
instructions = "",
initial_code = "",
solution = "",
test_cases = [],
editMode,
lessonId,
onChange
}: CodeLabBlockProps) {
const [isGenerating, setIsGenerating] = useState(false);
const [promptHint, setPromptHint] = useState("");
const handleGenerateAI = async () => {
setIsGenerating(true);
try {
const data = await cmsApi.generateCodeLab(lessonId, {
language,
prompt_hint: promptHint || undefined
});
onChange({
title: data.title,
instructions: data.instructions,
initial_code: data.initial_code,
solution: data.solution,
test_cases: data.test_cases,
language: data.language
});
} catch (error) {
console.error("AI Code Lab Generation failed:", error);
alert("No se pudo generar el laboratorio con IA. Por favor, intenta de nuevo.");
} finally {
setIsGenerating(false);
}
};
const addTestCase = () => {
const newTestCases = [...test_cases, { description: "", expected: "" }];
onChange({ test_cases: newTestCases });
};
const updateTestCase = (index: number, field: "description" | "expected", value: string) => {
const newTestCases = [...test_cases];
newTestCases[index] = { ...newTestCases[index], [field]: value };
onChange({ test_cases: newTestCases });
};
const removeTestCase = (index: number) => {
const newTestCases = test_cases.filter((_, i) => i !== index);
onChange({ test_cases: newTestCases });
};
if (!editMode) {
return (
<div className="space-y-6" id={id}>
<div className="flex items-center gap-4">
<div className="p-3 rounded-2xl bg-indigo-50 dark:bg-indigo-500/10 text-indigo-600 dark:text-indigo-400 shadow-inner">
<Code2 size={24} />
</div>
<div>
<h3 className="text-2xl font-black italic tracking-tight text-slate-900 dark:text-white uppercase">
{title || "Laboratorio de Código"}
</h3>
<p className="text-[10px] text-slate-400 dark:text-gray-500 uppercase tracking-[0.2em] font-black">
{language.toUpperCase()} Ejercicio Interactivo
</p>
</div>
</div>
<div className="p-8 rounded-[3rem] border border-slate-100 dark:border-white/5 bg-white dark:bg-black/20 shadow-xl">
<div className="prose dark:prose-invert max-w-none text-slate-600 dark:text-gray-300 mb-8">
{instructions || "Sigue las instrucciones del editor para completar el desafío."}
</div>
<div className="bg-slate-900 rounded-2xl p-6 font-mono text-sm text-indigo-100 mb-6">
<pre className="whitespace-pre-wrap">{initial_code || "# El código inicial aparecerá aquí"}</pre>
</div>
<div className="flex items-center gap-2 text-[10px] font-black uppercase tracking-widest text-slate-400">
<span className="w-2 h-2 rounded-full bg-emerald-500" />
Vista Previa del Estudiante activada
</div>
</div>
</div>
);
}
return (
<div className="space-y-6" id={id}>
<div className="p-10 bg-white dark:bg-white/5 border border-slate-100 dark:border-white/10 space-y-10 rounded-[3rem] shadow-xl relative overflow-hidden group/cleditor">
<div className="absolute top-0 right-0 w-64 h-64 bg-indigo-500/5 rounded-full blur-[80px] -translate-y-1/2 translate-x-1/2" />
<div className="grid grid-cols-1 md:grid-cols-2 gap-10 relative z-10">
<div className="space-y-6">
<div className="space-y-3">
<label className="text-[10px] font-black uppercase tracking-[0.2em] text-slate-400 pl-1">Título del Desafío</label>
<input
type="text"
value={title}
onChange={(e) => onChange({ title: e.target.value })}
placeholder="Ej. Calculadora de Factoriales..."
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-100 dark:border-white/10 rounded-2xl px-6 py-4 text-sm font-black uppercase text-slate-800 dark:text-white outline-none focus:ring-4 focus:ring-indigo-500/10 focus:border-indigo-500 transition-all"
/>
</div>
<div className="space-y-3">
<label className="text-[10px] font-black uppercase tracking-[0.2em] text-slate-400 pl-1">Lenguaje de Programación</label>
<select
value={language}
onChange={(e) => onChange({ language: e.target.value })}
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-100 dark:border-white/10 rounded-2xl px-6 py-4 text-sm font-bold text-slate-700 dark:text-gray-300 outline-none focus:ring-4 focus:ring-indigo-500/10 focus:border-indigo-500 transition-all appearance-none"
>
<option value="python">Python</option>
<option value="javascript">JavaScript</option>
<option value="sql">SQL</option>
<option value="bash">Bash</option>
</select>
</div>
<div className="space-y-3">
<label className="text-[10px] font-black uppercase tracking-[0.2em] text-slate-400 pl-1">Instrucciones</label>
<textarea
value={instructions}
onChange={(e) => onChange({ instructions: e.target.value })}
placeholder="Describe qué debe hacer el estudiante..."
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-100 dark:border-white/10 rounded-2xl px-6 py-4 text-sm text-slate-700 dark:text-gray-300 min-h-[150px] outline-none focus:ring-4 focus:ring-indigo-500/10 focus:border-indigo-500 transition-all"
/>
</div>
<div className="pt-6 border-t border-slate-100 dark:border-white/5 space-y-4">
<h4 className="text-sm font-black text-slate-800 dark:text-white uppercase tracking-tight">Generación con IA</h4>
<textarea
value={promptHint}
onChange={(e) => setPromptHint(e.target.value)}
placeholder="Ej. Crea un ejercicio sobre bucles for que use una lista de tareas..."
className="w-full bg-slate-50 dark:bg-black/40 border border-slate-100 dark:border-white/10 rounded-2xl px-6 py-4 text-sm text-slate-700 dark:text-gray-300 min-h-[80px] outline-none focus:ring-4 focus:ring-indigo-500/10 focus:border-indigo-500 transition-all"
/>
<button
onClick={handleGenerateAI}
disabled={isGenerating}
className="flex w-full justify-center items-center gap-2 px-6 py-4 bg-indigo-600 text-white rounded-2xl text-[11px] font-black uppercase tracking-widest hover:bg-indigo-700 transition-all disabled:opacity-50 shadow-xl shadow-indigo-500/20"
>
{isGenerating ? <Loader2 className="animate-spin" size={16} /> : <Wand2 size={16} />}
{isGenerating ? "Generando Laboratorio..." : "Auto-Generar con IA"}
</button>
</div>
</div>
<div className="space-y-8 flex flex-col">
<div className="space-y-3">
<label className="text-[10px] font-black uppercase tracking-[0.2em] text-slate-400 pl-1">Código Inicial (Para el estudiante)</label>
<textarea
value={initial_code}
onChange={(e) => onChange({ initial_code: e.target.value })}
placeholder="# Escribe el código base con TODOs..."
className="w-full bg-slate-900 border border-slate-800 rounded-2xl p-6 font-mono text-xs text-emerald-400 min-h-[200px] outline-none focus:ring-4 focus:ring-indigo-500/10 transition-all"
spellCheck={false}
/>
</div>
<div className="space-y-3">
<label className="text-[10px] font-black uppercase tracking-[0.2em] text-slate-400 pl-1">Solución (Oculta)</label>
<textarea
value={solution}
onChange={(e) => onChange({ solution: e.target.value })}
placeholder="# Escribe la solución completa..."
className="w-full bg-slate-900 border border-slate-800 rounded-2xl p-6 font-mono text-xs text-amber-400 min-h-[200px] outline-none focus:ring-4 focus:ring-indigo-500/10 transition-all"
spellCheck={false}
/>
</div>
</div>
</div>
<div className="pt-8 border-t border-slate-100 dark:border-white/5">
<div className="flex items-center justify-between mb-6">
<label className="text-[10px] font-black uppercase tracking-[0.2em] text-slate-400 pl-1">Casos de Prueba</label>
<button
onClick={addTestCase}
className="flex items-center gap-2 px-4 py-2 bg-slate-100 dark:bg-white/5 hover:bg-indigo-600 hover:text-white text-slate-600 dark:text-gray-300 rounded-xl text-[10px] font-black uppercase tracking-widest transition-all"
>
<Plus size={14} /> Añadir Caso
</button>
</div>
<div className="grid grid-cols-1 gap-4">
{test_cases.map((tc, index) => (
<div key={index} className="flex gap-4 items-start bg-slate-50 dark:bg-black/20 p-6 rounded-2xl border border-slate-100 dark:border-white/5 group/tc">
<div className="flex-1 grid grid-cols-1 md:grid-cols-2 gap-4">
<div className="space-y-2">
<span className="text-[9px] font-black uppercase text-slate-400">Descripción del Caso</span>
<input
type="text"
value={tc.description}
onChange={(e) => updateTestCase(index, "description", e.target.value)}
placeholder="Ej. Entrada válida: 5"
className="w-full bg-white dark:bg-black/40 border border-slate-100 dark:border-white/10 rounded-xl px-4 py-2 text-xs font-bold"
/>
</div>
<div className="space-y-2">
<span className="text-[9px] font-black uppercase text-slate-400">Resultado Esperado (Texto)</span>
<input
type="text"
value={tc.expected}
onChange={(e) => updateTestCase(index, "expected", e.target.value)}
placeholder="Ej. 120"
className="w-full bg-white dark:bg-black/40 border border-slate-100 dark:border-white/10 rounded-xl px-4 py-2 text-xs font-bold"
/>
</div>
</div>
<button
onClick={() => removeTestCase(index)}
className="p-2 text-slate-300 hover:text-red-500 opacity-0 group-hover/tc:opacity-100 transition-all mt-6"
>
<Trash2 size={16} />
</button>
</div>
))}
</div>
</div>
</div>
</div>
);
}
+31 -51
View File
@@ -68,7 +68,7 @@ export interface QuizQuestion {
export interface Block {
id: string;
type: 'description' | 'media' | 'quiz' | 'fill-in-the-blanks' | 'matching' | 'ordering' | 'short-answer' | 'document' | 'video_marker' | 'audio-response' | 'memory-match' | 'hotspot' | 'peer-review' | 'role-playing' | 'mermaid';
type: 'description' | 'media' | 'quiz' | 'fill-in-the-blanks' | 'matching' | 'ordering' | 'short-answer' | 'document' | 'video_marker' | 'audio-response' | 'memory-match' | 'hotspot' | 'peer-review' | 'role-playing' | 'mermaid' | 'code-lab';
title?: string;
content?: string;
url?: string;
@@ -107,6 +107,12 @@ export interface Block {
initial_message?: string;
// Mermaid fields
mermaid_code?: string;
// Code Lab fields
language?: string;
instructions?: string;
initial_code?: string;
solution?: string;
test_cases?: { description: string; expected: string }[];
}
export interface Lesson {
@@ -614,19 +620,27 @@ const apiFetch = (url: string, options: RequestInit = {}, isLms: boolean = false
export const cmsApi = {
// Organization
getOrganization: (): Promise<Organization> => apiFetch('/organization'),
getOrganizations: (): Promise<Organization[]> => apiFetch('/organizations'),
createOrganization: (name: string, domain?: string): Promise<Organization> => apiFetch('/organizations', { method: 'POST', body: JSON.stringify({ name, domain }) }),
updateOrganization: (id: string, payload: { name?: string, domain?: string }): Promise<Organization> => apiFetch(`/organizations/${id}`, { method: 'PUT', body: JSON.stringify(payload) }),
provisionOrganization: (data: ProvisionPayload): Promise<Organization> => apiFetch('/admin/provision', { method: 'POST', body: JSON.stringify(data) }),
updateOrganizationBranding: (payload: BrandingPayload): Promise<Organization> => apiFetch('/organization/branding', { method: 'PUT', body: JSON.stringify(payload) }),
uploadOrganizationLogo: (file: File): Promise<{ logo_url: string }> => {
const formData = new FormData();
formData.append('file', file);
return apiFetch('/organization/logo', { method: 'POST', body: formData });
},
uploadOrganizationFavicon: (file: File): Promise<{ favicon_url: string }> => {
const formData = new FormData();
formData.append('file', file);
return apiFetch('/organization/favicon', { method: 'POST', body: formData });
},
getSSOConfig: (): Promise<OrganizationSSOConfig> => apiFetch('/organization/sso'),
updateSSOConfig: (payload: Partial<OrganizationSSOConfig>): Promise<void> => apiFetch('/organization/sso', { method: 'PUT', body: JSON.stringify(payload) }),
// Auth
register: (payload: AuthPayload): Promise<AuthResponse> => apiFetch('/auth/register', { method: 'POST', body: JSON.stringify(payload) }),
login: (payload: AuthPayload): Promise<AuthResponse> => apiFetch('/auth/login', { method: 'POST', body: JSON.stringify(payload) }),
getMe: (): Promise<User> => apiFetch('/auth/me'),
// Organizations Search
searchOrganizations: (query: string): Promise<{ id: string, name: string, domain?: string }[]> => apiFetch(`/organizations/search?q=${encodeURIComponent(query)}`),
getBranding: (id: string): Promise<BrandingResponse> => apiFetch(`/organizations/${id}/branding`),
// Branding (Public)
getBranding: (): Promise<BrandingResponse> => apiFetch('/branding'),
// Courses
getCourses: (): Promise<Course[]> => apiFetch('/courses'),
@@ -675,6 +689,15 @@ export const cmsApi = {
body: JSON.stringify(payload)
});
},
async generateCodeLab(lessonId: string, payload: { language?: string; prompt_hint?: string }): Promise<{
language: string; title: string; instructions: string; initial_code: string; solution: string;
test_cases: { description: string; expected: string }[];
}> {
return apiFetch(`/lessons/${lessonId}/generate-code-lab`, {
method: 'POST',
body: JSON.stringify(payload)
});
},
async generateHotspots(lessonId: string, payload: { image_url: string, prompt_hint?: string }): Promise<{
label: string;
description: string;
@@ -785,49 +808,6 @@ export const cmsApi = {
xhr.send(formData);
});
},
// Organizations Branding
getOrganizationBranding: (id: string): Promise<Organization> => apiFetch(`/organizations/${id}/branding`),
updateOrganizationBranding: (id: string, payload: BrandingPayload): Promise<void> => apiFetch(`/organizations/${id}/branding`, { method: 'PUT', body: JSON.stringify(payload) }),
uploadOrganizationLogo: (id: string, file: File): Promise<UploadResponse> => {
const formData = new FormData();
formData.append('file', file);
const token = getToken();
const selectedOrgId = getSelectedOrgId();
const headers: Record<string, string> = {
...(token ? { 'Authorization': `Bearer ${token}` } : {}),
...(selectedOrgId ? { 'X-Organization-Id': selectedOrgId } : {})
};
return fetch(`${API_BASE_URL}/organizations/${id}/logo`, {
method: 'POST',
headers,
body: formData,
}).then(res => {
if (!res.ok) return res.json().then(err => Promise.reject(new Error(err.message || 'Logo upload failed')));
return res.json();
});
},
uploadOrganizationFavicon: (id: string, file: File): Promise<UploadResponse> => {
const formData = new FormData();
formData.append('file', file);
const token = getToken();
const selectedOrgId = getSelectedOrgId();
const headers: Record<string, string> = {
...(token ? { 'Authorization': `Bearer ${token}` } : {}),
...(selectedOrgId ? { 'X-Organization-Id': selectedOrgId } : {})
};
return fetch(`${API_BASE_URL}/organizations/${id}/favicon`, {
method: 'POST',
headers,
body: formData,
}).then(res => {
if (!res.ok) return res.json().then(err => Promise.reject(new Error(err.message || 'Favicon upload failed')));
return res.json();
});
},
// SSO
getSSOConfig: (): Promise<OrganizationSSOConfig | null> => apiFetch('/organization/sso'),
updateSSOConfig: (payload: Partial<OrganizationSSOConfig>): Promise<OrganizationSSOConfig> => apiFetch('/organization/sso', { method: 'PUT', body: JSON.stringify(payload) }),
initSSOLogin: (orgId: string): void => {
window.location.href = `${API_BASE_URL}/auth/sso/login/${orgId}`;
},
File diff suppressed because one or more lines are too long