Compare commits
2 Commits
9cef1c78d3
...
2e1f1e599b
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
2e1f1e599b | ||
|
|
71c4e2509f |
@@ -2328,6 +2328,7 @@ export default function AdminPage({ params }: { params: { locale: string } }) {
|
|||||||
<p style={{ marginBottom: '1rem', color: '#666' }}>
|
<p style={{ marginBottom: '1rem', color: '#666' }}>
|
||||||
These actions are destructive and cannot be undone.
|
These actions are destructive and cannot be undone.
|
||||||
</p>
|
</p>
|
||||||
|
<div style={{ display: 'flex', gap: '1rem', flexWrap: 'wrap' }}>
|
||||||
<button
|
<button
|
||||||
onClick={async () => {
|
onClick={async () => {
|
||||||
if (window.confirm('⚠️ WARNING: This will delete ALL data from the database (Songs, Genres, Specials, Puzzles) and re-import songs from the uploads folder.\n\nExisting genres and specials will be LOST and must be recreated manually.\n\nAre you sure you want to proceed?')) {
|
if (window.confirm('⚠️ WARNING: This will delete ALL data from the database (Songs, Genres, Specials, Puzzles) and re-import songs from the uploads folder.\n\nExisting genres and specials will be LOST and must be recreated manually.\n\nAre you sure you want to proceed?')) {
|
||||||
@@ -2361,7 +2362,83 @@ export default function AdminPage({ params }: { params: { locale: string } }) {
|
|||||||
☢️ Rebuild Database
|
☢️ Rebuild Database
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
onClick={async () => {
|
||||||
|
if (window.confirm('⚠️ WARNING: This will reset ALL user ratings for all songs to 0.\n\nThis action cannot be undone.\n\nAre you sure you want to proceed?')) {
|
||||||
|
try {
|
||||||
|
setMessage('Resetting all ratings...');
|
||||||
|
const res = await fetch('/api/admin/reset-ratings', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: getAuthHeaders()
|
||||||
|
});
|
||||||
|
if (res.ok) {
|
||||||
|
const data = await res.json();
|
||||||
|
alert(data.message);
|
||||||
|
fetchSongs();
|
||||||
|
setMessage('');
|
||||||
|
} else {
|
||||||
|
alert('Failed to reset ratings. Check server logs.');
|
||||||
|
setMessage('Reset failed.');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
alert('Reset failed due to network error.');
|
||||||
|
setMessage('');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
style={{
|
||||||
|
padding: '0.75rem 1.5rem',
|
||||||
|
background: '#f59e0b',
|
||||||
|
color: 'white',
|
||||||
|
border: 'none',
|
||||||
|
borderRadius: '0.25rem',
|
||||||
|
cursor: 'pointer',
|
||||||
|
fontWeight: 'bold'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
🔄 Reset All User Ratings
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
onClick={async () => {
|
||||||
|
if (window.confirm('⚠️ WARNING: This will delete ALL daily puzzles (activations) from the database.\n\nThis means all songs will show 0 activations.\n\nThis action cannot be undone.\n\nAre you sure you want to proceed?')) {
|
||||||
|
try {
|
||||||
|
setMessage('Resetting all activations...');
|
||||||
|
const res = await fetch('/api/admin/reset-activations', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: getAuthHeaders()
|
||||||
|
});
|
||||||
|
if (res.ok) {
|
||||||
|
const data = await res.json();
|
||||||
|
alert(data.message);
|
||||||
|
fetchSongs();
|
||||||
|
fetchDailyPuzzles();
|
||||||
|
setMessage('');
|
||||||
|
} else {
|
||||||
|
alert('Failed to reset activations. Check server logs.');
|
||||||
|
setMessage('Reset failed.');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
alert('Reset failed due to network error.');
|
||||||
|
setMessage('');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
style={{
|
||||||
|
padding: '0.75rem 1.5rem',
|
||||||
|
background: '#f59e0b',
|
||||||
|
color: 'white',
|
||||||
|
border: 'none',
|
||||||
|
borderRadius: '0.25rem',
|
||||||
|
cursor: 'pointer',
|
||||||
|
fontWeight: 'bold'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
🔄 Reset All Activations
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,14 +0,0 @@
|
|||||||
import type { Metadata } from "next";
|
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
|
||||||
title: "Hördle Admin Dashboard",
|
|
||||||
description: "Admin dashboard for managing songs and daily puzzles",
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function AdminLayout({
|
|
||||||
children,
|
|
||||||
}: {
|
|
||||||
children: React.ReactNode;
|
|
||||||
}) {
|
|
||||||
return <>{children}</>;
|
|
||||||
}
|
|
||||||
1185
app/admin/page.tsx
1185
app/admin/page.tsx
File diff suppressed because it is too large
Load Diff
@@ -1,89 +0,0 @@
|
|||||||
'use client';
|
|
||||||
|
|
||||||
import { useEffect, useState } from 'react';
|
|
||||||
import { useParams, useRouter, usePathname } from 'next/navigation';
|
|
||||||
import CurateSpecialEditor, { CurateSpecial } from '@/components/CurateSpecialEditor';
|
|
||||||
|
|
||||||
export default function SpecialEditorPage() {
|
|
||||||
const params = useParams();
|
|
||||||
const router = useRouter();
|
|
||||||
const pathname = usePathname();
|
|
||||||
const specialId = params.id as string;
|
|
||||||
|
|
||||||
// Locale aus dem Pfad ableiten (/en/..., /de/...)
|
|
||||||
const localeFromPath = pathname?.split('/')[1] as 'de' | 'en' | undefined;
|
|
||||||
const locale: 'de' | 'en' = localeFromPath === 'de' || localeFromPath === 'en' ? localeFromPath : 'en';
|
|
||||||
|
|
||||||
const [special, setSpecial] = useState<CurateSpecial | null>(null);
|
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
|
|
||||||
const fetchSpecial = async (showLoading = true) => {
|
|
||||||
try {
|
|
||||||
if (showLoading) {
|
|
||||||
setLoading(true);
|
|
||||||
}
|
|
||||||
const res = await fetch(`/api/specials/${specialId}`);
|
|
||||||
if (res.ok) {
|
|
||||||
const data = await res.json();
|
|
||||||
setSpecial(data);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error fetching special:', error);
|
|
||||||
} finally {
|
|
||||||
if (showLoading) {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
fetchSpecial(true);
|
|
||||||
}, [specialId]);
|
|
||||||
|
|
||||||
const handleSaveStartTime = async (songId: number, startTime: number) => {
|
|
||||||
const res = await fetch(`/api/specials/${specialId}/songs`, {
|
|
||||||
method: 'PUT',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify({ songId, startTime }),
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!res.ok) {
|
|
||||||
const errorText = await res.text().catch(() => res.statusText || 'Unknown error');
|
|
||||||
console.error('Error updating special song (admin):', res.status, errorText);
|
|
||||||
throw new Error(`Failed to save start time: ${errorText}`);
|
|
||||||
} else {
|
|
||||||
// Reload special data to update the start time in the song list
|
|
||||||
await fetchSpecial(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (loading) {
|
|
||||||
return (
|
|
||||||
<div style={{ padding: '2rem', textAlign: 'center' }}>
|
|
||||||
<p>Loading...</p>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!special) {
|
|
||||||
return (
|
|
||||||
<div style={{ padding: '2rem', textAlign: 'center' }}>
|
|
||||||
<p>Special not found</p>
|
|
||||||
<button onClick={() => router.push('/admin')}>Back to Admin</button>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<CurateSpecialEditor
|
|
||||||
special={special}
|
|
||||||
locale={locale}
|
|
||||||
onBack={() => router.push('/admin')}
|
|
||||||
onSaveStartTime={handleSaveStartTime}
|
|
||||||
backLabel="← Back to Admin"
|
|
||||||
headerPrefix="Edit Special:"
|
|
||||||
noSongsSubHint="Go back to the admin dashboard to add songs to this special."
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
23
app/api/admin/reset-activations/route.ts
Normal file
23
app/api/admin/reset-activations/route.ts
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { PrismaClient } from '@prisma/client';
|
||||||
|
|
||||||
|
const prisma = new PrismaClient();
|
||||||
|
|
||||||
|
export async function POST(req: NextRequest) {
|
||||||
|
try {
|
||||||
|
// Delete all daily puzzles (activations)
|
||||||
|
const result = await prisma.dailyPuzzle.deleteMany({});
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: `Successfully deleted ${result.count} daily puzzles (activations)`,
|
||||||
|
count: result.count,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error resetting activations:', error);
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to reset activations' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
28
app/api/admin/reset-ratings/route.ts
Normal file
28
app/api/admin/reset-ratings/route.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import { NextRequest, NextResponse } from 'next/server';
|
||||||
|
import { PrismaClient } from '@prisma/client';
|
||||||
|
|
||||||
|
const prisma = new PrismaClient();
|
||||||
|
|
||||||
|
export async function POST(req: NextRequest) {
|
||||||
|
try {
|
||||||
|
// Reset all song ratings to 0
|
||||||
|
const result = await prisma.song.updateMany({
|
||||||
|
data: {
|
||||||
|
averageRating: 0,
|
||||||
|
ratingCount: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return NextResponse.json({
|
||||||
|
success: true,
|
||||||
|
message: `Successfully reset ratings for ${result.count} songs`,
|
||||||
|
count: result.count,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error resetting ratings:', error);
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: 'Failed to reset ratings' },
|
||||||
|
{ status: 500 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,171 +0,0 @@
|
|||||||
'use client';
|
|
||||||
|
|
||||||
import { useTranslations, useLocale } from 'next-intl';
|
|
||||||
import { Link } from '@/lib/navigation';
|
|
||||||
|
|
||||||
export default function CuratorHelpClient() {
|
|
||||||
const t = useTranslations('CuratorHelp');
|
|
||||||
const locale = useLocale();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<main style={{ maxWidth: '960px', margin: '2rem auto', padding: '1rem' }}>
|
|
||||||
<header style={{ marginBottom: '2rem' }}>
|
|
||||||
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
|
||||||
<h1 style={{ fontSize: '1.75rem', marginBottom: '0.25rem' }}>{t('title')}</h1>
|
|
||||||
<Link
|
|
||||||
href="/curator"
|
|
||||||
style={{
|
|
||||||
padding: '0.5rem 1rem',
|
|
||||||
background: '#6b7280',
|
|
||||||
color: 'white',
|
|
||||||
textDecoration: 'none',
|
|
||||||
borderRadius: '0.375rem',
|
|
||||||
fontSize: '0.9rem',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{t('backToDashboard')}
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
<div style={{ display: 'flex', flexDirection: 'column', gap: '2rem' }}>
|
|
||||||
{/* Einführung */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('introductionTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<p style={{ marginBottom: '1rem' }}>{t('introductionText')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1.5rem', marginBottom: '0.75rem' }}>{t('permissionsTitle')}</h3>
|
|
||||||
<ul style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('permission1')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('permission2')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('permission3')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('permission4')}</li>
|
|
||||||
</ul>
|
|
||||||
<p style={{ marginTop: '1rem', padding: '0.75rem', background: '#fef3c7', borderRadius: '0.375rem', border: '1px solid #fbbf24' }}>
|
|
||||||
<strong>{t('note')}:</strong> {t('permissionNote')}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Song-Upload */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('uploadTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('uploadStepsTitle')}</h3>
|
|
||||||
<ol style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadStep1')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadStep2')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadStep3')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadStep4')}</li>
|
|
||||||
</ol>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1.5rem', marginBottom: '0.75rem' }}>{t('uploadBestPracticesTitle')}</h3>
|
|
||||||
<ul style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadBestPractice1')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadBestPractice2')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('uploadBestPractice3')}</li>
|
|
||||||
</ul>
|
|
||||||
<p style={{ marginTop: '1rem', padding: '0.75rem', background: '#dbeafe', borderRadius: '0.375rem', border: '1px solid #3b82f6' }}>
|
|
||||||
<strong>{t('tip')}:</strong> {t('uploadTip')}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Song-Bearbeitung */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('editingTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('singleEditTitle')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem' }}>{t('singleEditText')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1.5rem', marginBottom: '0.75rem' }}>{t('batchEditTitle')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem' }}>{t('batchEditText')}</p>
|
|
||||||
<ul style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('batchEditFeature1')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('batchEditFeature2')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('batchEditFeature3')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('batchEditFeature4')}</li>
|
|
||||||
</ul>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1.5rem', marginBottom: '0.75rem' }}>{t('genreSpecialAssignmentTitle')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem' }}>{t('genreSpecialAssignmentText')}</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Specials kuratieren */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('curateSpecialsHelpTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<p style={{ marginBottom: '1rem' }}>{t('curateSpecialsHelpIntro')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>
|
|
||||||
{t('curateSpecialsHelpStepsTitle')}
|
|
||||||
</h3>
|
|
||||||
<ol style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('curateSpecialsHelpStep1')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('curateSpecialsHelpStep2')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('curateSpecialsHelpStep3')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}>{t('curateSpecialsHelpStep4')}</li>
|
|
||||||
</ol>
|
|
||||||
<p style={{ marginTop: '1rem', padding: '0.75rem', background: '#fef3c7', borderRadius: '0.375rem', border: '1px solid #fbbf24' }}>
|
|
||||||
<strong>{t('note')}:</strong> {t('curateSpecialsPermissionsNote')}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Kommentar-Verwaltung */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('commentsTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<p style={{ marginBottom: '1rem' }}>{t('commentsText')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('commentsActionsTitle')}</h3>
|
|
||||||
<ul style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}><strong>{t('markAsRead')}:</strong> {t('markAsReadText')}</li>
|
|
||||||
<li style={{ marginBottom: '0.5rem' }}><strong>{t('archive')}:</strong> {t('archiveText')}</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Best Practices */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('bestPracticesTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<ul style={{ marginLeft: '1.5rem', marginBottom: '1rem' }}>
|
|
||||||
<li style={{ marginBottom: '0.75rem' }}>{t('bestPractice1')}</li>
|
|
||||||
<li style={{ marginBottom: '0.75rem' }}>{t('bestPractice2')}</li>
|
|
||||||
<li style={{ marginBottom: '0.75rem' }}>{t('bestPractice3')}</li>
|
|
||||||
<li style={{ marginBottom: '0.75rem' }}>{t('bestPractice4')}</li>
|
|
||||||
<li style={{ marginBottom: '0.75rem' }}>{t('bestPractice5')}</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
{/* Troubleshooting */}
|
|
||||||
<section>
|
|
||||||
<h2 style={{ fontSize: '1.5rem', marginBottom: '1rem', borderBottom: '2px solid #e5e7eb', paddingBottom: '0.5rem' }}>
|
|
||||||
{t('troubleshootingTitle')}
|
|
||||||
</h2>
|
|
||||||
<div style={{ fontSize: '0.95rem', lineHeight: '1.7' }}>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('troubleshootingQ1')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem', marginLeft: '1rem' }}>{t('troubleshootingA1')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('troubleshootingQ2')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem', marginLeft: '1rem' }}>{t('troubleshootingA2')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('troubleshootingQ3')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem', marginLeft: '1rem' }}>{t('troubleshootingA3')}</p>
|
|
||||||
<h3 style={{ fontSize: '1.1rem', marginTop: '1rem', marginBottom: '0.75rem' }}>{t('troubleshootingQ4')}</h3>
|
|
||||||
<p style={{ marginBottom: '1rem', marginLeft: '1rem' }}>{t('troubleshootingA4')}</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</main>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
export const dynamic = 'force-dynamic';
|
|
||||||
|
|
||||||
import CuratorHelpClient from './CuratorHelpClient';
|
|
||||||
|
|
||||||
export default function CuratorHelpPage() {
|
|
||||||
return <CuratorHelpClient />;
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
// Server-Wrapper für die Kuratoren-Seite.
|
|
||||||
// Markiert die Route als dynamisch und rendert die eigentliche Client-Komponente.
|
|
||||||
export const dynamic = 'force-dynamic';
|
|
||||||
|
|
||||||
import CuratorPageClient from './CuratorPageClient';
|
|
||||||
|
|
||||||
export default function CuratorPage() {
|
|
||||||
return <CuratorPageClient />;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,156 +0,0 @@
|
|||||||
'use client';
|
|
||||||
|
|
||||||
import { useEffect, useState } from 'react';
|
|
||||||
import { useRouter, usePathname } from 'next/navigation';
|
|
||||||
import { useLocale, useTranslations } from 'next-intl';
|
|
||||||
import { Link } from '@/lib/navigation';
|
|
||||||
import { getCuratorAuthHeaders } from '@/lib/curatorAuth';
|
|
||||||
import { getLocalizedValue } from '@/lib/i18n';
|
|
||||||
|
|
||||||
interface CuratorSpecial {
|
|
||||||
id: number;
|
|
||||||
name: string | { de?: string; en?: string };
|
|
||||||
songCount: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function CuratorSpecialsClient() {
|
|
||||||
const router = useRouter();
|
|
||||||
const pathname = usePathname();
|
|
||||||
const urlLocale = pathname?.split('/')[1] as 'de' | 'en' | undefined;
|
|
||||||
const intlLocale = useLocale() as 'de' | 'en';
|
|
||||||
const locale: 'de' | 'en' = urlLocale === 'de' || urlLocale === 'en' ? urlLocale : intlLocale;
|
|
||||||
const t = useTranslations('Curator');
|
|
||||||
|
|
||||||
const [specials, setSpecials] = useState<CuratorSpecial[]>([]);
|
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const fetchSpecials = async () => {
|
|
||||||
try {
|
|
||||||
setLoading(true);
|
|
||||||
const res = await fetch('/api/curator/specials', {
|
|
||||||
headers: getCuratorAuthHeaders(),
|
|
||||||
});
|
|
||||||
if (!res.ok) {
|
|
||||||
if (res.status === 403) {
|
|
||||||
setError(t('specialForbidden'));
|
|
||||||
} else {
|
|
||||||
setError('Failed to load specials');
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const data = await res.json();
|
|
||||||
setSpecials(data);
|
|
||||||
} catch (e) {
|
|
||||||
setError('Failed to load specials');
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
fetchSpecials();
|
|
||||||
}, [t]);
|
|
||||||
|
|
||||||
if (loading) {
|
|
||||||
return (
|
|
||||||
<div style={{ maxWidth: '960px', margin: '2rem auto', padding: '1rem' }}>
|
|
||||||
<p>{t('loading')}</p>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
return (
|
|
||||||
<div style={{ maxWidth: '960px', margin: '2rem auto', padding: '1rem' }}>
|
|
||||||
<p style={{ color: 'red' }}>{error}</p>
|
|
||||||
<Link
|
|
||||||
href="/curator"
|
|
||||||
style={{
|
|
||||||
display: 'inline-block',
|
|
||||||
marginTop: '1rem',
|
|
||||||
padding: '0.5rem 1rem',
|
|
||||||
background: '#6b7280',
|
|
||||||
color: 'white',
|
|
||||||
textDecoration: 'none',
|
|
||||||
borderRadius: '0.375rem',
|
|
||||||
fontSize: '0.9rem',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{t('backToDashboard') || 'Back to Dashboard'}
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div style={{ maxWidth: '960px', margin: '2rem auto', padding: '1rem' }}>
|
|
||||||
<header style={{ marginBottom: '2rem' }}>
|
|
||||||
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
|
||||||
<h1 style={{ fontSize: '1.75rem', marginBottom: '0.25rem' }}>
|
|
||||||
{t('curateSpecialsTitle') || 'Curate Specials'}
|
|
||||||
</h1>
|
|
||||||
<Link
|
|
||||||
href="/curator"
|
|
||||||
style={{
|
|
||||||
padding: '0.5rem 1rem',
|
|
||||||
background: '#6b7280',
|
|
||||||
color: 'white',
|
|
||||||
textDecoration: 'none',
|
|
||||||
borderRadius: '0.375rem',
|
|
||||||
fontSize: '0.9rem',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{t('backToDashboard') || 'Back to Dashboard'}
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
{specials.length === 0 ? (
|
|
||||||
<div style={{ padding: '2rem', textAlign: 'center', color: '#666' }}>
|
|
||||||
<p>{t('noSpecialsAssigned') || 'No specials assigned to you.'}</p>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div style={{ display: 'flex', flexDirection: 'column', gap: '1rem' }}>
|
|
||||||
{specials.map((special) => (
|
|
||||||
<Link
|
|
||||||
key={special.id}
|
|
||||||
href={`/curator/specials/${special.id}`}
|
|
||||||
style={{
|
|
||||||
display: 'block',
|
|
||||||
padding: '1.5rem',
|
|
||||||
background: '#f9fafb',
|
|
||||||
border: '1px solid #e5e7eb',
|
|
||||||
borderRadius: '0.5rem',
|
|
||||||
textDecoration: 'none',
|
|
||||||
color: 'inherit',
|
|
||||||
transition: 'all 0.2s',
|
|
||||||
}}
|
|
||||||
onMouseEnter={(e) => {
|
|
||||||
e.currentTarget.style.background = '#f3f4f6';
|
|
||||||
e.currentTarget.style.borderColor = '#d1d5db';
|
|
||||||
}}
|
|
||||||
onMouseLeave={(e) => {
|
|
||||||
e.currentTarget.style.background = '#f9fafb';
|
|
||||||
e.currentTarget.style.borderColor = '#e5e7eb';
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
|
|
||||||
<div>
|
|
||||||
<h2 style={{ fontSize: '1.25rem', marginBottom: '0.5rem', color: '#111827' }}>
|
|
||||||
{getLocalizedValue(special.name, locale)}
|
|
||||||
</h2>
|
|
||||||
<p style={{ fontSize: '0.875rem', color: '#6b7280' }}>
|
|
||||||
{special.songCount} {special.songCount === 1 ? 'song' : 'songs'}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<div style={{ fontSize: '1.5rem', color: '#10b981' }}>→</div>
|
|
||||||
</div>
|
|
||||||
</Link>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,177 +0,0 @@
|
|||||||
'use client';
|
|
||||||
|
|
||||||
export const dynamic = 'force-dynamic';
|
|
||||||
|
|
||||||
import { useEffect, useState } from 'react';
|
|
||||||
import { useParams, useRouter, usePathname } from 'next/navigation';
|
|
||||||
import { useLocale, useTranslations } from 'next-intl';
|
|
||||||
import CurateSpecialEditor, { CurateSpecial } from '@/components/CurateSpecialEditor';
|
|
||||||
import { getCuratorAuthHeaders } from '@/lib/curatorAuth';
|
|
||||||
import HelpTooltip from '@/components/HelpTooltip';
|
|
||||||
|
|
||||||
export default function CuratorSpecialEditorPage() {
|
|
||||||
const params = useParams();
|
|
||||||
const router = useRouter();
|
|
||||||
const pathname = usePathname();
|
|
||||||
const urlLocale = pathname?.split('/')[1] as 'de' | 'en' | undefined;
|
|
||||||
const intlLocale = useLocale() as 'de' | 'en';
|
|
||||||
const locale: 'de' | 'en' = urlLocale === 'de' || urlLocale === 'en' ? urlLocale : intlLocale;
|
|
||||||
const t = useTranslations('Curator');
|
|
||||||
const tHelp = useTranslations('CuratorHelp');
|
|
||||||
|
|
||||||
const specialId = params?.id as string;
|
|
||||||
|
|
||||||
const [special, setSpecial] = useState<CurateSpecial | null>(null);
|
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
|
|
||||||
const fetchSpecial = async (showLoading = true) => {
|
|
||||||
try {
|
|
||||||
if (showLoading) {
|
|
||||||
setLoading(true);
|
|
||||||
}
|
|
||||||
const res = await fetch(`/api/curator/specials/${specialId}`, {
|
|
||||||
headers: getCuratorAuthHeaders(),
|
|
||||||
});
|
|
||||||
if (res.status === 403) {
|
|
||||||
setError(t('specialForbidden'));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!res.ok) {
|
|
||||||
setError('Failed to load special');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const data = await res.json();
|
|
||||||
setSpecial(data);
|
|
||||||
} catch (e) {
|
|
||||||
setError('Failed to load special');
|
|
||||||
} finally {
|
|
||||||
if (showLoading) {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (specialId) {
|
|
||||||
fetchSpecial(true);
|
|
||||||
}
|
|
||||||
}, [specialId, t]);
|
|
||||||
|
|
||||||
const handleSaveStartTime = async (songId: number, startTime: number) => {
|
|
||||||
const res = await fetch(`/api/curator/specials/${specialId}/songs`, {
|
|
||||||
method: 'PUT',
|
|
||||||
headers: {
|
|
||||||
...getCuratorAuthHeaders(),
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({ songId, startTime }),
|
|
||||||
});
|
|
||||||
if (res.status === 403) {
|
|
||||||
setError(t('specialForbidden'));
|
|
||||||
} else if (!res.ok) {
|
|
||||||
setError('Failed to save changes');
|
|
||||||
} else {
|
|
||||||
// Reload special data to update the start time in the song list
|
|
||||||
await fetchSpecial(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (loading) {
|
|
||||||
return (
|
|
||||||
<div style={{ padding: '2rem', textAlign: 'center' }}>
|
|
||||||
<p>{t('loadingData')}</p>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
return (
|
|
||||||
<div style={{ padding: '2rem', textAlign: 'center' }}>
|
|
||||||
<p>{error}</p>
|
|
||||||
<button
|
|
||||||
onClick={() => router.push(`/${locale}/curator`)}
|
|
||||||
style={{
|
|
||||||
marginTop: '1rem',
|
|
||||||
padding: '0.5rem 1rem',
|
|
||||||
borderRadius: '0.5rem',
|
|
||||||
border: 'none',
|
|
||||||
background: '#e5e7eb',
|
|
||||||
cursor: 'pointer',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{t('backToDashboard')}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!special) {
|
|
||||||
return (
|
|
||||||
<div style={{ padding: '2rem', textAlign: 'center' }}>
|
|
||||||
<p>{t('specialNotFound')}</p>
|
|
||||||
<button
|
|
||||||
onClick={() => router.push(`/${locale}/curator`)}
|
|
||||||
style={{
|
|
||||||
marginTop: '1rem',
|
|
||||||
padding: '0.5rem 1rem',
|
|
||||||
borderRadius: '0.5rem',
|
|
||||||
border: 'none',
|
|
||||||
background: '#e5e7eb',
|
|
||||||
cursor: 'pointer',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{t('backToDashboard')}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div style={{ padding: '2rem', maxWidth: '1200px', margin: '0 auto' }}>
|
|
||||||
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center', marginBottom: '1rem' }}>
|
|
||||||
<div style={{ display: 'flex', alignItems: 'center', gap: '0.5rem' }}>
|
|
||||||
<h1 style={{ fontSize: '1.75rem', fontWeight: 'bold' }}>
|
|
||||||
{t('curateSpecialHeaderPrefix')}
|
|
||||||
</h1>
|
|
||||||
<HelpTooltip
|
|
||||||
shortText={tHelp('tooltipCurateSpecialEditorShort')}
|
|
||||||
longText={tHelp('tooltipCurateSpecialEditorLong')}
|
|
||||||
position="bottom"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={() => router.push(`/${locale}/curator/specials`)}
|
|
||||||
style={{
|
|
||||||
padding: '0.5rem 1rem',
|
|
||||||
background: '#e5e7eb',
|
|
||||||
borderRadius: '0.5rem',
|
|
||||||
border: 'none',
|
|
||||||
cursor: 'pointer',
|
|
||||||
fontSize: '0.9rem',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{t('backToCuratorSpecials')}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<CurateSpecialEditor
|
|
||||||
special={special}
|
|
||||||
locale={locale}
|
|
||||||
onBack={() => router.push(`/${locale}/curator/specials`)}
|
|
||||||
onSaveStartTime={handleSaveStartTime}
|
|
||||||
backLabel={t('backToCuratorSpecials')}
|
|
||||||
headerPrefix={t('curateSpecialHeaderPrefix')}
|
|
||||||
noSongsHint={t('curateSpecialNoSongs')}
|
|
||||||
noSongsSubHint={t('curateSpecialNoSongsSub')}
|
|
||||||
instructionsText={t('curateSpecialInstructions')}
|
|
||||||
savingLabel={t('saving')}
|
|
||||||
saveChangesLabel={t('saveChanges')}
|
|
||||||
savedLabel={t('saved')}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
'use client';
|
|
||||||
|
|
||||||
// Root /curator/specials route without locale:
|
|
||||||
// redirect users to the default English locale version.
|
|
||||||
|
|
||||||
import { redirect } from 'next/navigation';
|
|
||||||
|
|
||||||
export default function CuratorSpecialsPage() {
|
|
||||||
redirect('/en/curator/specials');
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,141 +0,0 @@
|
|||||||
.page {
|
|
||||||
--background: #fafafa;
|
|
||||||
--foreground: #fff;
|
|
||||||
|
|
||||||
--text-primary: #000;
|
|
||||||
--text-secondary: #666;
|
|
||||||
|
|
||||||
--button-primary-hover: #383838;
|
|
||||||
--button-secondary-hover: #f2f2f2;
|
|
||||||
--button-secondary-border: #ebebeb;
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
min-height: 100vh;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
font-family: var(--font-geist-sans);
|
|
||||||
background-color: var(--background);
|
|
||||||
}
|
|
||||||
|
|
||||||
.main {
|
|
||||||
display: flex;
|
|
||||||
min-height: 100vh;
|
|
||||||
width: 100%;
|
|
||||||
max-width: 800px;
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: flex-start;
|
|
||||||
justify-content: space-between;
|
|
||||||
background-color: var(--foreground);
|
|
||||||
padding: 120px 60px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.intro {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: flex-start;
|
|
||||||
text-align: left;
|
|
||||||
gap: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.intro h1 {
|
|
||||||
max-width: 320px;
|
|
||||||
font-size: 40px;
|
|
||||||
font-weight: 600;
|
|
||||||
line-height: 48px;
|
|
||||||
letter-spacing: -2.4px;
|
|
||||||
text-wrap: balance;
|
|
||||||
color: var(--text-primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.intro p {
|
|
||||||
max-width: 440px;
|
|
||||||
font-size: 18px;
|
|
||||||
line-height: 32px;
|
|
||||||
text-wrap: balance;
|
|
||||||
color: var(--text-secondary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.intro a {
|
|
||||||
font-weight: 500;
|
|
||||||
color: var(--text-primary);
|
|
||||||
}
|
|
||||||
|
|
||||||
.ctas {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
width: 100%;
|
|
||||||
max-width: 440px;
|
|
||||||
gap: 16px;
|
|
||||||
font-size: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.ctas a {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
height: 40px;
|
|
||||||
padding: 0 16px;
|
|
||||||
border-radius: 128px;
|
|
||||||
border: 1px solid transparent;
|
|
||||||
transition: 0.2s;
|
|
||||||
cursor: pointer;
|
|
||||||
width: fit-content;
|
|
||||||
font-weight: 500;
|
|
||||||
}
|
|
||||||
|
|
||||||
a.primary {
|
|
||||||
background: var(--text-primary);
|
|
||||||
color: var(--background);
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
a.secondary {
|
|
||||||
border-color: var(--button-secondary-border);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Enable hover only on non-touch devices */
|
|
||||||
@media (hover: hover) and (pointer: fine) {
|
|
||||||
a.primary:hover {
|
|
||||||
background: var(--button-primary-hover);
|
|
||||||
border-color: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
a.secondary:hover {
|
|
||||||
background: var(--button-secondary-hover);
|
|
||||||
border-color: transparent;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 600px) {
|
|
||||||
.main {
|
|
||||||
padding: 48px 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.intro {
|
|
||||||
gap: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.intro h1 {
|
|
||||||
font-size: 32px;
|
|
||||||
line-height: 40px;
|
|
||||||
letter-spacing: -1.92px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (prefers-color-scheme: dark) {
|
|
||||||
.logo {
|
|
||||||
filter: invert();
|
|
||||||
}
|
|
||||||
|
|
||||||
.page {
|
|
||||||
--background: #000;
|
|
||||||
--foreground: #000;
|
|
||||||
|
|
||||||
--text-primary: #ededed;
|
|
||||||
--text-secondary: #999;
|
|
||||||
|
|
||||||
--button-primary-hover: #ccc;
|
|
||||||
--button-secondary-hover: #1a1a1a;
|
|
||||||
--button-secondary-border: #1a1a1a;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "hoerdle",
|
"name": "hoerdle",
|
||||||
"version": "0.1.6.25",
|
"version": "0.1.6.26",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev",
|
"dev": "next dev",
|
||||||
|
|||||||
Reference in New Issue
Block a user