React 中如何可以实现 vue3.2 新增的 effectScope 特性呢?

2022-07-27 11:51:07 +08:00
 SystemLight

effectScope 可以让不同的组件之间共享相同的依赖存活周期,即可以实现多个组件共享相同状态,并且实现所有的组件都被销毁后触发响应式状态销毁,这样优化了相同的内容重复注册,还共享了数据状态

import {ref, onScopeDispose, effectScope} from 'vue';

export function createSharedComposable(composable) {
    let subscribers = 0
    let state, scope

    const dispose = () => {
        if (scope && --subscribers <= 0) {
            scope.stop()
            state = scope = null
        }
    }

    return (...args) => {
        subscribers++
        if (!state) {
            scope = effectScope(true)
            state = scope.run(() => composable(...args))
        }
        onScopeDispose(dispose)
        return state
    }
}

export let useShareMouse = createSharedComposable(useMouse)

export function useMouse() {
    const x = ref(0)
    const y = ref(0)

    function handler(e) {
        x.value = e.x
        y.value = e.y
    }

    window.addEventListener('mousemove', handler)

    onScopeDispose(() => {
        console.log('onUnmounted')
        window.removeEventListener('mousemove', handler)
    })

    return {x, y}
}

2160 次点击
所在节点    React
3 条回复
SystemLight
2022-07-27 12:02:50 +08:00
突然想到可以用高阶组件配合 context 可以实现,[code]( https://github.com/SystemLight/create-kiva/blob/master/packages/react-kiva/template/kiva/model/index.tsx),但是需要操作上层组件,但是 vue 中不需要操作上层组件就可以实现,根本问题可能是 react hook 无法存在分支,也不能主动触发内部声明周期
mxT52CRuqR6o5
2022-07-27 12:57:08 +08:00
只要放弃官方的 hooks 就能在 react 中实现类似 effectScope 的东西(比如 mobx )
serco
2022-07-27 13:28:24 +08:00
React 实现这种要方便的话基本都是靠生态里的工具,hook 只负责共享代码,加上 jotai 或者 recoil 就可以了。context 太麻烦了。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/868986

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX