[OQI-002] feat: Add education UX components

- CertificatePreview: Display earned certificates with download/share
- CourseReviews: Reviews section with rating summary and filtering
- LessonNotes: Note-taking with timestamps and auto-save
- RecommendedCourses: Course recommendations with multiple layouts

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
Adrian Flores Cortes 2026-01-25 10:27:25 -06:00
parent 442051f93b
commit fc0ab528c3
5 changed files with 1195 additions and 0 deletions

View File

@ -0,0 +1,193 @@
/**
* CertificatePreview Component
* Displays course completion certificate with download option
*/
import React from 'react';
import {
Award,
Download,
Share2,
ExternalLink,
Calendar,
CheckCircle,
} from 'lucide-react';
interface CertificatePreviewProps {
courseTitle: string;
courseThumbnail?: string;
studentName: string;
instructorName: string;
completedAt: string;
certificateUrl?: string;
certificateId?: string;
xpEarned?: number;
onDownload?: () => void;
onShare?: () => void;
compact?: boolean;
}
const CertificatePreview: React.FC<CertificatePreviewProps> = ({
courseTitle,
courseThumbnail,
studentName,
instructorName,
completedAt,
certificateUrl,
certificateId,
xpEarned,
onDownload,
onShare,
compact = false,
}) => {
const formattedDate = new Date(completedAt).toLocaleDateString('es-ES', {
year: 'numeric',
month: 'long',
day: 'numeric',
});
if (compact) {
return (
<div className="flex items-center gap-4 p-4 bg-gradient-to-r from-yellow-900/20 to-amber-900/20 border border-yellow-700/30 rounded-xl">
<div className="p-3 bg-yellow-500/20 rounded-full">
<Award className="w-6 h-6 text-yellow-400" />
</div>
<div className="flex-1 min-w-0">
<p className="font-medium text-white truncate">{courseTitle}</p>
<p className="text-sm text-gray-400">Completado el {formattedDate}</p>
</div>
<div className="flex items-center gap-2">
{certificateUrl && (
<a
href={certificateUrl}
target="_blank"
rel="noopener noreferrer"
className="p-2 text-gray-400 hover:text-white hover:bg-gray-700 rounded-lg transition-colors"
title="Ver certificado"
>
<ExternalLink className="w-4 h-4" />
</a>
)}
{onDownload && (
<button
onClick={onDownload}
className="p-2 text-gray-400 hover:text-white hover:bg-gray-700 rounded-lg transition-colors"
title="Descargar"
>
<Download className="w-4 h-4" />
</button>
)}
</div>
</div>
);
}
return (
<div className="bg-gradient-to-br from-gray-800 via-gray-800 to-gray-900 rounded-xl border border-gray-700 overflow-hidden">
{/* Certificate Header */}
<div className="relative bg-gradient-to-r from-yellow-600/20 via-amber-500/20 to-yellow-600/20 p-6 border-b border-yellow-700/30">
{/* Decorative elements */}
<div className="absolute top-0 left-0 w-full h-full opacity-10">
<div className="absolute top-2 left-4 w-8 h-8 border-2 border-yellow-400 rounded-full" />
<div className="absolute bottom-2 right-4 w-6 h-6 border-2 border-yellow-400 rounded-full" />
<div className="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 w-32 h-32 border border-yellow-400/30 rounded-full" />
</div>
<div className="relative flex items-center justify-center gap-4">
<Award className="w-12 h-12 text-yellow-400" />
<div className="text-center">
<h3 className="text-xl font-bold text-yellow-400">Certificado de Finalizacion</h3>
<p className="text-yellow-400/70 text-sm">Trading Academy</p>
</div>
<Award className="w-12 h-12 text-yellow-400" />
</div>
</div>
{/* Certificate Body */}
<div className="p-6 text-center space-y-4">
{/* Student Name */}
<div>
<p className="text-sm text-gray-400 mb-1">Este certificado se otorga a</p>
<h2 className="text-2xl font-bold text-white">{studentName}</h2>
</div>
{/* Course Info */}
<div>
<p className="text-sm text-gray-400 mb-2">Por completar exitosamente el curso</p>
<div className="flex items-center justify-center gap-4">
{courseThumbnail && (
<img
src={courseThumbnail}
alt={courseTitle}
className="w-16 h-10 rounded object-cover"
/>
)}
<h3 className="text-xl font-semibold text-blue-400">{courseTitle}</h3>
</div>
</div>
{/* Instructor */}
<div className="pt-2">
<p className="text-sm text-gray-500">Impartido por</p>
<p className="text-white font-medium">{instructorName}</p>
</div>
{/* Completion Info */}
<div className="flex items-center justify-center gap-6 pt-4">
<div className="flex items-center gap-2 text-sm">
<Calendar className="w-4 h-4 text-gray-400" />
<span className="text-gray-400">{formattedDate}</span>
</div>
{xpEarned && (
<div className="flex items-center gap-2 text-sm">
<CheckCircle className="w-4 h-4 text-green-400" />
<span className="text-green-400">+{xpEarned} XP</span>
</div>
)}
</div>
{/* Certificate ID */}
{certificateId && (
<p className="text-xs text-gray-600 pt-2">
ID: {certificateId}
</p>
)}
</div>
{/* Actions */}
<div className="flex items-center justify-center gap-3 p-4 bg-gray-900/50 border-t border-gray-700">
{certificateUrl && (
<a
href={certificateUrl}
target="_blank"
rel="noopener noreferrer"
className="flex items-center gap-2 px-4 py-2 bg-gray-700 hover:bg-gray-600 text-white rounded-lg transition-colors text-sm"
>
<ExternalLink className="w-4 h-4" />
Ver Completo
</a>
)}
{onDownload && (
<button
onClick={onDownload}
className="flex items-center gap-2 px-4 py-2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg transition-colors text-sm"
>
<Download className="w-4 h-4" />
Descargar PDF
</button>
)}
{onShare && (
<button
onClick={onShare}
className="flex items-center gap-2 px-4 py-2 bg-gray-700 hover:bg-gray-600 text-white rounded-lg transition-colors text-sm"
>
<Share2 className="w-4 h-4" />
Compartir
</button>
)}
</div>
</div>
);
};
export default CertificatePreview;

