feat: implement Daily management features and update UI

- Marked tasks as completed in TODO for Daily management service, data model, and interactive checkboxes.
- Added a new link to the Daily page in the Header component for easy navigation.
- Introduced DailyCheckbox model in Prisma schema and corresponding TypeScript interfaces for better data handling.
- Updated database schema to include daily checkboxes, enhancing task management capabilities.
This commit is contained in:
Julien Froidefond
2025-09-15 18:04:46 +02:00
parent 74ef79eb70
commit cf2e360ce9
14 changed files with 1423 additions and 10 deletions

View File

@@ -0,0 +1,63 @@
import { NextResponse } from 'next/server';
import { dailyService } from '@/services/daily';
/**
* API route pour mettre à jour une checkbox
*/
export async function PATCH(
request: Request,
{ params }: { params: Promise<{ id: string }> }
) {
try {
const body = await request.json();
const { id: checkboxId } = await params;
const checkbox = await dailyService.updateCheckbox(checkboxId, body);
return NextResponse.json(checkbox);
} catch (error) {
console.error('Erreur lors de la mise à jour de la checkbox:', error);
if (error instanceof Error && error.message.includes('Record to update not found')) {
return NextResponse.json(
{ error: 'Checkbox non trouvée' },
{ status: 404 }
);
}
return NextResponse.json(
{ error: 'Erreur interne du serveur' },
{ status: 500 }
);
}
}
/**
* API route pour supprimer une checkbox
*/
export async function DELETE(
request: Request,
{ params }: { params: Promise<{ id: string }> }
) {
try {
const { id: checkboxId } = await params;
await dailyService.deleteCheckbox(checkboxId);
return NextResponse.json({ success: true });
} catch (error) {
console.error('Erreur lors de la suppression de la checkbox:', error);
if (error instanceof Error && error.message.includes('Checkbox non trouvée')) {
return NextResponse.json(
{ error: 'Checkbox non trouvée' },
{ status: 404 }
);
}
return NextResponse.json(
{ error: 'Erreur interne du serveur' },
{ status: 500 }
);
}
}

View File

@@ -0,0 +1,38 @@
import { NextResponse } from 'next/server';
import { dailyService } from '@/services/daily';
/**
* API route pour réordonner les checkboxes d'une date
*/
export async function POST(request: Request) {
try {
const body = await request.json();
// Validation des données
if (!body.date || !Array.isArray(body.checkboxIds)) {
return NextResponse.json(
{ error: 'date et checkboxIds (array) sont requis' },
{ status: 400 }
);
}
const date = new Date(body.date);
if (isNaN(date.getTime())) {
return NextResponse.json(
{ error: 'Format de date invalide. Utilisez YYYY-MM-DD' },
{ status: 400 }
);
}
await dailyService.reorderCheckboxes(date, body.checkboxIds);
return NextResponse.json({ success: true });
} catch (error) {
console.error('Erreur lors du réordonnancement:', error);
return NextResponse.json(
{ error: 'Erreur interne du serveur' },
{ status: 500 }
);
}
}

View File

@@ -0,0 +1,96 @@
import { NextResponse } from 'next/server';
import { dailyService } from '@/services/daily';
/**
* API route pour récupérer la vue daily (hier + aujourd'hui)
*/
export async function GET(request: Request) {
try {
const { searchParams } = new URL(request.url);
const action = searchParams.get('action');
const date = searchParams.get('date');
if (action === 'history') {
// Récupérer l'historique
const limit = parseInt(searchParams.get('limit') || '30');
const history = await dailyService.getCheckboxHistory(limit);
return NextResponse.json(history);
}
if (action === 'search') {
// Recherche dans les checkboxes
const query = searchParams.get('q') || '';
const limit = parseInt(searchParams.get('limit') || '20');
if (!query.trim()) {
return NextResponse.json({ error: 'Query parameter required' }, { status: 400 });
}
const checkboxes = await dailyService.searchCheckboxes(query, limit);
return NextResponse.json(checkboxes);
}
// Vue daily pour une date donnée (ou aujourd'hui par défaut)
const targetDate = date ? new Date(date) : new Date();
if (date && isNaN(targetDate.getTime())) {
return NextResponse.json(
{ error: 'Format de date invalide. Utilisez YYYY-MM-DD' },
{ status: 400 }
);
}
const dailyView = await dailyService.getDailyView(targetDate);
return NextResponse.json(dailyView);
} catch (error) {
console.error('Erreur lors de la récupération du daily:', error);
return NextResponse.json(
{ error: 'Erreur interne du serveur' },
{ status: 500 }
);
}
}
/**
* API route pour ajouter une checkbox
*/
export async function POST(request: Request) {
try {
const body = await request.json();
// Validation des données
if (!body.date || !body.text) {
return NextResponse.json(
{ error: 'Date et text sont requis' },
{ status: 400 }
);
}
const date = new Date(body.date);
if (isNaN(date.getTime())) {
return NextResponse.json(
{ error: 'Format de date invalide. Utilisez YYYY-MM-DD' },
{ status: 400 }
);
}
const checkbox = await dailyService.addCheckbox({
date,
text: body.text,
taskId: body.taskId,
order: body.order,
isChecked: body.isChecked
});
return NextResponse.json(checkbox, { status: 201 });
} catch (error) {
console.error('Erreur lors de l\'ajout de la checkbox:', error);
return NextResponse.json(
{ error: 'Erreur interne du serveur' },
{ status: 500 }
);
}
}