# Chapitre 1 : useApi Le hook `useApi` simplifie les appels API avec gestion automatique de l'authentification JWT, du rafraîchissement de token et de la gestion des erreurs. ## Configuration L'API est configurée dans `appConfig.js` : ```javascript // src/appConfig.js export const config = { api: { prefixUrl: import.meta.env.VITE_API_URL, timeout: 30000, debug: import.meta.env.DEV, paths: { login: "login", logout: "logout", refresh: "refresh" } } }; ``` ## Structure retournée ```javascript import { useApi } from '@cap-rel/smartcommon'; const api = useApi(); ``` ^ Méthode ^ Description ^ | api.login(body) | Connexion utilisateur | | api.logout() | Déconnexion | | api.public | Instance pour requêtes publiques | | api.private | Instance pour requêtes authentifiées | ## Authentification ### Login ```javascript import { useApi, useNavigation } from '@cap-rel/smartcommon'; function LoginPage() { const api = useApi(); const navigate = useNavigation(); const handleLogin = async (email, password) => { try { const user = await api.login({ login: email, password: password, rememberMe: true }); console.log('Connecté :', user); navigate('/'); } catch (error) { console.error('Erreur de connexion:', error); alert('Identifiants incorrects'); } }; return (
); } ``` ### Logout ```javascript function LogoutButton() { const api = useApi(); const navigate = useNavigation(); const handleLogout = async () => { await api.logout(); navigate('/login'); }; return ( ); } ``` ## Requêtes publiques Pour les endpoints sans authentification : ```javascript const api = useApi(); // GET const info = await api.public.get('public/info').json(); // POST const result = await api.public.post('public/contact', { json: { name: 'Jean', message: 'Bonjour' } }).json(); ``` ## Requêtes authentifiées Pour les endpoints protégés par JWT : ```javascript const api = useApi(); // GET - Liste const items = await api.private.get('items').json(); // GET - Détail const item = await api.private.get(`items/${id}`).json(); // POST - Création const newItem = await api.private.post('items', { json: { label: 'Nouveau produit', price: 29.99 } }).json(); // PUT - Modification const updated = await api.private.put(`items/${id}`, { json: { label: 'Produit modifié' } }).json(); // DELETE - Suppression await api.private.delete(`items/${id}`); ``` ## Exemple complet : CRUD ```javascript import { useEffect } from 'react'; import { Page, Block, List, ListItem, Button, Spinner } from '@cap-rel/smartcommon'; import { useApi, useStates, useNavigation } from '@cap-rel/smartcommon'; export const ProductsPage = () => { const api = useApi(); const navigate = useNavigation(); const st = useStates({ initialStates: { products: [], loading: true, error: null } }); // Charger les produits useEffect(() => { loadProducts(); }, []); const loadProducts = async () => { st.set('loading', true); st.set('error', null); try { const data = await api.private.get('products').json(); st.set('products', data.products); } catch (err) { st.set('error', err.message); } finally { st.set('loading', false); } }; // Supprimer un produit const deleteProduct = async (id) => { if (!confirm('Supprimer ce produit ?')) return; try { await api.private.delete(`products/${id}`); // Retirer de la liste locale st.set('products', st.get('products').filter(p => p.id !== id)); } catch (err) { alert('Erreur : ' + err.message); } }; if (st.get('loading')) { returnErreur : {st.get('error')}