如何在一个react组件中同时使用forwardRef和useRef

在一个子组件中,有一种场景是子组件用forwardRef包了一层,传到子组件的outerRef需要绑定到子组件上,而与此同时,子组件内部也有一个通过useRef创建的innerRef,也需要绑定到子组件上,咋办呢?我做了几次尝试,如果你想直接参考成功的尝试,请跳转

尝试一

如果是两个useRef,可以这样绑定:

jsx 复制代码
const ref1 = useRef(null);
const ref2 = useRef(null);

return (
    <div ref={(el) => {ref1(el); ref2(el)}}>
        ...
    </div>
);

可是这是forwardedRef,我试了之后不成功。

尝试二

写一个mergeRefs方法:

jsx 复制代码
const mergeRefs = (...refs) => {
    return node => {
      for (const ref of refs) {
        ref.current = node
      }
    }
}

return(
    <div ref={mergeRefs(ref1, ref2)}>
        ...
    </div>
);

对于简单的情景,这样也可以,但我的场景还是不work。

尝试三

也就是优化mergeRefs方法,当ref是个function时,则调用上一个ref:

jsx 复制代码
const setRef = (node, ...refs) => {
    refs.forEach((ref) => {
      if (typeof ref === 'function') {
        ref(node);
      } else if (ref != null) {
        ref.current = node;
      }
    });
}

const mergeRefs = (...refs) => {
    return node => {
      setRef(node, ...refs);
    }
}

return(
    <div ref={mergeRefs(innerRef, forwardedRef)}>
        ...
    </div>
);

成功啦!

结尾

我是在开发带动画效果的draggable list时遇到的这个问题,每一个list item既是droppable的,要绑定dropRef,又是draggable的,要绑定dragRef,又需要从父组件那接收forwardedRef,通过以上第三个尝试,我就可以实现多重功能啦。如果你对这个draggable list感兴趣,可以参考这篇文章:《用react-dnd和react-flip-move做一个带动画的draggable list》

相关推荐
用户52709648744904 分钟前
前端登录菜单加载性能优化总结
前端
你觉得脆皮鸡好吃吗5 分钟前
Check Anti-CSRF Token (AI)
前端·网络·网络协议·安全·csrf·网络安全学习
一个快乐的咸鱼6 分钟前
nextjs接入AI实现流式输出
前端
誰在花里胡哨23 分钟前
Vue<前端页面装修组件>
前端·vue.js
张元清39 分钟前
Pareto 动态路由实战:[slug]、catch-all、嵌套布局
前端·javascript·面试
老王以为41 分钟前
深入理解 AbortController:从底层原理到跨语言设计哲学
javascript·设计模式·node.js
fix一个write十个42 分钟前
NativeWind v4 与 React Native UI Kit或三方库样式隔离指南
前端·react native
懂懂tty1 小时前
React中BeginWork和CompleteWork解析
前端·react.js
_下雨天.1 小时前
HAProxy搭建Web群集
前端
梦想CAD控件1 小时前
在线CAD开发包图纸转换功能使用指南
前端·javascript·vue.js