View File

@ -0,0 +1,322 @@
/**
* CourseReviews Component
* Displays course reviews with rating summary and individual reviews
*/
import React, { useState } from 'react';
import {
Star,
ThumbsUp,
MessageSquare,
ChevronDown,
User,
Filter,
} from 'lucide-react';
interface Review {
id: string;
userId: string;
userName: string;
userAvatar?: string;
rating: number;
title?: string;
comment: string;
helpful: number;
createdAt: string;
verified?: boolean;
}
interface RatingSummary {
average: number;
total: number;
distribution: {
5: number;
4: number;
3: number;
2: number;
1: number;
};
}
interface CourseReviewsProps {
courseId: string;
reviews: Review[];
ratingSummary: RatingSummary;
onLoadMore?: () => void;
onMarkHelpful?: (reviewId: string) => void;
onReport?: (reviewId: string) => void;
hasMore?: boolean;
loading?: boolean;
}
const CourseReviews: React.FC<CourseReviewsProps> = ({
courseId,
reviews,
ratingSummary,
onLoadMore,
onMarkHelpful,
onReport,
hasMore = false,
loading = false,
}) => {
const [filterRating, setFilterRating] = useState<number | null>(null);
const [sortBy, setSortBy] = useState<'recent' | 'helpful' | 'rating'>('recent');
const filteredReviews = reviews.filter((review) => {
if (filterRating === null) return true;
return review.rating === filterRating;
});
const sortedReviews = [...filteredReviews].sort((a, b) => {
if (sortBy === 'helpful') return b.helpful - a.helpful;
if (sortBy === 'rating') return b.rating - a.rating;
return new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime();
});
const renderStars = (rating: number, size: 'sm' | 'md' = 'sm') => {
const sizeClass = size === 'sm' ? 'w-4 h-4' : 'w-5 h-5';
return (
<div className="flex items-center gap-0.5">
{[1, 2, 3, 4, 5].map((star) => (
<Star
key={star}
className={`${sizeClass} ${
star <= rating
? 'text-yellow-400 fill-yellow-400'
: 'text-gray-600'
}`}
/>
))}
</div>
);
};
const formatDate = (dateString: string) => {
const date = new Date(dateString);
return date.toLocaleDateString('es-ES', {
year: 'numeric',
month: 'short',
day: 'numeric',
});
};
return (
<div className="space-y-6">
{/* Rating Summary */}
<div className="bg-gray-800 rounded-xl border border-gray-700 p-6">
<h3 className="text-xl font-bold text-white mb-6">Resenas del Curso</h3>
<div className="grid grid-cols-1 md:grid-cols-2 gap-6">
{/* Average Rating */}
<div className="flex items-center gap-6">
<div className="text-center">
<div className="text-5xl font-bold text-white">
{ratingSummary.average.toFixed(1)}
</div>
<div className="mt-2">{renderStars(Math.round(ratingSummary.average), 'md')}</div>
<p className="text-sm text-gray-400 mt-1">
{ratingSummary.total.toLocaleString()} resenas
</p>
</div>
</div>
{/* Rating Distribution */}
<div className="space-y-2">
{[5, 4, 3, 2, 1].map((rating) => {
const count = ratingSummary.distribution[rating as keyof typeof ratingSummary.distribution];
const percentage = ratingSummary.total > 0 ? (count / ratingSummary.total) * 100 : 0;
return (
<button
key={rating}
onClick={() => setFilterRating(filterRating === rating ? null : rating)}
className={`w-full flex items-center gap-3 p-1 rounded hover:bg-gray-700/50 transition-colors ${
filterRating === rating ? 'bg-gray-700/50' : ''
}`}
>
<div className="flex items-center gap-1 w-20">
<span className="text-sm text-gray-400">{rating}</span>
<Star className="w-4 h-4 text-yellow-400 fill-yellow-400" />
</div>
<div className="flex-1 h-2 bg-gray-700 rounded-full overflow-hidden">
<div
className="h-full bg-yellow-400 rounded-full transition-all"
style={{ width: `${percentage}%` }}
/>
</div>
<span className="text-sm text-gray-500 w-12 text-right">
{count.toLocaleString()}
</span>
</button>
);
})}
</div>
</div>
</div>
{/* Filters */}
<div className="flex items-center justify-between">
<div className="flex items-center gap-3">
<Filter className="w-4 h-4 text-gray-400" />
<span className="text-sm text-gray-400">
{filteredReviews.length} resenas
{filterRating && ` con ${filterRating} estrellas`}
</span>
{filterRating && (
<button
onClick={() => setFilterRating(null)}
className="text-sm text-blue-400 hover:text-blue-300"
>
Limpiar filtro
</button>
)}
</div>
<select
value={sortBy}
onChange={(e) => setSortBy(e.target.value as typeof sortBy)}
className="px-3 py-2 bg-gray-800 border border-gray-700 rounded-lg text-sm text-white focus:outline-none focus:border-blue-500"
>
<option value="recent">Mas recientes</option>
<option value="helpful">Mas utiles</option>
<option value="rating">Mayor calificacion</option>
</select>
</div>
{/* Reviews List */}
<div className="space-y-4">
{sortedReviews.length === 0 ? (
<div className="text-center py-12 bg-gray-800 rounded-xl border border-gray-700">
<MessageSquare className="w-12 h-12 text-gray-500 mx-auto mb-4" />
<h4 className="font-medium text-white mb-2">Sin resenas</h4>
<p className="text-gray-400 text-sm">
{filterRating
? `No hay resenas con ${filterRating} estrellas`
: 'Se el primero en dejar una resena'}
</p>
</div>
) : (
sortedReviews.map((review) => (
<ReviewCard
key={review.id}
review={review}
onMarkHelpful={() => onMarkHelpful?.(review.id)}
onReport={() => onReport?.(review.id)}
/>
))
)}
</div>
{/* Load More */}
{hasMore && (
<div className="text-center">
<button
onClick={onLoadMore}
disabled={loading}
className="inline-flex items-center gap-2 px-6 py-3 bg-gray-700 hover:bg-gray-600 text-white rounded-lg transition-colors disabled:opacity-50"
>
{loading ? (
<div className="w-5 h-5 border-2 border-white/30 border-t-white rounded-full animate-spin" />
) : (
<ChevronDown className="w-5 h-5" />
)}
Cargar mas resenas
</button>
</div>
)}
</div>
);
};
// Individual Review Card
interface ReviewCardProps {
review: Review;
onMarkHelpful?: () => void;
onReport?: () => void;
}
const ReviewCard: React.FC<ReviewCardProps> = ({ review, onMarkHelpful, onReport }) => {
const renderStars = (rating: number) => (
<div className="flex items-center gap-0.5">
{[1, 2, 3, 4, 5].map((star) => (
<Star
key={star}
className={`w-4 h-4 ${
star <= rating ? 'text-yellow-400 fill-yellow-400' : 'text-gray-600'
}`}
/>
))}
</div>
);
const formatDate = (dateString: string) => {
const date = new Date(dateString);
return date.toLocaleDateString('es-ES', {
year: 'numeric',
month: 'short',
day: 'numeric',
});
};
return (
<div className="bg-gray-800 rounded-xl border border-gray-700 p-5">
{/* Header */}
<div className="flex items-start justify-between mb-3">
<div className="flex items-center gap-3">
{review.userAvatar ? (
<img
src={review.userAvatar}
alt={review.userName}
className="w-10 h-10 rounded-full object-cover"
/>
) : (
<div className="w-10 h-10 rounded-full bg-gray-700 flex items-center justify-center">
<User className="w-5 h-5 text-gray-400" />
</div>
)}
<div>
<div className="flex items-center gap-2">
<span className="font-medium text-white">{review.userName}</span>
{review.verified && (
<span className="px-2 py-0.5 bg-green-500/20 text-green-400 text-xs rounded">
Verificado
</span>
)}
</div>
<div className="flex items-center gap-2 mt-0.5">
{renderStars(review.rating)}
<span className="text-xs text-gray-500">{formatDate(review.createdAt)}</span>
</div>
</div>
</div>
</div>
{/* Title */}
{review.title && (
<h4 className="font-medium text-white mb-2">{review.title}</h4>
)}
{/* Comment */}
<p className="text-gray-300 text-sm leading-relaxed">{review.comment}</p>
{/* Actions */}
<div className="flex items-center gap-4 mt-4 pt-4 border-t border-gray-700">
<button
onClick={onMarkHelpful}
className="flex items-center gap-2 text-sm text-gray-400 hover:text-white transition-colors"
>
<ThumbsUp className="w-4 h-4" />
<span>Util ({review.helpful})</span>
</button>
<button
onClick={onReport}
className="text-sm text-gray-500 hover:text-gray-400 transition-colors"
>
Reportar
</button>
</div>
</div>
);
};
export default CourseReviews;

