You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Web-Dev-For-Beginners/7-bank-project/4-state-management/translations/README.fr.md

17 KiB

Créer une application bancaire Partie 4: Concepts de gestion d'état

Quiz préalable

Quiz préalable

Introduction

Au fur et à mesure quune application Web se développe, il devient difficile de suivre tous les flux de données. Quel code obtient les données, quelle page les consomme, où et quand doit-il être mis à jour... il est facile de se retrouver avec un code désordonné et difficile à maintenir. Cela est particulièrement vrai lorsque vous devez partager des données entre différentes pages de votre application, par exemple des données utilisateur. Le concept de gestion de létat a toujours existé dans toutes sortes de programmes, mais comme les applications Web ne cessent de croître en complexité, cest maintenant un point clé à prendre en compte pendant le développement.

Dans cette dernière partie, nous examinerons lapplication que nous avons créée pour repenser la façon dont létat est géré, permettant la prise en charge de lactualisation du navigateur à tout moment et la persistance des données entre les sessions utilisateur.

Prérequis

Vous devez avoir terminé la récupération des données de lapplication web pour cette leçon. Vous devez également installer Node.js et exécuter lAPI du serveur localement afin que vous puissiez gérer les données du compte.

Vous pouvez tester que le serveur fonctionne correctement en exécutant cette commande dans un terminal:

curl http://localhost:5000/api
# -> doit renvoyer "Bank API v1.0.0" comme résultat

Repenser la gestion des états

Dans la leçon précédente, nous avons introduit un concept basique détat dans notre application avec la variable globale account qui contient les données bancaires de lutilisateur actuellement connecté. Cependant, notre mise en œuvre actuelle présente certains défauts. Essayez dactualiser la page lorsque vous êtes sur le tableau de bord. Que se passe-t-il?

Il y a 3 problèmes avec le code actuel:

  • Létat nest pas persistant, car une actualisation du navigateur vous ramène à la page de connexion.
  • Il existe plusieurs fonctions qui modifient létat. Au fur et à mesure que lapplication se développe, il peut être difficile de suivre les modifications et il est facile doublier den mettre à jour une.
  • Létat nest pas nettoyé, donc lorsque vous cliquez sur Logout les données du compte sont toujours là même si vous êtes sur la page de connexion.

Nous pourrions mettre à jour notre code pour résoudre ces problèmes un par un, mais cela créerait plus de duplication de code et rendrait lapplication plus complexe et difficile à maintenir. Ou nous pourrions faire une pause de quelques minutes et repenser notre stratégie.

Quels problèmes essayons-nous vraiment de résoudre ici?

La gestion de létat consiste à trouver une bonne approche pour résoudre ces deux problèmes particuliers:

  • Comment rendre compréhensibles les flux de données dans une application?
  • Comment garder les données détat toujours synchronisées avec linterface utilisateur (et vice versa)?

Une fois que vous vous êtes occupé de ceux-ci, tous les autres problèmes que vous pourriez avoir peuvent être déjà résolus ou sont devenus plus faciles à résoudre. Il existe de nombreuses approches possibles pour résoudre ces problèmes, mais nous opterons pour une solution commune qui consiste à centraliser les données et les moyens de les modifier. Les flux de données se dérouleraient comme suit:

Schéma montrant les flux de données entre le code HTML, les actions de l’utilisateur et l’état

Nous ne couvrirons pas ici la partie où les données déclenchent automatiquement la mise à jour de la vue, car elle est liée à des concepts plus avancés de Programmation réactive. Cest un bon sujet de suivi si vous êtes prêt à plonger profondément.

Il existe de nombreuses bibliothèques avec différentes approches de la gestion des États, Redux étant une option populaire. Jetez un coup dœil aux concepts et aux modèles utilisés, car cest souvent un bon moyen dapprendre quels problèmes potentiels vous pouvez rencontrer dans les grandes applications Web et comment ils peuvent être résolus.

Tâche

Nous allons commencer par un peu de refactorisation. Remplacer la déclaration account:

let account = null;

Par:

let state = {
  account: null
};

Lidée est de centraliser toutes nos données dapplication dans un seul objet détat. Nous navons que le account pour linstant dans létat, donc cela ne change pas beaucoup, mais cela crée un chemin pour les évolutions.

Nous devons également mettre à jour les fonctions en lutilisant. Dans les fonctions register() et login(), remplacez account = ... par state.account = ...;

En haut de la fonction updateDashboard(), ajoutez cette ligne:

const account = state.account;

Ce refactoring en lui-même na pas apporté beaucoup daméliorations, mais lidée était de jeter les bases des prochains changements.

Suivre les modifications de données

Maintenant que nous avons mis en place lobjet state pour stocker nos données, létape suivante consiste à centraliser les mises à jour. Lobjectif est de faciliter le suivi des changements et de leur moment.

