Mobile app: - Révision flashcards : liste decks, session flip-card SM-2, couleurs harmonisées web - Génération flashcards depuis note (FlashcardSheet + route /api/mobile/flashcards/generate) - Audio Whisper : hook useAudioRecorder reécrit, MicButton avec erreurs - IA : AISheet (améliorer/clarifier/résumer), TitleSheet (titre automatique) - Suppression note (soft delete + confirmation Alert) - Note du jour : titre lisible + HTML (plus JSON TipTap brut) - Parser TipTap→HTML côté mobile (tipTapToHtml) - Icône 🎓 dans header note → génération flashcards - Endpoint flashcardGenerate dans config.ts Web fixes: - Bug flashcards groupées par carnet → deck par note (migration + schema) - Bug filtre 'cartes dues' ignoré (suppression fallback buildSessionQueue) - Suppression UI création deck manuelle (inutile) - Fix setViewType is not defined dans home-client.tsx Drag handle menu: - Fix : clearNodes() avant transformation (heading→liste/code/citation) - Ajout : option 'Texte' (paragraphe) dans Transformer en - Ajout : Monter / Descendre le bloc - Ajout : Copier le contenu du bloc - Fix : sous-menu hover stable (délai 200ms) - Fix : Supprimer en rouge via classe --danger (plus :first-child) - i18n : nouvelles clés dans 15 locales Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
84 lines
2.4 KiB
TypeScript
84 lines
2.4 KiB
TypeScript
import { NextRequest, NextResponse } from 'next/server'
|
|
import prisma from '@/lib/prisma'
|
|
import { getMobileUserId } from '@/lib/mobile-auth'
|
|
|
|
export async function GET(req: NextRequest) {
|
|
const userId = getMobileUserId(req)
|
|
if (!userId) return NextResponse.json({ error: 'Non autorisé' }, { status: 401 })
|
|
|
|
const { searchParams } = new URL(req.url)
|
|
const notebookId = searchParams.get('notebookId')
|
|
|
|
const notes = await prisma.note.findMany({
|
|
where: {
|
|
userId,
|
|
trashedAt: null,
|
|
...(notebookId ? { notebookId } : {}),
|
|
},
|
|
select: {
|
|
id: true,
|
|
title: true,
|
|
updatedAt: true,
|
|
color: true,
|
|
notebook: { select: { name: true } },
|
|
},
|
|
orderBy: { updatedAt: 'desc' },
|
|
take: 50,
|
|
})
|
|
|
|
const notebookName = notebookId
|
|
? (await prisma.notebook.findUnique({ where: { id: notebookId }, select: { name: true } }))?.name
|
|
: undefined
|
|
|
|
return NextResponse.json({
|
|
notes: notes.map((n) => ({
|
|
id: n.id,
|
|
title: n.title,
|
|
updatedAt: n.updatedAt,
|
|
color: n.color,
|
|
notebookName: n.notebook?.name,
|
|
})),
|
|
...(notebookName ? { notebookName } : {}),
|
|
})
|
|
}
|
|
|
|
export async function POST(req: NextRequest) {
|
|
const userId = getMobileUserId(req)
|
|
if (!userId) return NextResponse.json({ error: 'Non autorisé' }, { status: 401 })
|
|
|
|
const { title, content, notebookId } = await req.json().catch(() => ({}))
|
|
if (!title?.trim()) return NextResponse.json({ error: 'Titre requis' }, { status: 400 })
|
|
|
|
// Convertir le texte brut en HTML TipTap simple si nécessaire
|
|
const htmlContent = buildHtmlContent(content ?? '')
|
|
|
|
const note = await prisma.note.create({
|
|
data: {
|
|
userId,
|
|
title: title.trim(),
|
|
content: htmlContent,
|
|
type: 'richtext',
|
|
...(notebookId ? { notebookId } : {}),
|
|
},
|
|
select: { id: true, title: true, updatedAt: true },
|
|
})
|
|
|
|
return NextResponse.json({ note }, { status: 201 })
|
|
}
|
|
|
|
/** Convertit du texte brut multiligne en paragraphes HTML TipTap */
|
|
function buildHtmlContent(text: string): string {
|
|
if (!text.trim()) return '<p></p>'
|
|
// Si déjà du HTML, retourner tel quel
|
|
if (text.trimStart().startsWith('<')) return text
|
|
return text
|
|
.split('\n')
|
|
.map((line) => `<p>${line.trim() ? escapeHtml(line) : ''}</p>`)
|
|
.join('')
|
|
}
|
|
|
|
function escapeHtml(s: string) {
|
|
return s.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')
|
|
}
|
|
|