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>
42 lines
1.6 KiB
TypeScript
42 lines
1.6 KiB
TypeScript
import { NextRequest, NextResponse } from 'next/server'
|
|
import { getMobileUserId } from '@/lib/mobile-auth'
|
|
import { paragraphRefactorService } from '@/lib/ai/services/paragraph-refactor.service'
|
|
import { checkEntitlementOrThrow, QuotaExceededError, incrementUsageAsync } from '@/lib/entitlements'
|
|
|
|
const MODE_MAP: Record<string, 'clarify' | 'shorten' | 'improveStyle' | 'fix_grammar'> = {
|
|
improve: 'improveStyle',
|
|
shorten: 'shorten',
|
|
clarify: 'clarify',
|
|
fix_grammar: 'fix_grammar',
|
|
}
|
|
|
|
export async function POST(req: NextRequest) {
|
|
const userId = getMobileUserId(req)
|
|
if (!userId) return NextResponse.json({ error: 'Non autorisé' }, { status: 401 })
|
|
|
|
const { text, mode = 'improve' } = await req.json().catch(() => ({}))
|
|
if (!text?.trim()) return NextResponse.json({ error: 'Texte requis' }, { status: 400 })
|
|
|
|
const refactorMode = MODE_MAP[mode]
|
|
if (!refactorMode) {
|
|
return NextResponse.json({ error: 'Mode invalide. Valeurs: improve, shorten, clarify, fix_grammar' }, { status: 400 })
|
|
}
|
|
|
|
const validation = paragraphRefactorService.validateWordCount(text)
|
|
if (!validation.valid) return NextResponse.json({ error: validation.error }, { status: 400 })
|
|
|
|
try {
|
|
await checkEntitlementOrThrow(userId, 'reformulate')
|
|
} catch (err) {
|
|
if (err instanceof QuotaExceededError) {
|
|
return NextResponse.json({ error: 'quota_exceeded' }, { status: 402 })
|
|
}
|
|
throw err
|
|
}
|
|
|
|
const result = await paragraphRefactorService.refactor(text, refactorMode, 'markdown', undefined)
|
|
incrementUsageAsync(userId, 'reformulate')
|
|
|
|
return NextResponse.json({ improved: result.refactored, original: result.original })
|
|
}
|