feat: Email Template Approval Workflow im Frontend aktivieren

Backend-Endpoints existierten bereits (submit/approve/reject/publish),
wurden aber vom Frontend nicht genutzt. Jetzt vollstaendiger Workflow:

- Submit for Review: Entwurf → Pruefung einreichen
- Approve/Reject: DSB kann genehmigen oder mit Begruendung ablehnen
- Publish: Genehmigte Version veroeffentlichen
- Test senden: Test-E-Mail an beliebige Adresse
- Approval History: Genehmigungshistorie abrufbar
- Status-Badges: draft/review/approved/published mit passenden Buttons

Alle Buttons sind kontextabhaengig — nur sichtbar wenn der Status passt.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Benjamin Admin
2026-04-27 23:42:26 +02:00
parent 918a9d8092
commit 8e0645481a
4 changed files with 136 additions and 20 deletions

View File

@@ -7,6 +7,7 @@ import {
SendLog,
Settings,
TabId,
TemplateApproval,
TemplateType,
TemplateVersion,
getHeaders,
@@ -194,6 +195,72 @@ export function useEmailTemplates(activeTab: TabId) {
}
}, [settingsForm])
// Workflow actions
const submitForReview = useCallback(async () => {
if (!editorVersion) return
setSaving(true)
try {
const res = await fetch(`${API_BASE}/versions/${editorVersion.id}/submit`, {
method: 'POST', headers: getHeaders(),
})
if (!res.ok) throw new Error(`HTTP ${res.status}`)
const updated = await res.json()
setEditorVersion(updated)
await loadTemplates()
} catch (e: any) { setError(e.message) } finally { setSaving(false) }
}, [editorVersion, loadTemplates])
const approveVersion = useCallback(async (comment?: string) => {
if (!editorVersion) return
setSaving(true)
try {
const res = await fetch(`${API_BASE}/versions/${editorVersion.id}/approve`, {
method: 'POST', headers: getHeaders(),
body: JSON.stringify({ comment: comment || '' }),
})
if (!res.ok) throw new Error(`HTTP ${res.status}`)
const updated = await res.json()
setEditorVersion(updated)
await loadTemplates()
} catch (e: any) { setError(e.message) } finally { setSaving(false) }
}, [editorVersion, loadTemplates])
const rejectVersion = useCallback(async (comment: string) => {
if (!editorVersion) return
setSaving(true)
try {
const res = await fetch(`${API_BASE}/versions/${editorVersion.id}/reject`, {
method: 'POST', headers: getHeaders(),
body: JSON.stringify({ comment }),
})
if (!res.ok) throw new Error(`HTTP ${res.status}`)
const updated = await res.json()
setEditorVersion(updated)
await loadTemplates()
} catch (e: any) { setError(e.message) } finally { setSaving(false) }
}, [editorVersion, loadTemplates])
const sendTestEmail = useCallback(async (recipientEmail: string) => {
if (!editorVersion) return
try {
const res = await fetch(`${API_BASE}/versions/${editorVersion.id}/send-test`, {
method: 'POST', headers: getHeaders(),
body: JSON.stringify({ recipient: recipientEmail }),
})
if (!res.ok) throw new Error(`HTTP ${res.status}`)
return await res.json()
} catch (e: any) { setError(e.message) }
}, [editorVersion])
const loadApprovalHistory = useCallback(async (versionId: string): Promise<TemplateApproval[]> => {
try {
const res = await fetch(`${API_BASE}/versions/${versionId}/approvals`, { headers: getHeaders() })
if (!res.ok) return []
const data = await res.json()
return Array.isArray(data) ? data : data.approvals || []
} catch { return [] }
}, [])
const initializeDefaults = useCallback(async () => {
try {
const res = await fetch(`${API_BASE}/initialize`, {
@@ -222,6 +289,8 @@ export function useEmailTemplates(activeTab: TabId) {
setSettingsForm,
// Actions
openEditor, saveVersion, publishVersion, loadPreview,
submitForReview, approveVersion, rejectVersion,
sendTestEmail, loadApprovalHistory,
saveSettings2, initializeDefaults,
}
}