Redux
À Mapado, nous commençons de plus en plus à travailler avec du React + Redux, et étant donné que ce sont simplement des librairies assez permissives, chaque développeur a un peu sa manière de faire.
Nous avions donc besoin de standardiser nos façons de faire. C'est chose faite !
Ce document est encore à approuver, nous ne sommes pas forcément sûrs de nos choix, mais ils nous parraissent à l'heure actuelle les plus pertinents dans notre cas.
Nous sommes d'ailleurs très preneurs de retours sur ces sujets si vous avez des utilisations différentes, soit en ouvrant une issue.
Quand utiliser Redux
Redux NE DOIT ÊTRE utilisé que lorsqu'il y a un besoin global de partage à travers toute l'application.
React permet déjà d'avoir des infos dans son state, et depuis quelques temps maintenant l'API contexte permet de partager efficacement les données entre composants.
En savoir plus : "You might not need Redux" par Dan Abramov.
Librairies utilisées avec Redux
- Sélecteurs: reselect
- Structures de données: immutable-js
- Middleware pour actions asynchrones: redux-thunk
Le guide de bonne pratique de redux
Redux propose le Redux Style Guide qui est leur guide "officiel" pour écrire du code redux. C'est un très bon guide pour savoir quelle approche utiliser et savoir comment architecturer son code.
Immutabilité des states
Les states DOIVENT être des Map
ou des Record
immutable-js.
Redux marque comme essentiel le fait de ne pas "muter" les states
Initialisation du "state"
Le "state" DEVRAIT être initialisé par défaut avec toutes les clés qu'il peut utiliser.
Cela permet aux composants de toujours avoir leur valeurs "settée" et de ne pas traiter les cas
if null
.Cela permet aussi aux "containers" de ne pas avoir cette logique à gérer, par exemple :
state.foo.get('bar') || false
.
On PEUT utiliser un Record
immutable pour interdire tout ajout de nouvelle clé, et ainsi avoir du sucre syntaxique pour y accéder (state.foo.bar
plutôt que state.foo.get('bar')
).
Suppression de clé
On ne DOIT PAS supprimer de clé du state.
Si l'on supprime une clé, on se retrouve dans le même cas que précédement.
Du coup : préférez state.set('foo', false)
, ou state.set('foo', null)
plutôt que state.remove('foo')
.
Dispatch dans les actions
On ne DEVRAIT PAS envoyer trop de dispatch
dans les actions.
On ne DEVRAIT PAS envoyer deux dispatch successifs s'il n'y a pas besoin de rafraîchir la vue.
Un
dispatch
entraine un rafraîchissement des composants React.
C'est plutôt au reducer de faire ce job:
👎
👍
cf. Avoid Dispatching Many Actions Sequentially
Statut des actions effectuées sur les données
Lors d'un appel asynchrone, nous avons souvent besoin de savoir l'état de cet appel (pour afficher un loader ou un message d'erreur, par exemple).
Appelons "clé de la donnée" la clé dans le state redux où sera stockée notre donnée une fois récupérée.
Nous POUVONS stocker le status de cette donnée sous une autre clé dans notre state redux.
La clé de la donnée concernant le status DOIT avoir la forme suivante : <clé de la donnée>-<action>-status
.
Les
-
ont ici la même utilité que dans BEM par exemple.Étant donné que nos states sont des objets immutable, l'utilisation d'un
-
n'est pas très gênant.
<action>
est le nom de l'action étant effectuée sur la donnée, par exemple “articleList-fetch-status”.
La valeur DOIT être une Map contenant une clé status
et une clé error
.
status
Valeur de la clé La valeur de status
DEVRAIT prendre les valeurs suivantes :
- null
- IN_PROGRESS
- SUCCEEDED
- FAILED
Elle PEUT aussi prendre une valeur personnalisée, par exemple :
- IN_PROGRESS_BOTTOM
- IN_PROGRESS_TOP
error
Valeur de la clé L’erreur DOIT se trouver dans l’objet de status avec la clé error
.
Sa valeur par défaut DOIT être null
.
Sa valeur DEVRAIT être une string ou un objet.
C’est au composant d’avoir la logique de quoi faire en fonction de l’erreur.
Cas particulier : Opérations sur les items d’une liste
Les opérations sur les items d’une liste DOIVENT suivre la même logique, mais PEUVENT être stockées dans une Map
.
Les clés de cette Map
DEVRAIT être l'identifiant de l'item concerné.
Exemple:
Sélecteur vs state complexe
Le state DEVRAIT être utilisé pour stocker les données brutes.
Un sélecteur DEVRAIT être utilisé pour “grouper” / “filtrer” / “trier”, finalement “travailler” sur les données brutes pour les envoyer au composant.
Vous pouvez consulter la doc de reselect pour plus d'informations sur les sélecteurs.
cf. Use Selector Functions to Extract and Transform Data
List vs Map vs Collection
On DEVRAIT suffixer le nom de la clé par le type de donnée.
- Si c'est une
List
d'articles, la clé devrait s'appelerarticleList
. - Si c'est une
Map
d'articles, la clé devrait s'appelerarticleMap
. - Si c'est un objet custom
FooBarCollection
contenant des articles, la clé devrait s'appelerarticleFooBarCollection
.
Il faut bien conserver le nommage de la clé de "status" en fonction.
Dans notre dernier exemple, la clé de status se nommera par exemple articleFooBarCollection-fetch-status
.
Profondeur du state
On NE DEVRAIT PAS avoir plus d'un niveau dans le state.
👎
👍
Mono-state vs multi-state
On DEVRAIT avoir un seul "state" dans le cas d'une application simple.
On PEUT avoir un découpage fonctionnel des states dans le cas d'une application complexe.