shinkan-jinkendo/frontend/src/pages/LegalPage.jsx
Lars 81b9e8f601
All checks were successful
Deploy Development / deploy (push) Successful in 40s
Test Suite / pytest-backend (push) Successful in 35s
Test Suite / lint-backend (push) Successful in 0s
Test Suite / build-frontend (push) Successful in 15s
Test Suite / playwright-tests (push) Successful in 56s
chore: bump version to 0.8.95 and update legal documents features
- Updated app version to 0.8.95 with a new build date of 2026-05-12.
- Enhanced legal documents functionality to support section numbering and Markdown formatting in the output.
- Updated dependencies in package.json to include 'marked' and 'react-markdown'.
- Added new CSS styles for legal document presentation.
- Refactored PDF generation logic to incorporate new metadata and improved document structure.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-12 10:52:06 +02:00

273 lines
10 KiB
JavaScript

import { useState, useEffect } from 'react'
import { Link } from 'react-router-dom'
import LegalDocumentBody from '../components/LegalDocumentBody'
import { generateLegalPdf, legalSectionNumber } from '../utils/legalPdfExport'
import api from '../utils/api'
// document_type values used in the DB / API
const TYPE_MAP = {
impressum: 'impressum',
datenschutz: 'privacy_policy',
nutzungsbedingungen: 'terms_of_use',
medienrichtlinie: 'media_policy',
}
const PAGES = {
impressum: {
title: 'Impressum',
sections: [
{
heading: 'Betreiber / Verantwortlicher',
placeholder: '[Name und Rechtsform des Betreibers — vom Betreiber einzutragen]',
},
{
heading: 'Anschrift',
placeholder: '[Straße, Hausnummer, PLZ, Ort — vom Betreiber einzutragen]',
},
{
heading: 'Vertretungsberechtigte Person',
placeholder: '[Name der vertretungsberechtigten Person — vom Betreiber einzutragen]',
},
{
heading: 'Kontakt',
placeholder: '[E-Mail-Adresse, ggf. Telefonnummer — vom Betreiber einzutragen]',
},
{
heading: 'Registerangaben (falls relevant)',
placeholder: '[Vereinsregister, Handelsregister o. ä. — vom Rechtsanwalt prüfen lassen]',
},
],
},
datenschutz: {
title: 'Datenschutzerklärung',
sections: [
{
heading: 'Verantwortlicher',
placeholder: '[Name, Anschrift und Kontakt des Verantwortlichen — vom Betreiber einzutragen]',
},
{
heading: 'Zwecke der Verarbeitung',
placeholder: '[Welche Daten werden zu welchem Zweck verarbeitet? — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Rechtsgrundlagen',
placeholder: '[Art. 6 DSGVO: Einwilligung, Vertrag, berechtigtes Interesse — vom Rechtsanwalt zu bestimmen]',
},
{
heading: 'Empfänger und Dienstleister',
placeholder: '[SMTP-Anbieter, Hosting, ggf. weitere — vom Rechtsanwalt und Betreiber zu listen]',
},
{
heading: 'Speicherdauern',
placeholder: '[Wie lange werden welche Daten gespeichert? — vom Rechtsanwalt zu bestimmen]',
},
{
heading: 'Betroffenenrechte',
placeholder: '[Auskunft, Berichtigung, Löschung, Widerspruch, Datenübertragbarkeit — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Browser-Speicher (localStorage, sessionStorage)',
placeholder: '[Technisch notwendige Speicherung des Auth-Tokens und Sitzungsdaten. Nach TDDDG §25 ggf. ohne Einwilligung zulässig — vom Rechtsanwalt zu prüfen]',
},
{
heading: 'Kontakt für Datenschutzanfragen',
placeholder: '[E-Mail oder Kontaktweg für Datenschutzanfragen — vom Betreiber einzutragen]',
},
{
heading: 'Zuständige Aufsichtsbehörde',
placeholder: '[Zuständige Datenschutzbehörde je nach Bundesland — vom Rechtsanwalt zu bestimmen]',
},
],
},
nutzungsbedingungen: {
title: 'Nutzungsbedingungen',
sections: [
{
heading: 'Nutzungsumfang',
placeholder: '[Für welche Nutzergruppen und Zwecke darf die Plattform genutzt werden? — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Registrierung',
placeholder: '[Voraussetzungen für die Registrierung, Wahrheitspflicht der Angaben — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Zulässige und unzulässige Inhalte',
placeholder: '[Was darf hochgeladen und veröffentlicht werden? Was ist verboten? — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Verantwortlichkeit der Nutzer',
placeholder: '[Nutzer sind für ihre hochgeladenen Inhalte selbst verantwortlich — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Sperrung und Löschung',
placeholder: '[Unter welchen Bedingungen können Konten oder Inhalte gesperrt oder gelöscht werden? — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Haftungshinweise',
placeholder: '[Haftungsausschluss für Nutzerinhalte, externe Links, Systemausfälle — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Geltungsbereich und anwendbares Recht',
placeholder: '[Welches Recht ist anwendbar? Welcher Gerichtsstand gilt? — vom Rechtsanwalt zu bestimmen]',
},
],
},
medienrichtlinie: {
title: 'Medienrichtlinie',
sections: [
{
heading: 'Urheberrechte',
placeholder: '[Nur eigene oder ausdrücklich lizenzierte Inhalte hochladen — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Rechte am eigenen Bild (§ 22 KUG)',
placeholder: '[Erkennbare Personen müssen eingewilligt haben; besondere Regeln für Minderjährige — juristisch zu prüfen und zu formulieren]',
},
{
heading: 'Minderjährige',
placeholder: '[Besondere Schutzpflichten bei Aufnahmen von Minderjährigen — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Musik und sonstige Fremdinhalte',
placeholder: '[Keine Hintergrundmusik oder andere Fremdinhalte ohne gültige Lizenz — vom Rechtsanwalt zu formulieren]',
},
{
heading: 'Sichtbarkeitsstufen',
placeholder: '[Erläuterung der Stufen: privat, vereinsintern, öffentlich — vom Betreiber zu beschreiben]',
},
{
heading: 'Meldewege für rechtsverletzende Inhalte',
placeholder: '[Wie können Inhalte gemeldet werden? — wird nach Umsetzung von P-13 (Content-Melde-Backend) ergänzt]',
},
{
heading: 'Lösch- und Sperrlogik',
placeholder: '[Wann und wie werden gemeldete Inhalte entfernt oder gesperrt? — wird nach Umsetzung von P-11 und P-13 ergänzt]',
},
],
},
}
const LEGAL_LINKS = [
{ to: '/impressum', label: 'Impressum' },
{ to: '/datenschutz', label: 'Datenschutz' },
{ to: '/nutzungsbedingungen', label: 'Nutzungsbedingungen' },
{ to: '/medienrichtlinie', label: 'Medienrichtlinie' },
]
function LegalPage({ type }) {
const fallback = PAGES[type]
const [apiDoc, setApiDoc] = useState(undefined) // undefined = loading, null = not found
const [loading, setLoading] = useState(true)
const documentType = TYPE_MAP[type]
useEffect(() => {
if (!documentType) {
setLoading(false)
return
}
api.getPublishedLegalDocument(documentType)
.then(doc => setApiDoc(doc))
.catch(() => setApiDoc(null))
.finally(() => setLoading(false))
}, [documentType])
if (!fallback) return null
const isPlaceholder = !apiDoc
const title = apiDoc ? apiDoc.title : fallback.title
const sections = apiDoc
? (apiDoc.content_sections || [])
: fallback.sections.map(s => ({ heading: s.heading, content: s.placeholder }))
return (
<div style={{ minHeight: '100vh', background: 'var(--bg)', padding: '2rem 1rem' }}>
<div style={{ maxWidth: '720px', margin: '0 auto' }}>
<div style={{ marginBottom: '1.5rem' }}>
<Link to="/login" style={{ color: 'var(--accent)', textDecoration: 'none', fontSize: '0.9rem' }}>
Zurück zur Anmeldung
</Link>
</div>
{loading ? (
<div className="spinner" />
) : (
<>
{isPlaceholder && (
<div
className="card"
style={{
marginBottom: '1.5rem',
borderLeft: '4px solid var(--danger)',
background: 'var(--surface)',
}}
>
<strong style={{ color: 'var(--danger)' }}> MUSTER / PLATZHALTER</strong>
<p style={{ margin: '0.5rem 0 0', color: 'var(--text2)', fontSize: '0.9rem' }}>
Inhalt wird vor Produktivbetrieb juristisch geprüft und durch den Betreiber ergänzt.
Diese Seite hat keinen rechtlich verbindlichen Charakter.
</p>
</div>
)}
<div style={{ display: 'flex', alignItems: 'baseline', gap: '1rem', marginBottom: '2rem', flexWrap: 'wrap' }}>
<h1 style={{ margin: 0, color: 'var(--text1)' }}>{title}</h1>
{apiDoc && (
<button
onClick={() => {
const dateStr = apiDoc.published_at
? new Date(apiDoc.published_at).toLocaleDateString('de-DE')
: new Date(apiDoc.updated_at || apiDoc.created_at).toLocaleDateString('de-DE')
const metaLine = `Version ${apiDoc.version} | Gueltig seit ${dateStr}`
generateLegalPdf(apiDoc, metaLine)
}}
className="btn btn-secondary"
style={{ fontSize: '0.82rem', padding: '4px 12px', flexShrink: 0 }}
>
PDF herunterladen
</button>
)}
</div>
{sections.map((section, i) => (
<div key={i} style={{ marginBottom: '1.75rem' }}>
<h2 style={{ fontSize: '1.05rem', marginBottom: '0.4rem', color: 'var(--text1)' }}>
{section.heading
? `${legalSectionNumber(i)} ${section.heading}`
: legalSectionNumber(i)}
</h2>
<LegalDocumentBody content={section.content} muted={isPlaceholder} />
</div>
))}
</>
)}
<div
style={{
marginTop: '3rem',
paddingTop: '1rem',
borderTop: '1px solid var(--border)',
fontSize: '0.82rem',
color: 'var(--text3)',
textAlign: 'center',
display: 'flex',
gap: '1.25rem',
justifyContent: 'center',
flexWrap: 'wrap',
}}
>
{LEGAL_LINKS.map((l) => (
<Link key={l.to} to={l.to} style={{ color: 'var(--text3)' }}>
{l.label}
</Link>
))}
</div>
</div>
</div>
)
}
export default LegalPage