如何在一个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》

相关推荐
乐兮创想 小林13 分钟前
企业官网移动端性能优化实战:从 Core Web Vitals 到图片/CDN/响应式的工程清单
前端·性能优化·网站建设·北京网站建设公司
疯狂SQL35 分钟前
JWT 在线解码、验签、生成一篇讲透:附前端实现、工具架构与在线体验地址
javascript·jwt·编解码·jwt测试
前端一小卒1 小时前
不手写代码的第 30 天,我才明白前端这个岗位还剩什么
前端·javascript·ai编程
Ajie'Blog1 小时前
Copilot Agent Tasks API 开放:AI 编程开始进入后台任务时代
服务器·前端·javascript·人工智能·copilot·ai编程
老毛肚1 小时前
jeecgboot vue TS & 模板化 04
前端·javascript·vue.js
晓13131 小时前
【Cocos Creator 2.x】篇——第二章 入门
javascript·游戏引擎
AI_零食2 小时前
鸿蒙PC Electron跨平台应用开发:24时区时间表应用详解
前端·华为·electron·开源·harmonyos·鸿蒙
Electrolux3 小时前
[onlyoffice-v9]纯前端怎么实现编辑预览office
前端·javascript·github
VidDown3 小时前
Webhook 调试器:让第三方回调“原形毕露”
java·开发语言·javascript·编辑器·postman
码云之上3 小时前
聊聊如何设计一个高效、稳定的 Node.js 接入层
前端·后端·node.js