Files
Momento/memento-note/app/api/mobile/notes/route.ts
Antigravity 0fa8978395
Some checks failed
CI / Lint, Unit Tests & Build (push) Failing after 1m32s
CI / Deploy production (on server) (push) Has been skipped
feat: mobile app complet + flashcards fixes + drag handle améliorations
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>
2026-05-29 18:49:40 +00:00

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, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;')
}