This repository has been archived on 2026-02-15. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
breakpilot-pwa/studio-v2/app/worksheet-editor/page.tsx
BreakPilot Dev 916ecef476 feat(worksheet-editor): Add OCR import panel for grid analysis data
Add OCRImportPanel component and ocr-integration utilities to import
OCR-analyzed data from the grid detection service into the worksheet editor.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-09 23:50:35 +01:00

493 lines
20 KiB
TypeScript

'use client'
import { useState, useEffect, useCallback } from 'react'
import dynamic from 'next/dynamic'
import { useTheme } from '@/lib/ThemeContext'
import { useLanguage } from '@/lib/LanguageContext'
import { WorksheetProvider, useWorksheet } from '@/lib/worksheet-editor/WorksheetContext'
import { Sidebar } from '@/components/Sidebar'
import { EditorToolbar } from '@/components/worksheet-editor/EditorToolbar'
import { PropertiesPanel } from '@/components/worksheet-editor/PropertiesPanel'
import { CanvasControls } from '@/components/worksheet-editor/CanvasControls'
import { PageNavigator } from '@/components/worksheet-editor/PageNavigator'
import { AIImageGenerator } from '@/components/worksheet-editor/AIImageGenerator'
import { ExportPanel } from '@/components/worksheet-editor/ExportPanel'
import { AIPromptBar } from '@/components/worksheet-editor/AIPromptBar'
import { DocumentImporter } from '@/components/worksheet-editor/DocumentImporter'
import { CleanupPanel } from '@/components/worksheet-editor/CleanupPanel'
import { OCRImportPanel } from '@/components/worksheet-editor/OCRImportPanel'
import { ThemeToggle } from '@/components/ThemeToggle'
import { LanguageDropdown } from '@/components/LanguageDropdown'
// Dynamic import to prevent SSR issues with Fabric.js
const FabricCanvas = dynamic(
() => import('@/components/worksheet-editor/FabricCanvas').then(mod => mod.FabricCanvas),
{
ssr: false,
loading: () => (
<div className="flex items-center justify-center h-full">
<div className="animate-spin w-8 h-8 border-4 border-purple-500 border-t-transparent rounded-full" />
</div>
)
}
)
// Storage key for saved worksheets
const WORKSHEETS_KEY = 'bp_worksheets'
interface SavedWorksheet {
id: string
title: string
updatedAt: string
thumbnail?: string
}
function WorksheetEditorContent() {
const { isDark } = useTheme()
const { t } = useLanguage()
const { document, setDocument, isDirty, setIsDirty, saveDocument, loadDocument, canvas } = useWorksheet()
const [mounted, setMounted] = useState(false)
const [isAIGeneratorOpen, setIsAIGeneratorOpen] = useState(false)
const [isExportPanelOpen, setIsExportPanelOpen] = useState(false)
const [isDocumentImporterOpen, setIsDocumentImporterOpen] = useState(false)
const [isCleanupPanelOpen, setIsCleanupPanelOpen] = useState(false)
const [isOCRImportOpen, setIsOCRImportOpen] = useState(false)
const [isDocumentListOpen, setIsDocumentListOpen] = useState(false)
const [savedWorksheets, setSavedWorksheets] = useState<SavedWorksheet[]>([])
const [isSaving, setIsSaving] = useState(false)
const [title, setTitle] = useState('')
useEffect(() => {
setMounted(true)
loadSavedWorksheets()
}, [])
useEffect(() => {
if (document) {
setTitle(document.title)
}
}, [document])
// Load saved worksheets from localStorage
const loadSavedWorksheets = useCallback(() => {
try {
const stored = localStorage.getItem(WORKSHEETS_KEY)
if (stored) {
setSavedWorksheets(JSON.parse(stored))
}
} catch (e) {
console.error('Failed to load worksheets:', e)
}
}, [])
// Save current worksheet
const handleSave = useCallback(async () => {
if (!document) return
setIsSaving(true)
try {
// Save to context (which saves to API or localStorage)
await saveDocument()
// Update worksheets list
const worksheetEntry: SavedWorksheet = {
id: document.id,
title: document.title,
updatedAt: new Date().toISOString(),
thumbnail: canvas?.toDataURL({ format: 'png', multiplier: 0.1 })
}
setSavedWorksheets(prev => {
const filtered = prev.filter(w => w.id !== document.id)
const updated = [worksheetEntry, ...filtered]
localStorage.setItem(WORKSHEETS_KEY, JSON.stringify(updated))
return updated
})
setIsDirty(false)
} catch (e) {
console.error('Save failed:', e)
} finally {
setIsSaving(false)
}
}, [document, saveDocument, canvas, setIsDirty])
// Load a saved worksheet
const handleLoadWorksheet = useCallback(async (id: string) => {
try {
await loadDocument(id)
setIsDocumentListOpen(false)
} catch (e) {
console.error('Failed to load worksheet:', e)
}
}, [loadDocument])
// Delete a saved worksheet
const handleDeleteWorksheet = useCallback((id: string) => {
setSavedWorksheets(prev => {
const updated = prev.filter(w => w.id !== id)
localStorage.setItem(WORKSHEETS_KEY, JSON.stringify(updated))
localStorage.removeItem(`worksheet_${id}`)
return updated
})
}, [])
// Create new worksheet
const handleNewWorksheet = useCallback(() => {
const newDoc = {
id: `ws_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`,
title: 'Neues Arbeitsblatt',
pages: [{
id: `page_${Date.now()}`,
index: 0,
canvasJSON: ''
}],
pageFormat: {
width: 210,
height: 297,
orientation: 'portrait' as const,
margins: { top: 15, right: 15, bottom: 15, left: 15 }
},
createdAt: new Date().toISOString(),
updatedAt: new Date().toISOString()
}
setDocument(newDoc)
setIsDocumentListOpen(false)
if (canvas) {
canvas.clear()
canvas.backgroundColor = '#ffffff'
canvas.renderAll()
}
}, [setDocument, canvas])
const handleTitleChange = (newTitle: string) => {
setTitle(newTitle)
if (document) {
setDocument({
...document,
title: newTitle,
updatedAt: new Date().toISOString()
})
}
}
if (!mounted) {
return (
<div className={`min-h-screen flex items-center justify-center ${
isDark ? 'bg-slate-900' : 'bg-slate-100'
}`}>
<div className="animate-spin w-8 h-8 border-4 border-purple-500 border-t-transparent rounded-full" />
</div>
)
}
return (
<div className={`min-h-screen flex relative overflow-hidden ${
isDark
? 'bg-gradient-to-br from-indigo-900 via-purple-900 to-pink-800'
: 'bg-gradient-to-br from-slate-100 via-blue-50 to-indigo-100'
}`}>
{/* Animated Background Blobs */}
<div className={`absolute -top-40 -right-40 w-96 h-96 rounded-full mix-blend-multiply filter blur-3xl animate-blob ${
isDark ? 'bg-purple-500 opacity-30' : 'bg-purple-300 opacity-40'
}`} />
<div className={`absolute top-1/2 -left-40 w-96 h-96 rounded-full mix-blend-multiply filter blur-3xl animate-blob animation-delay-2000 ${
isDark ? 'bg-pink-500 opacity-30' : 'bg-pink-300 opacity-40'
}`} />
<div className={`absolute -bottom-40 right-1/3 w-96 h-96 rounded-full mix-blend-multiply filter blur-3xl animate-blob animation-delay-4000 ${
isDark ? 'bg-blue-500 opacity-30' : 'bg-blue-300 opacity-40'
}`} />
{/* Sidebar */}
<div className="relative z-10 p-4">
<Sidebar />
</div>
{/* Main Content */}
<div className="flex-1 flex flex-col relative z-10 p-4 overflow-hidden">
{/* Header */}
<div className="flex items-center justify-between mb-4">
<div className="flex items-center gap-4">
<div>
<h1 className={`text-2xl font-bold mb-1 ${isDark ? 'text-white' : 'text-slate-900'}`}>Arbeitsblatt-Editor</h1>
<div className="flex items-center gap-2">
<input
type="text"
value={title}
onChange={(e) => handleTitleChange(e.target.value)}
placeholder="Arbeitsblatt-Titel..."
className={`text-sm px-3 py-1.5 rounded-lg border transition-all w-56 ${
isDark
? 'bg-white/10 border-white/20 text-white placeholder-white/40 focus:border-purple-400'
: 'bg-white/50 border-slate-300 text-slate-900 placeholder-slate-400 focus:border-purple-500'
}`}
/>
{isDirty && (
<span className={`px-2 py-1 rounded-lg text-xs font-medium ${
isDark ? 'bg-yellow-500/20 text-yellow-300' : 'bg-yellow-100 text-yellow-700'
}`}>
Ungespeichert
</span>
)}
{/* Save Button */}
<button
onClick={handleSave}
disabled={isSaving || !isDirty}
className={`flex items-center gap-1.5 px-3 py-1.5 rounded-lg text-sm font-medium transition-all ${
isDirty
? 'bg-green-500 text-white hover:bg-green-600'
: isDark
? 'bg-white/10 text-white/40 cursor-not-allowed'
: 'bg-slate-200 text-slate-400 cursor-not-allowed'
}`}
>
{isSaving ? (
<div className="w-4 h-4 border-2 border-white border-t-transparent rounded-full animate-spin" />
) : (
<svg className="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M8 7H5a2 2 0 00-2 2v9a2 2 0 002 2h14a2 2 0 002-2V9a2 2 0 00-2-2h-3m-1 4l-3 3m0 0l-3-3m3 3V4" />
</svg>
)}
Speichern
</button>
</div>
</div>
</div>
<div className="flex items-center gap-2">
{/* Document List Button */}
<button
onClick={() => setIsDocumentListOpen(true)}
className={`flex items-center gap-2 px-3 py-2 rounded-xl font-medium transition-all ${
isDark
? 'bg-white/10 text-white hover:bg-white/20'
: 'bg-white text-slate-700 hover:bg-slate-100'
}`}
>
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
</svg>
Meine Arbeitsblätter
</button>
{/* Export Button */}
<button
onClick={() => setIsExportPanelOpen(true)}
className={`flex items-center gap-2 px-3 py-2 rounded-xl font-medium transition-all ${
isDark
? 'bg-purple-500/30 text-purple-300 hover:bg-purple-500/40'
: 'bg-purple-600 text-white hover:bg-purple-700'
}`}
>
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4" />
</svg>
Exportieren
</button>
{/* Theme Toggle */}
<ThemeToggle />
{/* Language Dropdown */}
<LanguageDropdown />
</div>
</div>
{/* Editor Area - New Layout */}
<div className="flex-1 flex gap-4 overflow-hidden">
{/* Left Toolbar */}
<div className="flex-shrink-0">
<EditorToolbar
onOpenAIGenerator={() => setIsAIGeneratorOpen(true)}
onOpenDocumentImporter={() => setIsDocumentImporterOpen(true)}
onOpenCleanupPanel={() => setIsCleanupPanelOpen(true)}
onOpenOCRImport={() => setIsOCRImportOpen(true)}
className="h-full"
/>
</div>
{/* Canvas Area - takes remaining space */}
<div className="flex-1 flex flex-col min-w-0 overflow-hidden">
{/* Canvas with fixed aspect ratio container */}
<div className={`flex-1 overflow-auto rounded-xl ${
isDark ? 'bg-slate-800/50' : 'bg-slate-200/50'
}`}>
<FabricCanvas className="h-full" />
</div>
{/* Bottom Controls */}
<div className="flex items-center justify-between gap-4 py-2">
<PageNavigator />
<CanvasControls />
</div>
</div>
{/* Right Panel - AI Prompt + Properties */}
<div className="w-80 flex-shrink-0 flex flex-col gap-4 overflow-hidden">
{/* AI Prompt Bar */}
<div className="flex-shrink-0">
<AIPromptBar />
</div>
{/* Properties Panel */}
<div className="flex-1 overflow-hidden">
<PropertiesPanel className="h-full" />
</div>
</div>
</div>
</div>
{/* Document List Modal */}
{isDocumentListOpen && (
<div className="fixed inset-0 z-50 flex items-center justify-center p-4">
<div className="absolute inset-0 bg-black/50 backdrop-blur-sm" onClick={() => setIsDocumentListOpen(false)} />
<div className={`relative w-full max-w-2xl rounded-3xl p-6 ${
isDark ? 'bg-slate-900/95' : 'bg-white/95'
} backdrop-blur-xl border ${isDark ? 'border-white/10' : 'border-slate-200'}`}>
<div className="flex items-center justify-between mb-6">
<h2 className={`text-xl font-bold ${isDark ? 'text-white' : 'text-slate-900'}`}>
Meine Arbeitsblätter
</h2>
<button
onClick={() => setIsDocumentListOpen(false)}
className={`p-2 rounded-lg transition-colors ${isDark ? 'hover:bg-white/10' : 'hover:bg-slate-100'}`}
>
<svg className={`w-5 h-5 ${isDark ? 'text-white' : 'text-slate-600'}`} fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
</svg>
</button>
</div>
{/* New Worksheet Button */}
<button
onClick={handleNewWorksheet}
className={`w-full mb-4 p-4 rounded-xl border-2 border-dashed transition-all flex items-center justify-center gap-2 ${
isDark
? 'border-white/20 text-white/60 hover:border-purple-400 hover:text-purple-300 hover:bg-purple-500/10'
: 'border-slate-300 text-slate-500 hover:border-purple-500 hover:text-purple-600 hover:bg-purple-50'
}`}
>
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 4v16m8-8H4" />
</svg>
Neues Arbeitsblatt erstellen
</button>
{/* Worksheets List */}
<div className="max-h-96 overflow-y-auto space-y-2">
{savedWorksheets.length === 0 ? (
<div className={`text-center py-8 ${isDark ? 'text-white/40' : 'text-slate-400'}`}>
<svg className="w-12 h-12 mx-auto mb-3 opacity-50" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={1.5} d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
</svg>
<p>Noch keine Arbeitsblätter gespeichert</p>
</div>
) : (
savedWorksheets.map((worksheet) => (
<div
key={worksheet.id}
className={`flex items-center gap-4 p-4 rounded-xl transition-all cursor-pointer ${
isDark
? 'bg-white/5 hover:bg-white/10'
: 'bg-slate-50 hover:bg-slate-100'
} ${document?.id === worksheet.id ? (isDark ? 'ring-2 ring-purple-500' : 'ring-2 ring-purple-500') : ''}`}
onClick={() => handleLoadWorksheet(worksheet.id)}
>
{/* Thumbnail */}
<div className={`w-16 h-20 rounded-lg flex-shrink-0 overflow-hidden ${
isDark ? 'bg-white/10' : 'bg-slate-200'
}`}>
{worksheet.thumbnail ? (
<img src={worksheet.thumbnail} alt="" className="w-full h-full object-cover" />
) : (
<div className="w-full h-full flex items-center justify-center">
<svg className={`w-6 h-6 ${isDark ? 'text-white/30' : 'text-slate-400'}`} fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={1.5} d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z" />
</svg>
</div>
)}
</div>
{/* Info */}
<div className="flex-1 min-w-0">
<h3 className={`font-medium truncate ${isDark ? 'text-white' : 'text-slate-900'}`}>
{worksheet.title}
</h3>
<p className={`text-sm ${isDark ? 'text-white/50' : 'text-slate-500'}`}>
{new Date(worksheet.updatedAt).toLocaleDateString('de-DE', {
day: '2-digit',
month: '2-digit',
year: 'numeric',
hour: '2-digit',
minute: '2-digit'
})}
</p>
{document?.id === worksheet.id && (
<span className="inline-block mt-1 px-2 py-0.5 rounded text-xs bg-purple-500/20 text-purple-400">
Aktuell geöffnet
</span>
)}
</div>
{/* Delete Button */}
<button
onClick={(e) => {
e.stopPropagation()
if (confirm('Arbeitsblatt wirklich löschen?')) {
handleDeleteWorksheet(worksheet.id)
}
}}
className={`p-2 rounded-lg transition-colors ${
isDark ? 'hover:bg-red-500/20 text-white/50 hover:text-red-400' : 'hover:bg-red-50 text-slate-400 hover:text-red-500'
}`}
>
<svg className="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16" />
</svg>
</button>
</div>
))
)}
</div>
</div>
</div>
)}
{/* Modals */}
<AIImageGenerator
isOpen={isAIGeneratorOpen}
onClose={() => setIsAIGeneratorOpen(false)}
/>
<ExportPanel
isOpen={isExportPanelOpen}
onClose={() => setIsExportPanelOpen(false)}
/>
<DocumentImporter
isOpen={isDocumentImporterOpen}
onClose={() => setIsDocumentImporterOpen(false)}
/>
<CleanupPanel
isOpen={isCleanupPanelOpen}
onClose={() => setIsCleanupPanelOpen(false)}
/>
<OCRImportPanel
isOpen={isOCRImportOpen}
onClose={() => setIsOCRImportOpen(false)}
/>
</div>
)
}
export default function WorksheetEditorPage() {
return (
<WorksheetProvider>
<WorksheetEditorContent />
</WorksheetProvider>
)
}