Maison >interface Web >js tutoriel >Architecture pilotée par les événements pour la communication des composants Clean React

Architecture pilotée par les événements pour la communication des composants Clean React

Barbara Streisand
Barbara Streisandoriginal
2024-12-06 15:26:13347parcourir

Vous en avez assez de l'enchevêtrement sans fin de chaînes de forage et de rappel d'accessoires dans vos applications React ? La gestion de l'état et de la communication entre des composants profondément imbriqués ressemble-t-elle à lutter avec du code spaghetti ?

Une architecture basée sur les événements peut simplifier les interactions de vos composants, réduire la complexité et rendre votre application plus maintenable. Dans cet article, je vais vous montrer comment utiliser un hook useEvent personnalisé pour découpler les composants et améliorer la communication au sein de votre application React.

Laissez-moi vous expliquer, commençons par


Le problème : chaînes de forage d'accessoires et de rappel

Dans le développement d'applications modernes, la gestion de l'état et de la communication entre les composants peut rapidement devenir fastidieuse. Cela est particulièrement vrai dans les scénarios impliquant du perçage d'accessoires (dans lequel les données doivent être transmises à travers plusieurs niveaux de composants imbriqués) et des chaînes de rappel, qui peuvent conduire à une logique embrouillée et rendre le code plus difficile à utiliser. maintenir ou déboguer.

Ces défis créent souvent des composants étroitement couplés, réduisent la flexibilité et augmentent la charge cognitive des développeurs qui tentent de retracer la façon dont les données circulent dans l'application. Sans une meilleure approche, cette complexité peut ralentir considérablement le développement et conduire à une base de code fragile.

Le flux traditionnel : accessoires vers le bas, rappels vers le haut

Dans une application React typique, les composants parents transmettent des accessoires à leurs enfants, et les enfants communiquent en retour avec le parent en déclenchant des rappels. Cela fonctionne bien pour les arbres de composants peu profonds, mais à mesure que la hiérarchie s'approfondit, les choses commencent à devenir compliquées :

Props Drilling : les données doivent être transmises manuellement à travers plusieurs niveaux de composants, même si seul le composant le plus profond en a besoin.

Chaînes de rappel : de même, les composants enfants doivent transmettre les gestionnaires d'événements vers le haut de l'arborescence, créant ainsi des structures étroitement couplées et difficiles à maintenir.

Un problème courant : la complexité des rappels

Prenons ce scénario, par exemple :

  • Le Parent passe les accessoires aux Enfants A.
  • À partir de là, les accessoires sont analysés jusqu'à GrandChildren A/B et finalement à SubChildren N.
  • Si SubChildren N doit informer le parent d'un événement, il déclenche un rappel qui remonte à travers chaque composant intermédiaire.

Cette configuration devient plus difficile à gérer à mesure que l'application se développe. Les composants intermédiaires n'agissent souvent que comme des intermédiaires, transmettant des accessoires et des rappels, ce qui gonfle le code et réduit la maintenabilité.

Event-Driven Architecture for Clean React Component Communication

Pour résoudre le problème du forage d'accessoires, nous nous tournons souvent vers des solutions telles que les bibliothèques globales de gestion d'état (par exemple, Zustand) pour rationaliser le partage de données. Mais qu’en est-il de la gestion des rappels ?

C’est là qu’une approche événementielle peut changer la donne. En découplant les composants et en nous appuyant sur des événements pour gérer les interactions, nous pouvons considérablement simplifier la gestion des rappels. Explorons comment cette approche fonctionne.


La solution : entrez dans l'approche événementielle

Event-Driven Architecture for Clean React Component Communication

Au lieu de s'appuyer sur des rappels directs pour communiquer dans l'arborescence, une architecture basée sur les événements découple les composants et centralise la communication. Voici comment cela fonctionne :

Répartition des événements

Lorsque SubChildren N déclenche un événement (par exemple, onMyEvent), il n'appelle pas directement de rappel dans le parent.
Au lieu de cela, il distribue un événement qui est géré par un gestionnaire d'événements centralisé.

Gestion centralisée

Le Gestionnaire d'événements écoute l'événement distribué et le traite.
Il peut avertir le Parent (ou tout autre élément intéressé) ou déclencher des actions supplémentaires selon les besoins.

Les accessoires restent à la baisse

Les accessoires sont toujours transmis dans la hiérarchie, garantissant que les composants reçoivent les données dont ils ont besoin pour fonctionner.

Ceci peut être résolu avec des outils de gestion d'état centralisés comme zustand, redux, mais ne sera pas abordé dans cet article.