View File

@ -0,0 +1,341 @@
/**
* LessonNotes Component
* Note-taking interface for lessons with auto-save
*/
import React, { useState, useEffect, useCallback, useRef } from 'react';
import {
StickyNote,
Save,
Clock,
Trash2,
Download,
ChevronDown,
ChevronUp,
BookmarkPlus,
CheckCircle,
Loader2,
} from 'lucide-react';
interface Note {
id: string;
lessonId: string;
content: string;
timestamp?: number; // Video timestamp in seconds
createdAt: string;
updatedAt: string;
}
interface LessonNotesProps {
lessonId: string;
courseId: string;
notes: Note[];
currentVideoTime?: number;
onSaveNote: (content: string, timestamp?: number) => Promise<void>;
onUpdateNote: (noteId: string, content: string) => Promise<void>;
onDeleteNote: (noteId: string) => Promise<void>;
onSeekToTimestamp?: (timestamp: number) => void;
onExportNotes?: () => void;
collapsed?: boolean;
onToggleCollapse?: () => void;
}
const LessonNotes: React.FC<LessonNotesProps> = ({
lessonId,
courseId,
notes,
currentVideoTime,
onSaveNote,
onUpdateNote,
onDeleteNote,
onSeekToTimestamp,
onExportNotes,
collapsed = false,
onToggleCollapse,
}) => {
const [newNote, setNewNote] = useState('');
const [editingNoteId, setEditingNoteId] = useState<string | null>(null);
const [editContent, setEditContent] = useState('');
const [saving, setSaving] = useState(false);
const [lastSaved, setLastSaved] = useState<Date | null>(null);
const [includeTimestamp, setIncludeTimestamp] = useState(true);
const textareaRef = useRef<HTMLTextAreaElement>(null);
// Auto-resize textarea
const adjustTextareaHeight = useCallback((textarea: HTMLTextAreaElement) => {
textarea.style.height = 'auto';
textarea.style.height = `${Math.min(textarea.scrollHeight, 200)}px`;
}, []);
useEffect(() => {
if (textareaRef.current) {
adjustTextareaHeight(textareaRef.current);
}
}, [newNote, adjustTextareaHeight]);
const formatTimestamp = (seconds: number): string => {
const mins = Math.floor(seconds / 60);
const secs = Math.floor(seconds % 60);
return `${mins}:${secs.toString().padStart(2, '0')}`;
};
const handleSaveNote = async () => {
if (!newNote.trim()) return;
setSaving(true);
try {
const timestamp = includeTimestamp && currentVideoTime ? Math.floor(currentVideoTime) : undefined;
await onSaveNote(newNote.trim(), timestamp);
setNewNote('');
setLastSaved(new Date());
} catch (error) {
console.error('Error saving note:', error);
} finally {
setSaving(false);
}
};
const handleUpdateNote = async (noteId: string) => {
if (!editContent.trim()) return;
setSaving(true);
try {
await onUpdateNote(noteId, editContent.trim());
setEditingNoteId(null);
setEditContent('');
setLastSaved(new Date());
} catch (error) {
console.error('Error updating note:', error);
} finally {
setSaving(false);
}
};
const handleDeleteNote = async (noteId: string) => {
if (!confirm('Estas seguro de eliminar esta nota?')) return;
try {
await onDeleteNote(noteId);
} catch (error) {
console.error('Error deleting note:', error);
}
};
const startEditing = (note: Note) => {
setEditingNoteId(note.id);
setEditContent(note.content);
};
const cancelEditing = () => {
setEditingNoteId(null);
setEditContent('');
};
const handleKeyDown = (e: React.KeyboardEvent<HTMLTextAreaElement>) => {
if (e.key === 'Enter' && (e.metaKey || e.ctrlKey)) {
e.preventDefault();
handleSaveNote();
}
};
// Sort notes by timestamp or creation date
const sortedNotes = [...notes].sort((a, b) => {
if (a.timestamp !== undefined && b.timestamp !== undefined) {
return a.timestamp - b.timestamp;
}
return new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime();
});
return (
<div className="bg-gray-800 rounded-xl border border-gray-700 overflow-hidden">
{/* Header */}
<button
onClick={onToggleCollapse}
className="w-full flex items-center justify-between p-4 hover:bg-gray-700/50 transition-colors"
>
<div className="flex items-center gap-3">
<StickyNote className="w-5 h-5 text-yellow-400" />
<span className="font-medium text-white">Mis Notas</span>
<span className="px-2 py-0.5 bg-gray-700 text-gray-400 text-xs rounded-full">
{notes.length}
</span>
</div>
<div className="flex items-center gap-2">
{lastSaved && (
<span className="text-xs text-gray-500 flex items-center gap-1">
<CheckCircle className="w-3 h-3 text-green-400" />
Guardado
</span>
)}
{collapsed ? (
<ChevronDown className="w-5 h-5 text-gray-400" />
) : (
<ChevronUp className="w-5 h-5 text-gray-400" />
)}
</div>
</button>
{!collapsed && (
<div className="p-4 pt-0 space-y-4">
{/* New Note Input */}
<div className="space-y-3">
<div className="relative">
<textarea
ref={textareaRef}
value={newNote}
onChange={(e) => {
setNewNote(e.target.value);
adjustTextareaHeight(e.target);
}}
onKeyDown={handleKeyDown}
placeholder="Escribe una nota... (Ctrl+Enter para guardar)"
className="w-full px-4 py-3 bg-gray-900 border border-gray-700 rounded-lg text-white placeholder-gray-500 resize-none focus:outline-none focus:border-blue-500 min-h-[80px]"
/>
</div>
<div className="flex items-center justify-between">
<div className="flex items-center gap-3">
{currentVideoTime !== undefined && (
<label className="flex items-center gap-2 text-sm text-gray-400 cursor-pointer">
<input
type="checkbox"
checked={includeTimestamp}
onChange={(e) => setIncludeTimestamp(e.target.checked)}
className="rounded border-gray-600 bg-gray-700 text-blue-500 focus:ring-blue-500"
/>
<Clock className="w-4 h-4" />
<span>Incluir timestamp ({formatTimestamp(currentVideoTime)})</span>
</label>
)}
</div>
<div className="flex items-center gap-2">
{onExportNotes && notes.length > 0 && (
<button
onClick={onExportNotes}
className="p-2 text-gray-400 hover:text-white hover:bg-gray-700 rounded-lg transition-colors"
title="Exportar notas"
>
<Download className="w-4 h-4" />
</button>
)}
<button
onClick={handleSaveNote}
disabled={!newNote.trim() || saving}
className="flex items-center gap-2 px-4 py-2 bg-blue-600 hover:bg-blue-500 text-white rounded-lg transition-colors disabled:opacity-50 disabled:cursor-not-allowed text-sm"
>
{saving ? (
<Loader2 className="w-4 h-4 animate-spin" />
) : (
<BookmarkPlus className="w-4 h-4" />
)}
Guardar Nota
</button>
</div>
</div>
</div>
{/* Notes List */}
{sortedNotes.length > 0 && (
<div className="space-y-3 pt-4 border-t border-gray-700">
{sortedNotes.map((note) => (
<div
key={note.id}
className="p-3 bg-gray-900/50 rounded-lg group"
>
{editingNoteId === note.id ? (
<div className="space-y-2">
<textarea
value={editContent}
onChange={(e) => setEditContent(e.target.value)}
className="w-full px-3 py-2 bg-gray-800 border border-gray-600 rounded-lg text-white resize-none focus:outline-none focus:border-blue-500 min-h-[60px]"
autoFocus
/>
<div className="flex items-center justify-end gap-2">
<button
onClick={cancelEditing}
className="px-3 py-1.5 text-gray-400 hover:text-white text-sm transition-colors"
>
Cancelar
</button>
<button
onClick={() => handleUpdateNote(note.id)}
disabled={!editContent.trim() || saving}
className="flex items-center gap-1 px-3 py-1.5 bg-blue-600 hover:bg-blue-500 text-white rounded text-sm transition-colors disabled:opacity-50"
>
{saving && <Loader2 className="w-3 h-3 animate-spin" />}
Guardar
</button>
</div>
</div>
) : (
<>
{/* Timestamp */}
{note.timestamp !== undefined && onSeekToTimestamp && (
<button
onClick={() => onSeekToTimestamp(note.timestamp!)}
className="inline-flex items-center gap-1 px-2 py-0.5 bg-blue-500/20 text-blue-400 text-xs rounded mb-2 hover:bg-blue-500/30 transition-colors"
>
<Clock className="w-3 h-3" />
{formatTimestamp(note.timestamp)}
</button>
)}
{/* Note Content */}
<p className="text-gray-300 text-sm whitespace-pre-wrap">
{note.content}
</p>
{/* Actions */}
<div className="flex items-center justify-between mt-2 pt-2 border-t border-gray-700/50">
<span className="text-xs text-gray-500">
{new Date(note.updatedAt).toLocaleDateString('es-ES', {
month: 'short',
day: 'numeric',
hour: '2-digit',
minute: '2-digit',
})}
</span>
<div className="flex items-center gap-1 opacity-0 group-hover:opacity-100 transition-opacity">
<button
onClick={() => startEditing(note)}
className="p-1.5 text-gray-400 hover:text-white hover:bg-gray-700 rounded transition-colors"
title="Editar"
>
<Save className="w-3.5 h-3.5" />
</button>
<button
onClick={() => handleDeleteNote(note.id)}
className="p-1.5 text-gray-400 hover:text-red-400 hover:bg-gray-700 rounded transition-colors"
title="Eliminar"
>
<Trash2 className="w-3.5 h-3.5" />
</button>
</div>
</div>
</>
)}
</div>
))}
</div>
)}
{/* Empty State */}
{notes.length === 0 && (
<div className="text-center py-8">
<StickyNote className="w-10 h-10 text-gray-600 mx-auto mb-3" />
<p className="text-gray-500 text-sm">
Aun no tienes notas para esta leccion
</p>
<p className="text-gray-600 text-xs mt-1">
Escribe arriba para agregar tu primera nota
</p>
</div>
)}
</div>
)}
</div>
);
};
export default LessonNotes;