Pour éviter que des modifications soient apportées à lobjet state, il est également recommandé de le considérer comme immuable, ce qui signifie quil ne peut pas être modifié du tout. Cela signifie également que vous devez créer un nouvel objet détat si vous souhaitez y modifier quoi que ce soit. Ce faisant, vous créez une protection contre les effets secondaires potentiellement indésirables et ouvrez des possibilités de nouvelles fonctionnalités dans votre application, telles que la mise en œuvre de lannulation/rétablissement, tout en facilitant le débogage. Par exemple, vous pouvez consigner chaque modification apportée à létat et conserver un historique des modifications pour comprendre la source dun bogue.

En JavaScript, vous pouvez utiliser Object.freeze() pour créer une version immuable dun objet. Si vous essayez dapporter des modifications à un objet immuable, une exception sera déclenchée.

Connaissez-vous la différence entre un objet immuable peu profond et un objet immuable profond? Vous pouvez en apprendre plus sur ce sujet ici.

Tâche

Créons une nouvelle fonction updateState():

function updateState(property, newData) {
  state = Object.freeze({
    ...state,
    [property]: newData
  });
}

Dans cette fonction, nous créons un nouvel objet détat et copions les données de létat précédent à laide de lopérateur spread (...). Ensuite, nous remplaçons une propriété particulière de lobjet détat par les nouvelles données en utilisant la notation entre crochets [property] pour laffectation. Enfin, nous verrouillons lobjet pour empêcher les modifications en utilisant Object.freeze(). Nous navons que la propriété account stockée dans létat pour linstant, mais avec cette approche, vous pouvez ajouter autant de propriétés que nécessaire dans létat.

Nous allons également mettre à jour linitialisation state pour nous assurer que létat initial est également gelé:

let state = Object.freeze({
  account: null
});

Après cela, mettez à jour la fonction register en remplaçant laffectation state.account = result; par:

updateState('account', result);

Faites de même avec la fonction login, en remplaçant state.account = data; par:

updateState('account', data);

Nous allons maintenant saisir loccasion de résoudre le problème des données de compte qui ne sont pas effacées lorsque lutilisateur clique sur Logout.

Créez une nouvelle fonction logout():

function logout() {
  updateState('account', null);
  navigate('/login');
}

Dans updateDashboard(), remplacez la redirection return navigate('/login'); par return logout();

Essayez denregistrer un nouveau compte, de vous déconnecter et de vous reconnecter pour vérifier que tout fonctionne toujours correctement.

Conseil: vous pouvez jeter un coup dœil à tous les changements détat en ajoutant console.log(state) au bas de updateState() et en ouvrant la console dans les outils de développement de votre navigateur.

Conserver létat

La plupart des applications Web doivent conserver les données pour pouvoir fonctionner correctement. Toutes les données critiques sont généralement stockées dans une base de données et accessibles via une API de serveur, comme les données du compte utilisateur dans notre cas. Mais parfois, il est également intéressant de conserver certaines données sur lapplication cliente qui sexécute dans votre navigateur, pour une meilleure expérience utilisateur ou pour améliorer les performances de chargement.

Lorsque vous souhaitez conserver des données dans votre navigateur, vous devez vous poser quelques questions importantes:

  • Les données sont-elles sensibles? Vous devez éviter de stocker des données sensibles sur le client, telles que les mots de passe des utilisateurs.
  • Pendant combien de temps devez-vous conserver ces données? Prévoyez-vous daccéder à ces données uniquement pour la session en cours ou souhaitez-vous quelles soient stockées pour toujours?

Il existe plusieurs façons de stocker des informations dans une application web, en fonction de ce que vous souhaitez réaliser. Par exemple, vous pouvez utiliser les URL pour stocker une requête de recherche et la rendre partageable entre les utilisateurs. Vous pouvez également utiliser des cookies HTTP si les données doivent être partagées avec le serveur, comme les informations d'authentification.

Une autre option consiste à utiliser lune des nombreuses API de navigateur pour stocker des données. Deux dentre eux sont particulièrement intéressants:

  • localStorage: un Key/Value store permettant de conserver des données spécifiques au site web actuel sur différentes sessions. Les données qui y sont enregistrées nexpirent jamais.
  • sessionStorage: celui-ci fonctionne de la même manière que localStorage sauf que les données qui y sont stockées sont effacées à la fin de la session (lorsque le navigateur est fermé).

Notez que ces deux API autorisent uniquement le stockage de chaînes. Si vous souhaitez stocker des objets complexes, vous devrez les sérialiser au format JSON à laide de JSON.stringify().

