Files
Momento/memento-mobile/app/(tabs)/notebooks.tsx
Antigravity aeedb2846f
Some checks failed
CI / Lint, Unit Tests & Build (push) Failing after 1m21s
CI / Deploy production (on server) (push) Has been skipped
feat: App mobile Expo + API mobile dédiée
memento-mobile/ (Expo + React Native + expo-router):
- Auth: login email/password → Bearer token (expo-secure-store)
- Layout: guard auth → redirect /(auth)/login ou /(tabs)/home
- Tabs: Accueil, Carnets, Recherche, Profil
- Screens: login, home (recent notes + quick actions), notebooks list,
  note viewer (WebView HTML), search (texte), notebook detail, profile
- Design: tokens brand-accent (#A47148), ink, concrete, paper, border
- lib/config.ts: API_URL dev/prod configurable
- lib/api.ts: apiFetch avec Bearer token automatique
- lib/store.ts: Zustand auth store (login/logout/restore)

memento-note/ (API mobile dédiée):
- lib/mobile-auth.ts: createMobileToken / verifyMobileToken (HMAC-SHA256, 90j)
- POST /api/mobile/auth/login: email+password → token + user
- GET /api/mobile/auth/me: valider token, retourner profil
- GET /api/mobile/notebooks: liste carnets avec nb notes
- GET /api/mobile/notes: notes récentes (filtre par carnet optionnel)
- GET /api/mobile/notes/[id]: contenu complet d'une note
- GET /api/mobile/search: recherche fulltext titre+contenu

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
2026-05-29 15:53:13 +00:00

79 lines
2.4 KiB
TypeScript

import { useEffect, useState } from 'react'
import {
View, Text, FlatList, TouchableOpacity,
ActivityIndicator, RefreshControl,
} from 'react-native'
import { SafeAreaView } from 'react-native-safe-area-context'
import { useRouter } from 'expo-router'
import { ChevronRight } from 'lucide-react-native'
import { apiFetch } from '@/lib/api'
import { ENDPOINTS } from '@/lib/config'
interface Notebook {
id: string
name: string
icon: string | null
_count: { notes: number }
}
export default function NotebooksScreen() {
const [notebooks, setNotebooks] = useState<Notebook[]>([])
const [loading, setLoading] = useState(true)
const [refreshing, setRefreshing] = useState(false)
const router = useRouter()
const load = async () => {
try {
const res = await apiFetch(ENDPOINTS.notebooks)
if (res.ok) {
const data = await res.json()
setNotebooks(data.notebooks ?? [])
}
} finally {
setLoading(false)
setRefreshing(false)
}
}
useEffect(() => { load() }, [])
if (loading) {
return (
<SafeAreaView className="flex-1 bg-paper items-center justify-center">
<ActivityIndicator color="#A47148" />
</SafeAreaView>
)
}
return (
<SafeAreaView className="flex-1 bg-paper">
<View className="px-5 pt-4 pb-2">
<Text className="text-2xl font-serif italic text-ink">Carnets</Text>
</View>
<FlatList
data={notebooks}
keyExtractor={(item) => item.id}
contentContainerClassName="px-5 pb-8"
refreshControl={<RefreshControl refreshing={refreshing} onRefresh={() => { setRefreshing(true); load() }} tintColor="#A47148" />}
renderItem={({ item }) => (
<TouchableOpacity
onPress={() => router.push(`/notebook/${item.id}`)}
className="flex-row items-center bg-white border border-border rounded-2xl p-4 mb-2.5 active:opacity-70"
>
<Text className="text-2xl mr-3">{item.icon ?? '📓'}</Text>
<View className="flex-1">
<Text className="text-[15px] font-semibold text-ink">{item.name}</Text>
<Text className="text-[12px] text-concrete">{item._count?.notes ?? 0} notes</Text>
</View>
<ChevronRight size={16} color="#8A8A82" />
</TouchableOpacity>
)}
ListEmptyComponent={
<Text className="text-concrete text-center mt-12">Aucun carnet.</Text>
}
/>
</SafeAreaView>
)
}