View File

@ -0,0 +1,330 @@
/**
* RecommendedCourses Component
* Displays recommended courses based on user progress and interests
*/
import React from 'react';
import { Link } from 'react-router-dom';
import {
BookOpen,
Clock,
Star,
Users,
ChevronRight,
Sparkles,
TrendingUp,
Zap,
} from 'lucide-react';
import type { CourseListItem } from '../../../types/education.types';
interface RecommendedCoursesProps {
courses: CourseListItem[];
title?: string;
subtitle?: string;
reason?: 'similar' | 'popular' | 'new' | 'continue' | 'trending';
maxItems?: number;
layout?: 'horizontal' | 'vertical' | 'compact';
showViewAll?: boolean;
viewAllLink?: string;
loading?: boolean;
}
const difficultyColors = {
beginner: 'bg-green-500/20 text-green-400',
intermediate: 'bg-yellow-500/20 text-yellow-400',
advanced: 'bg-red-500/20 text-red-400',
};
const difficultyLabels = {
beginner: 'Principiante',
intermediate: 'Intermedio',
advanced: 'Avanzado',
};
const reasonIcons = {
similar: <BookOpen className="w-4 h-4" />,
popular: <TrendingUp className="w-4 h-4" />,
new: <Sparkles className="w-4 h-4" />,
continue: <ChevronRight className="w-4 h-4" />,
trending: <Zap className="w-4 h-4" />,
};
const reasonLabels = {
similar: 'Similar a tus cursos',
popular: 'Mas populares',
new: 'Nuevos cursos',
continue: 'Continua aprendiendo',
trending: 'En tendencia',
};
const RecommendedCourses: React.FC<RecommendedCoursesProps> = ({
courses,
title = 'Cursos Recomendados',
subtitle,
reason = 'popular',
maxItems = 4,
layout = 'horizontal',
showViewAll = true,
viewAllLink = '/education/courses',
loading = false,
}) => {
const displayCourses = courses.slice(0, maxItems);
const formatDuration = (minutes: number) => {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) return `${mins}m`;
if (mins === 0) return `${hours}h`;
return `${hours}h ${mins}m`;
};
if (loading) {
return (
<div className="space-y-4">
<div className="flex items-center justify-between">
<div className="h-7 w-48 bg-gray-700 rounded animate-pulse" />
<div className="h-5 w-20 bg-gray-700 rounded animate-pulse" />
</div>
<div className={`grid gap-4 ${layout === 'horizontal' ? 'grid-cols-1 md:grid-cols-2 lg:grid-cols-4' : 'grid-cols-1'}`}>
{[...Array(maxItems)].map((_, i) => (
<div key={i} className="bg-gray-800 rounded-xl border border-gray-700 p-4 animate-pulse">
<div className="aspect-video bg-gray-700 rounded-lg mb-4" />
<div className="h-5 bg-gray-700 rounded w-3/4 mb-2" />
<div className="h-4 bg-gray-700 rounded w-1/2" />
</div>
))}
</div>
</div>
);
}
if (courses.length === 0) {
return null;
}
return (
<div className="space-y-4">
{/* Header */}
<div className="flex items-center justify-between">
<div>
<div className="flex items-center gap-2">
<span className="text-blue-400">{reasonIcons[reason]}</span>
<h3 className="text-lg font-bold text-white">{title}</h3>
</div>
{subtitle && <p className="text-sm text-gray-400 mt-0.5">{subtitle}</p>}
</div>
{showViewAll && (
<Link
to={viewAllLink}
className="flex items-center gap-1 text-sm text-blue-400 hover:text-blue-300 transition-colors"
>
Ver todos
<ChevronRight className="w-4 h-4" />
</Link>
)}
</div>
{/* Reason Badge */}
<div className="inline-flex items-center gap-2 px-3 py-1 bg-blue-500/10 border border-blue-500/20 rounded-full">
{reasonIcons[reason]}
<span className="text-xs text-blue-400">{reasonLabels[reason]}</span>
</div>
{/* Courses Grid */}
{layout === 'compact' ? (
<div className="space-y-2">
{displayCourses.map((course) => (
<CompactCourseCard key={course.id} course={course} />
))}
</div>
) : layout === 'vertical' ? (
<div className="space-y-4">
{displayCourses.map((course) => (
<VerticalCourseCard key={course.id} course={course} />
))}
</div>
) : (
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
{displayCourses.map((course) => (
<HorizontalCourseCard key={course.id} course={course} />
))}
</div>
)}
</div>
);
};
// Horizontal Card (Default)
const HorizontalCourseCard: React.FC<{ course: CourseListItem }> = ({ course }) => {
const formatDuration = (minutes: number) => {
const hours = Math.floor(minutes / 60);
return hours > 0 ? `${hours}h` : `${minutes}m`;
};
return (
<Link
to={`/education/courses/${course.slug}`}
className="group bg-gray-800 rounded-xl border border-gray-700 overflow-hidden hover:border-gray-600 transition-all hover:shadow-lg"
>
{/* Thumbnail */}
<div className="relative aspect-video bg-gray-900">
{course.thumbnailUrl ? (
<img
src={course.thumbnailUrl}
alt={course.title}
className="w-full h-full object-cover group-hover:scale-105 transition-transform duration-300"
/>
) : (
<div className="w-full h-full flex items-center justify-center">
<BookOpen className="w-10 h-10 text-gray-600" />
</div>
)}
{course.isFree && (
<span className="absolute top-2 right-2 px-2 py-0.5 bg-green-500 text-white text-xs font-medium rounded">
GRATIS
</span>
)}
<div className="absolute bottom-2 right-2 px-2 py-0.5 bg-black/70 text-white text-xs rounded">
{formatDuration(course.totalDuration)}
</div>
</div>
{/* Content */}
<div className="p-4">
<span className={`inline-block px-2 py-0.5 text-xs rounded mb-2 ${difficultyColors[course.difficultyLevel]}`}>
{difficultyLabels[course.difficultyLevel]}
</span>
<h4 className="font-medium text-white line-clamp-2 mb-2 group-hover:text-blue-400 transition-colors">
{course.title}
</h4>
<div className="flex items-center justify-between text-sm">
<div className="flex items-center gap-1 text-yellow-400">
<Star className="w-4 h-4 fill-yellow-400" />
<span>{course.avgRating.toFixed(1)}</span>
</div>
<span className="text-gray-500">{course.totalLessons} lecciones</span>
</div>
</div>
</Link>
);
};
// Vertical Card (List view)
const VerticalCourseCard: React.FC<{ course: CourseListItem }> = ({ course }) => {
const formatDuration = (minutes: number) => {
const hours = Math.floor(minutes / 60);
const mins = minutes % 60;
if (hours === 0) return `${mins} min`;
return `${hours}h ${mins}m`;
};
return (
<Link
to={`/education/courses/${course.slug}`}
className="group flex gap-4 p-4 bg-gray-800 rounded-xl border border-gray-700 hover:border-gray-600 transition-all"
>
{/* Thumbnail */}
<div className="relative w-40 h-24 flex-shrink-0 rounded-lg overflow-hidden bg-gray-900">
{course.thumbnailUrl ? (
<img
src={course.thumbnailUrl}
alt={course.title}
className="w-full h-full object-cover group-hover:scale-105 transition-transform"
/>
) : (
<div className="w-full h-full flex items-center justify-center">
<BookOpen className="w-8 h-8 text-gray-600" />
</div>
)}
{course.isFree && (
<span className="absolute top-1 left-1 px-1.5 py-0.5 bg-green-500 text-white text-xs font-medium rounded">
GRATIS
</span>
)}
</div>
{/* Content */}
<div className="flex-1 min-w-0">
<div className="flex items-start justify-between gap-2">
<h4 className="font-medium text-white line-clamp-2 group-hover:text-blue-400 transition-colors">
{course.title}
</h4>
<span className={`flex-shrink-0 px-2 py-0.5 text-xs rounded ${difficultyColors[course.difficultyLevel]}`}>
{difficultyLabels[course.difficultyLevel]}
</span>
</div>
{course.shortDescription && (
<p className="text-sm text-gray-400 line-clamp-1 mt-1">{course.shortDescription}</p>
)}
<div className="flex items-center gap-4 mt-2 text-sm text-gray-500">
<div className="flex items-center gap-1">
<Clock className="w-4 h-4" />
{formatDuration(course.totalDuration)}
</div>
<div className="flex items-center gap-1">
<BookOpen className="w-4 h-4" />
{course.totalLessons} lecciones
</div>
<div className="flex items-center gap-1 text-yellow-400">
<Star className="w-4 h-4 fill-yellow-400" />
{course.avgRating.toFixed(1)}
</div>
<div className="flex items-center gap-1">
<Users className="w-4 h-4" />
{course.totalEnrollments.toLocaleString()}
</div>
</div>
</div>
</Link>
);
};
// Compact Card (Sidebar)
const CompactCourseCard: React.FC<{ course: CourseListItem }> = ({ course }) => (
<Link
to={`/education/courses/${course.slug}`}
className="flex items-center gap-3 p-2 rounded-lg hover:bg-gray-700/50 transition-colors group"
>
{/* Thumbnail */}
<div className="w-12 h-12 flex-shrink-0 rounded-lg overflow-hidden bg-gray-700">
{course.thumbnailUrl ? (
<img
src={course.thumbnailUrl}
alt={course.title}
className="w-full h-full object-cover"
/>
) : (
<div className="w-full h-full flex items-center justify-center">
<BookOpen className="w-5 h-5 text-gray-500" />
</div>
)}
</div>
{/* Content */}
<div className="flex-1 min-w-0">
<h4 className="font-medium text-sm text-white truncate group-hover:text-blue-400 transition-colors">
{course.title}
</h4>
<div className="flex items-center gap-2 mt-0.5">
<div className="flex items-center gap-0.5 text-yellow-400">
<Star className="w-3 h-3 fill-yellow-400" />
<span className="text-xs">{course.avgRating.toFixed(1)}</span>
</div>
<span className="text-xs text-gray-500">{course.totalLessons} lecciones</span>
</div>
</div>
{/* Price/Free badge */}
{course.isFree ? (
<span className="px-2 py-0.5 bg-green-500/20 text-green-400 text-xs rounded">
Gratis
</span>
) : course.priceUsd ? (
<span className="text-sm font-medium text-white">
${course.priceUsd}
</span>
) : null}
</Link>
);
export default RecommendedCourses;

View File

@ -0,0 +1,9 @@
/**
* Education Components Index
* Export all education-related components
*/
export { default as CertificatePreview } from './CertificatePreview';
export { default as CourseReviews } from './CourseReviews';
export { default as LessonNotes } from './LessonNotes';
export { default as RecommendedCourses } from './RecommendedCourses';