Mise en œuvre

Mais, comment mettre en œuvre cette architecture ?

crochet useEvent

Créons un hook personnalisé appelé useEvent, ce hook sera chargé de gérer l'abonnement aux événements et de renvoyer une fonction de répartition pour déclencher l'événement cible.

Comme j'utilise TypeScript, je dois étendre l'interface de la fenêtre Event afin de créer des événements personnalisés :

interface AppEvent<PayloadType = unknown> extends Event {
  detail: PayloadType;
}

export const useEvent = <PayloadType = unknown>(
  eventName: keyof CustomWindowEventMap,
  callback?: Dispatch<PayloadType> | VoidFunction
) => {
  ...
};

Ce faisant, nous pouvons définir une carte d'événements personnalisée et transmettre des paramètres personnalisés :

interface AppEvent<PayloadType = unknown> extends Event {
  detail: PayloadType;
}

export interface CustomWindowEventMap extends WindowEventMap {
  /* Custom Event */
  onMyEvent: AppEvent<string>; // an event with a string payload
}

export const useEvent = <PayloadType = unknown>(
  eventName: keyof CustomWindowEventMap,
  callback?: Dispatch<PayloadType> | VoidFunction
) => {
  ...
};

Maintenant que nous avons défini les interfaces nécessaires, voyons le code de hook final

import { useCallback, useEffect, type Dispatch } from "react";

interface AppEvent<PayloadType = unknown> extends Event {
  detail: PayloadType;
}

export interface CustomWindowEventMap extends WindowEventMap {
  /* Custom Event */
  onMyEvent: AppEvent<string>;
}

export const useEvent = <PayloadType = unknown>(
  eventName: keyof CustomWindowEventMap,
  callback?: Dispatch<PayloadType> | VoidFunction
) => {
  useEffect(() => {
    if (!callback) {
      return;
    }

    const listener = ((event: AppEvent<PayloadType>) => {
      callback(event.detail); // Use `event.detail` for custom payloads
    }) as EventListener;

    window.addEventListener(eventName, listener);
    return () => {
      window.removeEventListener(eventName, listener);
    };
  }, [callback, eventName]);

  const dispatch = useCallback(
    (detail: PayloadType) => {
      const event = new CustomEvent(eventName, { detail });
      window.dispatchEvent(event);
    },
    [eventName]
  );

  // Return a function to dispatch the event
  return { dispatch };
};

Le hook useEvent est un hook React personnalisé permettant de s'abonner et de distribuer des événements de fenêtre personnalisés. Il vous permet d'écouter des événements personnalisés et de les déclencher avec une charge utile spécifique.

Ce que nous faisons ici est assez simple, nous utilisons le système de gestion d'événements standard et l'étendons afin d'accueillir nos événements personnalisés.

Paramètres :

  • eventName (string) : Le nom de l'événement à écouter.
  • callback (facultatif) : une fonction à appeler lorsque l'événement est déclenché, recevant la charge utile en argument.

Caractéristiques:

  • Event Listener : il écoute l'événement spécifié et appelle le rappel fourni avec les détails de l'événement (charge utile personnalisée).
  • Répartition des événements : le hook fournit une fonction de répartition pour déclencher l'événement avec une charge utile personnalisée.

Exemple:

interface AppEvent<PayloadType = unknown> extends Event {
  detail: PayloadType;
}

export const useEvent = <PayloadType = unknown>(
  eventName: keyof CustomWindowEventMap,
  callback?: Dispatch<PayloadType> | VoidFunction
) => {
  ...
};

Ok cool mais, qu'en est-il d'un

Exemple du monde réel ?

