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>
36 lines
865 B
TypeScript
36 lines
865 B
TypeScript
import * as SecureStore from 'expo-secure-store'
|
|
|
|
const TOKEN_KEY = 'memento_token'
|
|
|
|
export async function getToken(): Promise<string | null> {
|
|
try {
|
|
return await SecureStore.getItemAsync(TOKEN_KEY)
|
|
} catch {
|
|
return null
|
|
}
|
|
}
|
|
|
|
export async function setToken(token: string): Promise<void> {
|
|
try {
|
|
await SecureStore.setItemAsync(TOKEN_KEY, token)
|
|
} catch (e) {
|
|
console.warn('[SecureStore] setToken failed:', e)
|
|
}
|
|
}
|
|
|
|
export async function clearToken(): Promise<void> {
|
|
await SecureStore.deleteItemAsync(TOKEN_KEY)
|
|
}
|
|
|
|
export async function apiFetch(url: string, options: RequestInit = {}): Promise<Response> {
|
|
const token = await getToken()
|
|
return fetch(url, {
|
|
...options,
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
...(token ? { Authorization: `Bearer ${token}` } : {}),
|
|
...options.headers,
|
|
},
|
|
})
|
|
}
|