Si vous souhaitez créer une application web qui ne fonctionne pas avec un serveur, il est également possible de créer une base de données sur le client à laide de lAPI IndexedDB. Celui-ci est réservé aux cas dutilisation avancés ou si vous avez besoin de stocker une quantité importante de données, car il est plus complexe à utiliser.

Tâche

Nous voulons que nos utilisateurs restent connectés jusquà ce quils cliquent explicitement sur le bouton Logout, nous utiliserons donc localStorage pour stocker les données du compte. Tout dabord, définissons une clé que nous utiliserons pour stocker nos données.

const storageKey = 'savedAccount';

Ajoutez ensuite cette ligne à la fin de la fonction updateState():

localStorage.setItem(storageKey, JSON.stringify(state.account));

Avec cela, les données du compte utilisateur seront conservées et toujours à jour car nous avons centralisé précédemment toutes nos mises à jour détat. Cest là que nous commençons à bénéficier de tous nos refactors précédents 🙂.

Au fur et à mesure que les données sont enregistrées, nous devons également nous occuper de les restaurer lorsque lapplication est chargée. Puisque nous allons commencer à avoir plus de code dinitialisation, il peut être judicieux de créer une nouvelle fonction init, qui inclut également notre code précédent au bas de app.js:

function init() {
  const savedAccount = localStorage.getItem(storageKey);
  if (savedAccount) {
    updateState('account', JSON.parse(savedAccount));
  }

  // Notre précédent code d'initialisation
  window.onpopstate = () => updateRoute();
  updateRoute();
}

init();

Ici, nous récupérons les données enregistrées, et sil y en a, nous mettons à jour létat en conséquence. Il est important de le faire avant de mettre à jour litinéraire, car il peut y avoir du code sappuyant sur létat lors de la mise à jour de la page.

Nous pouvons également faire de la page Dashboard notre page par défaut de lapplication, car nous conservons maintenant les données du compte. Si aucune donnée nest trouvée, le tableau de bord se charge de rediriger vers la page Login de toute façon. Dans updateRoute(), remplacez le secours return navigate('/login'); par return navigate('/dashboard');.

Maintenant, connectez-vous à lapplication et essayez dactualiser la page. Vous devez rester sur le tableau de bord. Avec cette mise à jour, nous avons résolu tous nos problèmes initiaux...

Actualiser les données

... Mais nous pourrions aussi en avoir créé un nouveau. Oups!

Accédez au tableau de bord à laide du compte test, puis exécutez cette commande sur un terminal pour créer une nouvelle transaction:

curl --request POST \
     --header "Content-Type: application/json" \
     --data "{ \"date\": \"2020-07-24\", \"object\": \"Bought book\", \"amount\": -20 }" \
     http://localhost:5000/api/accounts/test/transactions

Essayez dactualiser la page du tableau de bord dans le navigateur maintenant. Que se passe-t-il? Voyez-vous la nouvelle transaction?

Létat est conservé indéfiniment grâce au localStorage, mais cela signifie également quil nest jamais mis à jour tant que vous ne vous déconnectez pas de lapplication et que vous ne vous connectez pas à nouveau!

Une stratégie possible pour résoudre ce problème consiste à recharger les données du compte chaque fois que le tableau de bord est chargé, afin déviter les données de blocage.

Tâche

Créez une nouvelle fonction updateAccountData:

async function updateAccountData() {
  const account = state.account;
  if (!account) {
    return logout();
  }

  const data = await getAccount(account.user);
  if (data.error) {
    return logout();
  }

  updateState('account', data);
}

Cette méthode vérifie que nous sommes actuellement connectés puis recharge les données du compte à partir du serveur.

Créez une autre fonction nommée refresh:

async function refresh() {
  await updateAccountData();
  updateDashboard();
}

Celui-ci met à jour les données du compte, puis se charge de mettre à jour le code HTML de la page du tableau de bord. Cest ce que nous devons appeler lorsque litinéraire du tableau de bord est chargé. Mettez à jour la définition ditinéraire avec:

const routes = {
  '/login': { templateId: 'login' },
  '/dashboard': { templateId: 'dashboard', init: refresh }
};

Essayez de recharger le tableau de bord maintenant, il devrait afficher les données de compte mises à jour.


🚀 Challenge

Maintenant que nous rechargeons les données du compte chaque fois que le tableau de bord est chargé, pensez-vous que nous devons encore conserver toutes les données du compte?

Essayez de travailler ensemble pour modifier ce qui est enregistré et chargé à partir de localStorage pour ninclure que ce qui est absolument nécessaire pour que lapplication fonctionne.

Quiz de validation des connaissances

Quiz de validation des connaissances

Affectation

Implémenter la boîte de dialogue "Ajouter une transaction"

Voici un exemple de résultat après avoir terminé laffectation:

Capture d’écran montrant un exemple de boîte de dialogue "Ajouter une transaction"