react当我们有两个完全不相关的组件想要通信时,就可以利用这种模式,其中一个组件负责订阅某个消息,而另一个元素则负责发送这个消息。使用Context配合

在nextjs项目中,发现两个组件没啥关系,例如一个是一直存在的头部组件,另一个是页面中的组件,当我点击头部组件中的特定按钮时,把数据传递到页面组件中,页面组件接受到canshu数据后在做其他操作,那么他们两个如何通讯,通过context配合观察者模式实现。

  1. 首先在其共同的祖先组件中使用context
    下面的代码文件是根路由组件Layout.js
    其中children就是根据路由渲染的对应的页面

    ....
    import {ConContext} from "../utils/test1";
    import EventBus from "../utils/test";
    const Layout = () => {
    <ConContext.Provider value={EventBus()}>


    {children}

    </ConContext.Provider>

    }
    export default Layout

test1.js

创建context

复制代码
import { createContext } from 'react';

export const ConContext = createContext(null);

test.js

使用 useEffect 包装下,否则会在服务端渲染,而服务端又没有浏览器对象,所以会报错,导致打包失败,下面代码中,注释的代码就会报找不到document,所以使用useEffect包装下创建了一个自定义hook勾子 useEvent,这个钩子创建一个变量用于接收EventBus这个类,为什么要在useEffect 创建EventBus呢,因为只有在useEffect中才能拿到浏览器对象,然后在返回变量。然后导出这个useEvent

复制代码
"use client"
import {useEffect, useState} from 'react'
// 使用 useEffect 包装下,否则会在服务端渲染,而服务端又没有浏览器对象,所以会报错,导致打包失败
const useEvent = () => {
  const [val, setVal] = useState('')
  useEffect(() => {

    class EventBus {
      constructor() {
          this.bus = document.createElement('fakeelement');
      }

      addEventListener(event, callback) {
          this.bus.addEventListener(event, callback);
      }

      removeEventListener(event, callback) {
          this.bus.removeEventListener(event, callback);
      }

      dispatchEvent(event, detail = {}){
          this.bus.dispatchEvent(new CustomEvent(event, { detail }));
      }
    }
    setVal(new EventBus)
  },[])
  return val
}

export default useEvent


// "use client"
// class EventBus {
//   constructor() {
//       this.bus = document.createElement('fakeelement');
//   }

//   addEventListener(event, callback) {
//       this.bus.addEventListener(event, callback);
//   }

//   removeEventListener(event, callback) {
//       this.bus.removeEventListener(event, callback);
//   }

//   dispatchEvent(event, detail = {}){
//       this.bus.dispatchEvent(new CustomEvent(event, { detail }));
//   }
// }

// export default new EventBus
  1. 在header中触发事件,
    header.js
    使用useContext接收数据

    import React, {useState, useContext} from 'react';
    const Header = () => {
    const EventBus = useContext(ConContext)
    // 点击按钮时触发
    const clickTab = (url) => {
    setOpenMenu(false)
    // 点击头部菜单切换轮播图开始
    // EventBus.dispatchEvent('myEvent', {log: 2})
    // 点击头部菜单切换轮播图结束
    push(url)
    }
    }
    export default Header

  2. 在对应的页面组件中接收myEvent这个自定义事件

    import React, { useEffect,useRef, useContext } from "react";
    import { ConContext } from '../../utils/test1';
    const Products = (props) => {
    const EventBus = useContext(ConContext)
    console.log(EventBus,'EventBus');
    const slider = useRef();
    const handleEvent = (e) => {
    slider.current.goTo(e.detail.log)
    }
    useEffect(() => {
    EventBus.addEventListener('myEvent', handleEvent)
    }, [EventBus]);
    }

    export default Products;

参考:10种React组件之间通信的方法

相关推荐
神奇的程序员5 小时前
开发了一个管理本地开发环境的软件
前端·flutter
天若有情6735 小时前
程序员原创|借鉴JS事件冒泡,根治电脑文件混乱的“冒泡整理法”
开发语言·javascript·windows·ecmascript·电脑·办公·日常
XiYang-DING6 小时前
HTML 核心标签
前端·html
Csvn6 小时前
技术选型方法论
前端
Csvn6 小时前
前端架构演进:从页面到平台的十年变革
前端
李伟_Li慢慢7 小时前
ShaderToy-山峦+蓝天+白云
前端·webgl
小码哥_常7 小时前
Android字体字重设置全攻略:XML黑科技+Kotlin动态实现,告别.ttf臃肿
前端
FYKJ_20107 小时前
springboot校园兼职平台--附源码02041
java·javascript·spring boot·python·eclipse·django·php
言萧凡_CookieBoty8 小时前
AI 编程省 Token 实战:从 Spec、上下文工程到模型分层的降本策略
前端·ai编程
DFT计算杂谈9 小时前
wannier90 参数详解大全
java·前端·css·html·css3