Découvrez ce StackBlitz (s'il ne se charge pas, veuillez le vérifier ici)

Cet exemple simple montre le but du hook useEvent, essentiellement, le bouton du corps distribue un événement qui est intercepté à partir des composants Sidebar, Header et Footer, qui est mis à jour en conséquence.

Cela nous permet de définir des réactions de cause à effet sans avoir besoin de propager un rappel à de nombreux composants.


Cas d'utilisation réels pour useEvent

Voici quelques cas d'utilisation réels où le hook useEvent peut simplifier la communication et découpler les composants dans une application React :


1. Système de notifications

Un système de notification nécessite souvent une communication globale.

  • Scénario :

    • Lorsqu'un appel API réussit, une notification de « réussite » doit être affichée dans l'application.
    • Les composants tels qu'un « badge de notifications » dans l'en-tête doivent également être mis à jour.
  • Solution : utilisez le hook useEvent pour envoyer un événement onNotification avec les détails de la notification. Des composants tels que NotificationBanner et Header peuvent écouter cet événement et se mettre à jour indépendamment.

2. Changement de thème

Lorsqu'un utilisateur change de thème (par exemple, mode clair/sombre), plusieurs composants peuvent devoir répondre.

  • Scénario :

    • Un composant ThemeToggle distribue un événement onThemeChange personnalisé.
    • Les composants tels que la barre latérale et l'en-tête écoutent cet événement et mettent à jour leurs styles en conséquence.
  • Avantages : Pas besoin de transmettre l'état du thème ou les fonctions de rappel via des accessoires dans toute l'arborescence des composants.

3. Liaisons de clés globales

Implémentez des raccourcis globaux, comme appuyer sur « Ctrl S » pour enregistrer un brouillon ou sur « Échap » pour fermer un modal.

  • Scénario :
    • Un écouteur global keydown distribue un événement onShortcutPressed avec les détails de la touche enfoncée.
    • Les composants modaux ou autres éléments de l'interface utilisateur répondent à des raccourcis spécifiques sans dépendre des composants parents pour transmettre l'événement clé.

4. Mises à jour en temps réel

Les applications telles que les applications de chat ou les tableaux de bord en direct nécessitent plusieurs composants pour réagir aux mises à jour en temps réel.

  • Scénario :
    • Une connexion WebSocket distribue des événements onNewMessage ou onDataUpdate lorsque de nouvelles données arrivent.
    • Des composants tels qu'une fenêtre de discussion, des notifications et des compteurs de messages non lus peuvent gérer indépendamment les mises à jour.

5. Validation du formulaire entre les composants

Pour les formulaires complexes comportant plusieurs sections, les événements de validation peuvent être centralisés.

  • Scénario :
    • Un composant de formulaire distribue des événements onFormValidate lorsque les utilisateurs remplissent des champs.
    • Un composant récapitulatif écoute ces événements pour afficher les erreurs de validation sans couplage étroit avec la logique du formulaire.

6. Suivi analytique

Suivez les interactions des utilisateurs (par exemple, clics sur des boutons, événements de navigation) et envoyez-les à un service d'analyse.

  • Scénario :
    • Envoyez les événements onUserInteraction avec les détails pertinents (par exemple, l'étiquette du bouton cliqué).
    • Un gestionnaire d'analyse central écoute ces événements et les envoie à une API d'analyse.

7. Outils collaboratifs

Pour les outils collaboratifs comme les tableaux blancs partagés ou les éditeurs de documents, les événements peuvent gérer les interactions multi-utilisateurs.

  • Scénario :
    • Distribuez des événements onUserAction chaque fois qu'un utilisateur dessine, tape ou déplace un objet.
    • D'autres clients et composants de l'interface utilisateur écoutent ces événements pour refléter les changements en temps réel.

En tirant parti du hook useEvent dans ces scénarios, vous pouvez créer des applications modulaires, maintenables et évolutives sans compter sur des accessoires ou des chaînes de rappel profondément imbriqués.


Conclusions

Les événements peuvent transformer la façon dont vous créez des applications React en réduisant la complexité et en améliorant la modularité. Commencez petit : identifiez quelques composants de votre application qui bénéficieraient d'une communication découplée et implémentez le hook useEvent.

Avec cette approche, vous simplifierez non seulement votre code, mais vous faciliterez également sa maintenance et sa mise à l'échelle à l'avenir.

Pourquoi utiliser les événements ?
Les événements brillent lorsque vous avez besoin que vos composants réagissent à quelque chose qui s'est produit ailleurs dans votre application, sans introduire de dépendances inutiles ni de chaînes de rappel alambiquées. Cette approche réduit la charge cognitive et évite les pièges liés au couplage étroit de composants.

Ma recommandation
Utilisez des événements pour la communication entre composants : lorsqu'un composant doit informer les autres d'une action ou d'un changement d'état, quel que soit leur emplacement dans l'arborescence des composants.
Évitez d'utiliser des événements pour la communication intra-composant, en particulier pour les composants étroitement liés ou directement connectés. Pour ces scénarios, comptez sur les mécanismes intégrés de React tels que les accessoires, l'état ou le contexte.

Une approche équilibrée
Même si les événements sont puissants, leur utilisation excessive peut conduire au chaos. Utilisez-les judicieusement pour simplifier la communication entre des composants faiblement connectés, mais ne les laissez pas remplacer les outils standard de React pour gérer les interactions locales.

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn