首页 > 其他分享 >react useReactStore.js

react useReactStore.js

时间:2023-05-12 13:11:29浏览次数:39  
标签:watcherList const watcher react nextValue useReactStore path js store

import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import _get from 'lodash.get';
import _set from 'lodash.set';

export const KEY_SAVED_TICK_COUNT = 'KEY_SAVED_TICK_COUNT';

export class GlobalStore {
  constructor(initialValues) {
    this.storeData = initialValues || {};
    this.watcherList = []; // {path, watcher, preValue}
    this.tickCount = 0;
  }

  getValue(path) {
    return _get(this.storeData, path);
  }

  setValue(path, value) {
    this.tickCount += 1;
    _set(this.storeData, path, value);
    _set(this.storeData, KEY_SAVED_TICK_COUNT, this.tickCount);
    this.notifyWatcher();
  }

  watch(path, watcher, from) {
    if (typeof watcher !== 'function') {
      throw 'watcher must function';
    }
    this.watcherList.push({ path, watcher, preValue: undefined });
    console.log('GlobalStore watch', from, this.watcherList.length);
  }

  unwatch(path, watcher, from) {
    this.watcherList = this.watcherList.filter((obj) => {
      return obj.path !== path || obj.watcher !== watcher;
    });

    console.log('GlobalStore unwatch', from, this.watcherList.length);
  }

  notifyWatcher() {
    const watcherList = this.watcherList || [];
    for (let i = 0; i < watcherList.length; i++) {
      const { path, watcher, preValue } = watcherList[i];
      const nextValue = this.getValue(path);
      if (preValue !== nextValue) {
        watcher(nextValue, preValue, this);
        watcherList[i].preValue = nextValue;
      }
    }
  }
}

/**
 * 创建一个GlobalStore
 * @param initialValues
 */
function useCreateGlobalStore(initialValues) {
  return useMemo(() => {
    return new GlobalStore(initialValues);
  }, []);
}

/**
 * 订阅并获取value的变化
 * @param store
 * @param path
 */
function useGlobalValue(store, path) {
  const valueRef = useRef();

  const [value, setValue] = useState(() => {
    const nowValue = store.getValue(path);
    valueRef.current = nowValue;
    return nowValue;
  });

  useEffect(() => {
    const init = () => {
      const nextValue = store.getValue(path);
      if (valueRef.current !== nextValue) {
        valueRef.current = nextValue;
        setValue(nextValue);
      }
    };

    init();

    const watcher = (nextValue) => {
      console.log('useGlobalValue nextValue', nextValue);
      if (valueRef.current !== nextValue) {
        valueRef.current = nextValue;
        setValue(nextValue);
      }
    };

    store.watch(path, watcher);

    return () => {
      store.unwatch(path, watcher);
    };
  }, [store, path]);

  /**
   * 更新值
   */
  const updateValue = useCallback(
    (nextValue) => {
      store.setValue(path, nextValue);
    },
    [store, path],
  );

  /**
   * 在Render之前获取最新值
   */
  const getCurrent = useCallback(() => {
    return valueRef.current;
  }, [[store, path]]);

  return [value, updateValue, getCurrent, store];
}

export { useCreateGlobalStore, useGlobalValue };

  

标签:watcherList,const,watcher,react,nextValue,useReactStore,path,js,store
From: https://www.cnblogs.com/lhp2012/p/17393792.html

相关文章

  • JS的预解析
    JS的预解析是指在代码执行之前,JavaScript引擎会先对代码进行一次扫描,将变量声明和函数声明提升到当前作用域的顶部,以便在代码执行时能够正确地访问这些变量和函数。这个过程也被称为“提升”。具体来说,在预解析过程中,JavaScript引擎会将函数声明和变量声明提升到当前作用域的顶部......
  • JS 判断电脑是 Win 还是 mac?
    //判断系统类型functionOSnow(){constagent=navigator.userAgent.toLowerCase()constisMac=/macintosh|macosx/i.test(navigator.userAgent)if(agent.indexOf('win32')>=0||agent.indexOf('wow32')>=0){//yourcode......
  • 如何将bash变量传递给JSON
    我正在尝试编写示例脚本,在其中生成诸如"student-101...student-160"之类的名称。我需要发布JSON数据,当我这样做时,出现JSON解析错误。这是我的脚本:123456789name="student-10"for i in {1..1}do   r_name=$name$i   echo $r_name  curl -i -H 'Au......
  • 浅谈 Node.js
    Node.js是什么?Node.js®是一个开源、跨平台的JavaScript运行时环境。官网:https://nodejs.org/zh-cn更多精彩内容,请微信搜索“前端爱好者“,戳我查看。Node.js≠JavaScriptNode.js中,没有BOM和DOM。Nodejs不是一门语言,只是一个跨平台的JavaScript运行时环境。Node......
  • json在前端的使用
    调用接口后一般传递的都是json格式的数据,这个json格式的数据吗可能会有多层嵌套,如何获取这个最内层的嵌套value值便是关键。而数据解析放在前端便更加的简单,可以像数组一样使用但是需要解决以下问题:1.数据在后端的时候已经是json格式的object,如何异步发送这个json数据到前端(最新......
  • React笔记-组件通信(六)
    React笔记-组件通信(六)props概念props是组件之间通讯的纽带props也是组件中内置的一个属性通过父级组件传入在类组件里可以直接通过this.props获取注意:props是不可变的(只读)修改需要从传值的组件中修改props改变会使视图重新渲染组件传值父传子在父组件......
  • web游览器的标签页仿 ios mac 苹果的墓碑机制 (js代码)
    背景:本来项目开发系统防挂机功能,在其余游览器中均可以使用。但是呢在苹果的safair游览器中会出现几率失效,最后经过排查发现是苹果的墓碑机制导致。即:此标签页活跃,其他标签页假死。然后就导致防挂机失效了。原理:假如当前游览器中有3个标签页分别是A,B,C,每个标签页都有倒计时。正......
  • JScrollPane
    JScrollPane组件()功能介绍:    当容器的显示区域不足以同时显示所有组件的时候,滚动面版JScrollPane(后省略为JS)可以通过滚动的方式将组件的内容展示出来。使用方法:JS通过将一些组件先添加到JPanel中,再将JPanel添加到JS上,而JTextArea、JList、JTable等组件都没有自......
  • js过滤粘贴内容
    <html><head><title>HtmlEditorFilter</title><styletype="text/css">.editor{border:1pxsolid#333333;}</style></head><body><iframesrc="iframeblankpage.html"id="e......
  • inputstream转json
    inputstream是一个Java中的输入流,它允许从源读取数据。如果要将inputstream转换为JSON,需要使用相应的解析器,例如GSON或Jackson。以下是使用GSON的示例代码:InputStreaminputStream=...;Gsongson=newGson();JsonReaderreader=newJsonReader(newInputStreamReader(in......