Files
edbridge-scholars/src/components/LessonModal.tsx
2026-02-20 19:10:13 +06:00

214 lines
6.4 KiB
TypeScript

import { useEffect, useState } from "react";
import { Dialog, DialogContent, DialogHeader } from "../components/ui/dialog";
import { api } from "../utils/api";
import { useAuthStore } from "../stores/authStore";
import { Loader, X } from "lucide-react";
interface LessonModalProps {
lessonId: string | null;
open: boolean;
onOpenChange: (open: boolean) => void;
}
const STYLES = `
@import url('https://fonts.googleapis.com/css2?family=Nunito:wght@400;700;800;900&family=Nunito+Sans:wght@400;600;700&display=swap');
/* Override Dialog defaults */
.lm-content {
font-family: 'Nunito', sans-serif;
background: #fffbf4;
border: 2.5px solid #f3f4f6;
border-radius: 28px !important;
padding: 0;
overflow: hidden;
max-width: 680px;
width: calc(100vw - 2rem);
box-shadow: 0 20px 60px rgba(0,0,0,0.12);
max-height: 90vh;
display: flex;
flex-direction: column;
}
/* Header bar */
.lm-header {
display: flex; align-items: flex-start; justify-content: space-between;
padding: 1.25rem 1.5rem 0;
flex-shrink: 0;
gap: 1rem;
}
.lm-title-wrap { display:flex;flex-direction:column;gap:0.2rem; flex:1; }
.lm-eyebrow {
font-size: 0.62rem; font-weight: 800; letter-spacing: 0.16em;
text-transform: uppercase; color: #a855f7;
}
.lm-title {
font-size: 1.2rem; font-weight: 900; color: #1e1b4b;
letter-spacing: -0.01em; line-height: 1.25;
}
.lm-close-btn {
width: 34px; height: 34px; flex-shrink: 0;
border-radius: 50%; border: 2.5px solid #f3f4f6;
background: white; cursor: pointer;
display: flex; align-items: center; justify-content: center;
box-shadow: 0 2px 8px rgba(0,0,0,0.06);
transition: all 0.15s ease;
}
.lm-close-btn:hover { border-color: #fecdd3; background: #fff1f2; }
/* Scrollable body */
.lm-body {
overflow-y: auto;
flex: 1;
padding: 1rem 1.5rem 1.5rem;
display: flex; flex-direction: column; gap: 1rem;
-webkit-overflow-scrolling: touch;
}
/* Video player */
.lm-video {
width: 100%; border-radius: 18px;
aspect-ratio: 16/9; background: #1e1b4b;
display: block;
}
/* Topic chip */
.lm-topic-chip {
display: inline-flex; align-items: center; gap: 0.4rem;
background: #f3e8ff; border: 2px solid #e9d5ff;
border-radius: 100px; padding: 0.3rem 0.8rem;
font-size: 0.7rem; font-weight: 800;
letter-spacing: 0.08em; text-transform: uppercase;
color: #9333ea; width: fit-content;
}
/* Description & content cards */
.lm-card {
background: white; border: 2.5px solid #f3f4f6;
border-radius: 18px; padding: 1rem 1.1rem;
box-shadow: 0 3px 10px rgba(0,0,0,0.04);
}
.lm-card-label {
font-size: 0.62rem; font-weight: 800; letter-spacing: 0.14em;
text-transform: uppercase; color: #9ca3af; margin-bottom: 0.4rem;
}
.lm-card-text {
font-family: 'Nunito Sans', sans-serif;
font-size: 0.88rem; font-weight: 600; color: #374151;
line-height: 1.6;
}
/* Loading state */
.lm-loading {
display: flex; flex-direction: column; align-items: center;
justify-content: center; gap: 0.75rem;
padding: 3rem 1.5rem;
flex: 1;
}
.lm-loading-spinner { animation: lmSpin 0.8s linear infinite; }
@keyframes lmSpin { to { transform: rotate(360deg); } }
.lm-loading-text {
font-size: 0.85rem; font-weight: 700; color: #9ca3af;
}
`;
export const LessonModal = ({
lessonId,
open,
onOpenChange,
}: LessonModalProps) => {
const user = useAuthStore((state) => state.user);
const [loading, setLoading] = useState(false);
const [lesson, setLesson] = useState<any>(null);
useEffect(() => {
if (!open || !lessonId || !user) return;
const fetchLesson = async () => {
try {
setLoading(true);
const authStorage = localStorage.getItem("auth-storage");
if (!authStorage) return;
const {
state: { token },
} = JSON.parse(authStorage) as { state?: { token?: string } };
if (!token) return;
const response = await api.fetchLessonById(token, lessonId);
setLesson(response);
} catch (err) {
console.error("Failed to fetch lesson", err);
} finally {
setLoading(false);
}
};
fetchLesson();
}, [open, lessonId, user]);
return (
<Dialog open={open} onOpenChange={onOpenChange}>
<style>{STYLES}</style>
<DialogContent className="lm-content" showCloseButton={false}>
<DialogHeader style={{ display: "none" }} />
{/* Header */}
<div className="lm-header">
<div className="lm-title-wrap">
<span className="lm-eyebrow">📖 Lesson</span>
<h2 className="lm-title">
{loading ? "Loading..." : (lesson?.title ?? "Lesson details")}
</h2>
</div>
<button className="lm-close-btn" onClick={() => onOpenChange(false)}>
<X size={16} color="#6b7280" />
</button>
</div>
{/* Body */}
{loading ? (
<div className="lm-loading">
<Loader size={28} color="#a855f7" className="lm-loading-spinner" />
<p className="lm-loading-text">Loading lesson...</p>
</div>
) : (
lesson && (
<div className="lm-body">
{lesson.video_url && (
<video src={lesson.video_url} controls className="lm-video" />
)}
{lesson.topic?.name && (
<div>
<span className="lm-topic-chip">
<span
style={{
width: 6,
height: 6,
borderRadius: "50%",
background: "#a855f7",
flexShrink: 0,
}}
/>
{lesson.topic.name}
</span>
</div>
)}
{lesson.description && (
<div className="lm-card">
<p className="lm-card-label">About this lesson</p>
<p className="lm-card-text">{lesson.description}</p>
</div>
)}
{lesson.content && (
<div className="lm-card">
<p className="lm-card-label">Content</p>
<p className="lm-card-text">{lesson.content}</p>
</div>
)}
</div>
)
)}
</DialogContent>
</Dialog>
);
};