# Cells
Ce que nous cherchons à faire ici constituent en réalité des objectifs partagés par la plupart des applications web. Nous voulions voir s'il était possible de faciliter la vie aux développeurs. Nous pensons être arrivé à réaliser quelque chose d'utile. Nous appelons ça les Cells (ou cellules en français). Les Cells proposent une approche simple et déclarative pour récupérer des données au sein de vos composants. (Vous pouvez lire la documentation complète à propos des Cells. You can read the full documentation about Cells ici.
Lorsque vous créez une nouvelle Cell, vous exportez quelques constantes, toujours nommées de façon identique, et Redwood s'appuie dessus pour mettre en place la mécanique. Une Cell ressemble typiquement à ceci:
export const QUERY = gql`
query {
posts {
id
title
body
createdAt
}
}
`
export const Loading = () => <div>Chargement...</div>
export const Empty = () => <div>Aucun article disponible!</div>
export const Failure = ({ error }) => (
<div>Erreur lors du chargement des articles: {error.message}</div>
)
export const Success = ({ posts }) => {
return posts.map((post) => (
<article>
<h2>{post.title}</h2>
<div>{post.body}</div>
</article>
))
}
Lorsque React affiche ce composant, Redwood va:
- Exécuter la requête
QUERY
et afficher le composantLoading
jusqu'à ce qu'une réponse soit reçue - Lorsque la requête retourne une réponse, il va afficher un des trois états suivants:
- S'il y a eu une erreur, le composant
Failure
- Si aucune donnée n'est retournée (c'est à dire
null
ou un tableau vide), le composantEmpty
- Dans le cas contraire (ni erreur, ni vide), le composant
Success
- S'il y a eu une erreur, le composant
Il existe également quelques outils supplémentaire pour générer le cycle de vie du composant comme beforeQuery
(pour manipuler les propriétés passées à QUERY
) et afterQuery
(pour manipuler les données retournées par GraphQL avant qu'elles ne soient transmises au composant Success
)
Le minimum dont vous avez besoin pour une Cell sont les exports QUERY
et Success
. Si vous n'exportez pas Empty
, Success
recevra les données vides. Si vous n'exportez pas Failure
, les éventuelles erreurs seront envoyées à la console.
Pour déterminer dans quels cas utiliser les Cells, gardez en tête qu'elles sont utiles lorsque vos composants ont besoin de récupérer des données depuis la base, ou depuis tout autre service qui pourrait avoir un délai de réponse. Laissez Redwood se charger de jongler avec les états, de manière à pouvoir porter votre attention sur le comportement attendu de vos composants correctement affichés avec leur données.
# Notre Première Cell
La page d'accueil affichant une liste d'articles est un candidat parfait pour réaliser notre première cellule. Naturellement, nous avons prévu un générateur pour ça:
yarn rw g cell BlogPosts
L'exécution de cette commande provoque la création d'un nouveau fichier /web/src/components/BlogPostsCell/BlogPostsCell.js
(et son fichier de test associé) avec un peu de code par défaut pour vous faciliter la tâche:
// web/src/components/BlogPostsCell/BlogPostsCell.js
export const QUERY = gql`
query BlogPostsQuery {
blogPosts {
id
}
}
`
export const Loading = () => <div>Loading...</div>
export const Empty = () => <div>Empty</div>
export const Failure = ({ error }) => <div>Error: {error.message}</div>
export const Success = ({ blogPosts }) => {
return JSON.stringify(blogPosts)
}
Lorsque vous utilisez le générateur, vous pouvez employer le type de casse qui vous plaît. Redwood fera en sorte de s'adapter pour créer une cellule avec un nom de fichier correct. Ainsi toutes les commandes ci-dessous aboutissent à créer un fichier avec le même nom:
yarn rw g cell blog_posts yarn rw g cell blog-posts yarn rw g cell blogPosts yarn rw g cell BlogPosts
Vous devez juste pensez à indiquer d'une façon ou d'une autre que vous utilisez plusieurs mots. Appeler
yarn redwood g cell blogposts
sans utiliser aucune casse pour séparer "blog" et "posts" va générer un fichierweb/src/components/BlogpostsCell/BlogpostsCell.js
.
Pour vous aider à être efficace, le générateur suppose que vous utiliserez une requête racine GraphQL nommées de la même façon que votre Cell et écrit pour vous une requête minimale pour récupérer des données depuis la base. Dans le cas présent, la requête a donc été nommée blogPosts
. Cependant, ce nom de requête n'est pas valide par rapport à ce qui a déjà été créé dans nos fichiers SDL et Service. Nous devons donc renommer blogPosts
en posts
à la fois dans le nom de la requête GraphQL et dans la propriété passée à Success
:
// web/src/components/BlogPostsCell/BlogPostsCell.js
export const QUERY = gql`
query BlogPostsQuery {
posts { id
}
}
`
export const Loading = () => <div>Loading...</div>
export const Empty = () => <div>Empty</div>
export const Failure = ({ error }) => <div>Error: {error.message}</div>
export const Success = ({ posts }) => { return JSON.stringify(posts)}
Insérons cette Cell dans notre HomePage
et voyons ce qui se passe:
// web/src/pages/HomePage/HomePage.js
import BlogLayout from 'src/layouts/BlogLayout'
import BlogPostsCell from 'src/components/BlogPostsCell'
const HomePage = () => {
return (
<BlogLayout>
<BlogPostsCell /> </BlogLayout>
)
}
export default HomePage
Le navigateur devrait en principe montrer un tableau avec un peu de contenu (en supposant que vous ayez créé un article à l'étape du scaffolding un peu plus tôt). Impeccable!
Dans le composant
Success
, d'où vient doncposts
?Remarquez que dans le composant
QUERY
, nous avons nommée notre requêteposts
. Quelque soit le nom de la requête, ce sera le nom de la propriété qui sera transmise au composantSuccess
et qui contiendra vos données. Vous pouvez toutefois créer un alias de la façon suivante:
export const QUERY = gql` query BlogPostsQuery { postIds: posts { id } } `
De cette manière la propriété
postIds
sera transmise àSuccess
au lieu deposts
En plus de l'identifiant id
qui a été ajouté dans QUERY
par le générateur, récupérons également le titre, le contenu et la date de création de l'article:
// web/src/components/BlogPostsCell/BlogPostsCell.js
export const QUERY = gql`
query BlogPostsQuery {
posts {
id
title body createdAt }
}
`
La page devrait désormais afficher un dump de l'ensemble des données pour tous les articles enregistrés:
Success
est ni plus ni moins qu'un bon vieux composant React, vous pouvez donc le modifier simplement pour afficher chaque article dans un format un peu plus sympa et lisible:
// web/src/components/BlogPostsCell/BlogPostsCell.js
export const Success = ({ posts }) => {
return posts.map((post) => ( <article key={post.id}> <header> <h2>{post.title}</h2> </header> <p>{post.body}</p> <div>Créé le: {post.createdAt}</div> </article> ))}
Et ce faisant, nous avons maintenant notre blog! Ok, à ce stade c'est encore le plus basique et hideux blog jamais vu sur Internet.. mais c'est déjà quelque chose! (Pas d'inquiétude, nous avons encore un tas de fonctionnalités à ajouter)
# Résumé
Pour résumer, qu'avons nous réalisé jusqu'ici ?
- Génération de la page d'accueil
- Génération du Layout pour notre blog
- Définition du schéma de la base de données
- Application d'une migrations pour mettre à jour la base de données et créer une table
- Réalisation d'un Scaffold pour créer une interface CRUD sur la table
- Création d'une Cell pour charger les donner et gérer les états "loading", "empty", "failure" et enfin "success".
- Ajout de la Cell à notre page d'accueil
En réalité, cette différentes étapes sont ni plus ni moins ce qui deviendra votre façon habituelle d'ajouter de nouvelles fonctionnalités dans une application Redwood.
Jusqu'ici, hormis un peu de code HTML, nous n'avons pas écrit grand chose à la main. En particulier, nous n'avons pratiquement pas eu à écrire de code pour récupérer les données depuis la base. Le développement web s'en trouve facilité et devient même agréable, qu'en pensez-vous?