resize-observer源码解读

resize-observer

github 地址:https://github.com/devrelm/resize-observer

本地启动

npm installnpm start

node 18.16.0 (npm 9.5.1) 启动失败报错

node:internal/crypto/hash:71this[kHandle] = new _Hash(algorithm, xofLen);^Error: error:0308010C:digital envelope routines::unsupported

解决:更改 node 版本

node 16.16.0 (npm 8.11.0) 启动成功

在这里插入图片描述

使用示例

const onResize: ResizeObserverProps["onResize"] = ({width,height,offsetHeight,offsetWidth,
}) => {setTimes((prevTimes) => prevTimes + 1);console.log("Resize:","\n","BoundingBox",width,height,"\n","Offset",offsetWidth,offsetHeight);
};<ResizeObserver onResize={onResize} disabled={disabled}><Wrapper><textarea ref={textareaRef} placeholder="I'm a textarea!" /></Wrapper>
</ResizeObserver>;
export type OnResize = (size: SizeInfo, element: HTMLElement) => void;export interface ResizeObserverProps {/** Pass to ResizeObserver.Collection with additional data */data?: any;children:| React.ReactNode| ((ref: React.RefObject<any>) => React.ReactElement);disabled?: boolean;/** Trigger if element resized. Will always trigger when first time render. */onResize?: OnResize;
}

ResizeObserve 組件

真正组件在ResizeObserver组件

const RefResizeObserver = React.forwardRef(ResizeObserver) as React.ForwardRefExoticComponent<React.PropsWithoutRef<ResizeObserverProps> & React.RefAttributes<any>
> & {Collection: typeof Collection;
};RefResizeObserver.Collection = Collection;export default RefResizeObserver;

ResizeObserver

里面还有一层组件SingleObserver

//src\index.tsx
function ResizeObserver(props: ResizeObserverProps, ref: React.Ref<HTMLElement>) {return childNodes.map((child, index) => {const key = child?.key || `${INTERNAL_PREFIX_KEY}-${index}`;return (<SingleObserver {...props} key={key} ref={index === 0 ? ref : undefined}>{child}</SingleObserver>);}) as any as React.ReactElement;
}

SingleObserver 組件

真正实现观察的方法在这个组件

const RefSingleObserver = React.forwardRef(SingleObserver);
//src\SingleObserver\index.tsx
function SingleObserver(props: SingleObserverProps,ref: React.Ref<HTMLElement>
) {return (<DomWrapper ref={wrapperRef}>{canRef? React.cloneElement(mergedChildren as any, {ref: mergedRef,}): mergedChildren}</DomWrapper>);
}

实现元素变化逻辑

监听 elementRef.current 的變化
在 SingleObserver 组件

import { observe, unobserve } from "../utils/observerUtil";// Dynamic observe
React.useEffect(() => {// getDom获取要被侦听的elementconst currentElement: HTMLElement = getDom();if (currentElement && !disabled) {// 执行侦听observe(currentElement, onInternalResize);}// 清除侦听return () => unobserve(currentElement, onInternalResize);
}, [elementRef.current, disabled]);

创建侦听器实例

// src\utils\observerUtil.ts
const elementListeners = new Map<Element, Set<ResizeListener>>();
import ResizeObserver from 'resize-observer-polyfill';// interface ResizeObserverEntry {
//     readonly target: Element;
//     readonly contentRect: DOMRectReadOnly;
// }// onResize 创建侦听器传入的callback
function onResize(entities: ResizeObserverEntry[]) {entities.forEach((entity) => {const { target } = entity;// elementListeners.get(target)是set集合 ,listener是回调函数onInternalResizeelementListeners.get(target)?.forEach((listener) => listener(target));});
}// Note: ResizeObserver polyfill not support option to measure border-box resize
const resizeObserver = new ResizeObserver(onResize);
// resize-observer-polyfill中ResizeObserverSPI类
const observer = new ResizeObserverSPI(callback, controller, this);

ResizeObserverSPI类的broadcastActive方法
callback返回的信息,entries是一个数组,返回所有正在活跃的目标element列表

// Create ResizeObserverEntry instance for every active observation.
const entries = this.activeObservations_.map((observation) => {// 返回被觀察element最新的大小return new ResizeObserverEntry(observation.target,// 執行observation.broadcastRect函數獲取最新的大小observation.broadcastRect());
});// 改变回调函数的this指向ctx
this.callback_.call(ctx, entries, ctx);

observe 函数

const elementListeners = new Map<Element, Set<ResizeListener>>();function observe(element: Element, callback: ResizeListener) {if (!elementListeners.has(element)) {// 给elementListeners添加一个键值对elementListeners.set(element, new Set());//resizeObserver.observe(element);}
// elementListeners.get(element) 是set结构,给set插入一个新元素callback回调函数即onInternalResizeelementListeners.get(element).add(callback);
}

unobserve 函数

const elementListeners = new Map<Element, Set<ResizeListener>>();// 取消侦听
function unobserve(element: Element, callback: ResizeListener) {if (elementListeners.has(element)) {//set集合移除callback回调函数elementListeners.get(element).delete(callback);if (!elementListeners.get(element).size) {// 取消侦听resizeObserver.unobserve(element);// 移除目标elementelementListeners.delete(element);}}
}

onInternalResize 函数

CollectionContext = React.createContext<onCollectionResize>(null);
const onCollectionResize = React.useContext(CollectionContext);

const propsRef = React.useRef < SingleObserverProps > props;
propsRef.current = props;// Handler
const onInternalResize = React.useCallback((target: HTMLElement) => {const { onResize, data } = propsRef.current;// getBoundingClientRect侦听器内部实现的一个方法,获取元素尺寸大小const { width, height } = target.getBoundingClientRect();const { offsetWidth, offsetHeight } = target;/*** Resize observer trigger when content size changed.* In most case we just care about element size,* let's use `boundary` instead of `contentRect` here to avoid shaking.*/const fixedWidth = Math.floor(width);const fixedHeight = Math.floor(height);if (sizeRef.current.width !== fixedWidth ||sizeRef.current.height !== fixedHeight ||sizeRef.current.offsetWidth !== offsetWidth ||sizeRef.current.offsetHeight !== offsetHeight) {const size = {width: fixedWidth,height: fixedHeight,offsetWidth,offsetHeight,};sizeRef.current = size;// IE is strange, right?const mergedOffsetWidth =offsetWidth === Math.round(width) ? width : offsetWidth;const mergedOffsetHeight =offsetHeight === Math.round(height) ? height : offsetHeight;const sizeInfo = {...size,offsetWidth: mergedOffsetWidth,offsetHeight: mergedOffsetHeight,};// Let collection know what happenedonCollectionResize?.(sizeInfo, target, data);if (onResize) {// defer the callback but not defer to next framePromise.resolve().then(() => {// 给父组件传递信息onResize(sizeInfo, target);});}}
}, []);

getDom 函數

const getDom = () =>findDOMNode<HTMLElement>(elementRef.current) ||// Support `nativeElement` format(elementRef.current && typeof elementRef.current === 'object'? findDOMNode<HTMLElement>((elementRef.current as any)?.nativeElement): null) ||findDOMNode<HTMLElement>(wrapperRef.current);

findDOMNode函數

github:https://github.com/react-component/util/blob/master/src/Dom/findDOMNode.ts

/*** Return if a node is a DOM node. Else will return by `findDOMNode`*/
function findDOMNode<T = Element | Text>(node: React.ReactInstance | HTMLElement | SVGElement,
): T {if (isDOM(node)) {return (node as unknown) as T;}if (node instanceof React.Component) {return (ReactDOM.findDOMNode(node) as unknown) as T;}return null;
}
function isDOM(node: any): node is HTMLElement | SVGElement {// https://developer.mozilla.org/en-US/docs/Web/API/Element// Since XULElement is also subclass of Element, we only need HTMLElement and SVGElementreturn node instanceof HTMLElement || node instanceof SVGElement;
}

Collection组件

function Collection({ children, onBatchResize }: CollectionProps) {const resizeIdRef = React.useRef(0);const resizeInfosRef = React.useRef<ResizeInfo[]>([]);const onCollectionResize = React.useContext(CollectionContext);const onResize = React.useCallback<onCollectionResize>((size, element, data) => {resizeIdRef.current += 1;const currentId = resizeIdRef.current;resizeInfosRef.current.push({size,element,data,});Promise.resolve().then(() => {if (currentId === resizeIdRef.current) {onBatchResize?.(resizeInfosRef.current);resizeInfosRef.current = [];}});// Continue bubbling if parent existonCollectionResize?.(size, element, data);},[onBatchResize, onCollectionResize],);return <CollectionContext.Provider value={onResize}>{children}</CollectionContext.Provider>;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/756082.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Py/Java/C++三种语言OD2023C卷真题】20天拿下华为OD笔试之【模拟】2023C-攀登者2【欧弟算法】全网注释最详细分类最全的华为OD真题题解

有LeetCode算法/华为OD考试扣扣交流群可加 948025485 可上全网独家的 欧弟OJ系统 练习华子OD、大厂真题 绿色聊天软件戳 od1336了解算法冲刺训练 文章目录 题目描述与示例题目描述输入描述输出描述 示例输入输出 解题思路原路返回和非原路返回原路返回走过的总路程从左边空地出…

流畅的Python(二十一)-类元编程

一、核心要义 1.类元编程时指在运行时创建或定制类的技艺 2.类是一等对象,因此任何时候都可以使用函数新建类&#xff0c;而无需使用class关键字 3.类装饰器也是函数&#xff0c;不过能够审查、修改&#xff0c;甚至把被装饰的类替换为其它类。 4.元类(type类的子类)类编程…

内网渗透学习-环境搭建

1、环境搭建测试 虚拟机网络环境配置&#xff0c;模拟外网和内网 主机操作系统网络内网ip外网ip物理主机window10vmnet8192.168.70.1攻击机kali Linuxvmnet8192.168.70.134域控主机win server 2008 r2vmnet0192.168.52.138域成员主机win server 2k3vmnet0192.168.52.141服务器…

HarmonyOS NEXT应用开发之swiper指示器导航点位于swiper下方

介绍 本示例介绍通过分割swiper区域&#xff0c;实现指示器导航点位于swiper下方的效果。 效果预览图 使用说明 加载完成后swiper指示器导航点&#xff0c;位于显示内容下方。 实现思路 将swiper区域分割为两块区域&#xff0c;上方为内容区域&#xff0c;下方为空白区域。…

【数据结构】链表力扣刷题详解

前言 题目链接 移除链表元素 链表的中间结点 反转链表 分割链表 环形链表的约瑟夫问题 ​ 欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 移除链表元素 题述 给你一个链表的头节点 head 和一个整数 val &#xff0c;请…

【Spring Cloud Gateway】路由配置uri三种方式及区别

websocket配置方式 ws:// 或 wss:// 开头的 URI&#xff0c;表示配置的是支持 Websocket 协议的目标地址。 这种方式适用于需要与客户端建立长连接、实现双向通信的场景&#xff0c;比如实时消息推送、即时聊天等。 使用 Websocket 配置方式可以让 Spring Cloud Gateway 能够…

代理IP品质对Tik Tok代理的重要性

随着Tik Tok的迅速崛起&#xff0c;越来越多的人开始关注如何透过Tik Tok进行行销和推广。其中&#xff0c;使用Tik Tok代理程式是常见的方法。 然而&#xff0c;在选择和使用代理时&#xff0c;IP品质是一个不可忽视的因素。本文将探讨IP品质对Tik Tok代理的重要性&#xff0…

鸿蒙Harmony应用开发—ArkTS声明式开发(容器组件:GridRow)

栅格布局可以为布局提供规律性的结构&#xff0c;解决多尺寸多设备的动态布局问题&#xff0c;保证不同设备上各个模块的布局一致性。 栅格容器组件&#xff0c;仅可以和栅格子组件(GridCol)在栅格布局场景中使用。 说明&#xff1a; 该组件从API Version 9开始支持。后续版本…

初中分班怎么分按什么标准

初中是学生生涯中一个重要的阶段&#xff0c;它承接着小学的基础教育&#xff0c;同时为高中的深入学习打下基础。在这个关键时期&#xff0c;分班成为学校、家长和学生共同关注的焦点。那么&#xff0c;初中分班是按照什么标准来进行的呢&#xff1f; 学业成绩是初中分班的首要…

考研复习C语言进阶(4)

1. 为什么存在动态内存分配 我们已经掌握的内存开辟方式有&#xff1a; int val 20;//在栈空间上开辟四个字节 char arr[10] {0};//在栈空间上开辟10个字节的连续空间 但是上述的开辟空间的方式有两个特点&#xff1a; 1. 空间开辟大小是固定的。 2. 数组在申明的时候&#…

Docker知识--01

虚拟化 # 什么是虚拟化 在计算机中&#xff0c;虚拟化&#xff08;英语&#xff1a;Virtualization&#xff09;是一种资源管理技术&#xff0c;是将计算机的各种实体资源&#xff0c;如服务器、网络、内存及存储等&#xff0c;予以抽象、转换后呈现出来&#xff0c;打…

深度学习模型参数的计算

1. 模型尺寸与参数量 根据神经网络模型中的参数数量估计模型体积的大小涉及多个考虑因素&#xff0c;包括参数的数据类型、每种数据类型的存储要求以及存储模型结构或元数据的任何额外开销。下面是估计模型大小的一般方法&#xff1a; 1.1.数据类型 神经网络参数通常存储为浮…

Vue3使用高德地图(3分钟快速上手)

一.、在高德开发平台注册账号 高德开放平台 | 高德地图API (amap.com) 二、我的 > 管理管理中添加Key 三、安装依赖 npm i amap/amap-jsapi-loader --save 四、创建一个放置地图的容器 <template><div class"container"></div> </template…

网工内推 | 信息安全主管,CISP认证优先,最高25K,加绩效提成

01 福建省数字福建云计算运营有限公司 招聘岗位&#xff1a;网络及信息安全主管 职责描述&#xff1a; 1.负责带领IT运维团队&#xff0c;对公司网络与安全整体架构规划&#xff0c;设计、运营及IT设备的全生命周期运维规划等&#xff1b; 2.负责对网络、网络安全的日常运维管…

C++中虚表是什么

定义 虚函数&#xff08;Virtual Function&#xff09; 定义&#xff1a;类中使用virtual 关键字修饰的函数 叫做虚函数 语法&#xff1a; class Base { public:virtual void show() { cout << "Base show" << endl; } }; 虚函数表&#xff08;Virtual…

印刷企业实施MES管理系统如何做好需求分析

在数字化、信息化的大潮中&#xff0c;印刷企业面临着转型升级的迫切需求。MES管理系统作为连接企业资源计划ERP和现场自动化系统的桥梁&#xff0c;对于提升印刷企业的生产效率、优化资源配置、提高产品质量具有重要意义。因此&#xff0c;做好MES管理系统的需求分析&#xff…

多线程(部分)

Day28下2 多线程 一、什么是进程 进程是系统进行资源分配和调用的独立单元&#xff0c;每一个进程都有它的独立内存空间和系统资源。 二、单进程操作系统和多进程操作系统的区别 单进程操作系统&#xff1a;dos&#xff08;一瞬间只能执行一个任务&#xff09; 多进程单用户操…

equals与时间序列攻击

引言 随着信息技术的迅速发展&#xff0c;网络安全和隐私问题变得愈发重要。黑客和攻击者不断寻找新的攻击方法&#xff0c;其中之一是时间序列攻击&#xff08;Timing Attack&#xff09;。时间序列攻击是一种侧信道攻击&#xff0c;攻击者试图通过测量程序的执行时间来推断程…

Vue3语法插件Volar在vsCode中搜不到,Volar正式更名为Vue-Official

问题描述 今天拿Vue3的cli练手&#xff0c;之前用的语法插件是Vue2的Vetur&#xff0c;对于Vue3来说该插件不能完美匹配了&#xff0c;所以就想切换回Volar&#xff0c;结果万万没想到&#xff0c;找不着了&#xff1f;&#xff1f;&#xff1f;&#xff01;&#xff01;&…

git stash clear/drop 后如何恢复

git stash clear/drop 后代码如何恢复 事故经过 切换分支前有修改未提交的代码&#xff0c;使用 git stash 存储了当前的代码切换分支再返回自己开发的分支本来要进行 git stash pop 操作&#xff0c;然后 git stash list 发现有好几个 stash记录于是想清除没用的 stash 记录…