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,一经查实,立即删除!

相关文章

深入挖掘C语言之——枚举

目录 1. 枚举的定义 2. 枚举常量的赋值 3. 枚举的使用示例 4. 注意事项 在C语言中&#xff0c;枚举&#xff08;Enum&#xff09;是一种用户定义的数据类型&#xff0c;用于定义一组具名的整型常量。枚举常常用于提高代码的可读性和可维护性&#xff0c;使程序更易于理解。…

【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类的子类)类编程…

FDU 2019 | 3. 有向树形态

文章目录 1. 题目描述2. 我的尝试1. 卡特兰数2. 动态规划 1. 题目描述 求 N 个结点能够组成的二叉树的个数。 输入格式 一个整数 N 。 输出格式 输出能组成的二叉树的个数。 数据范围 1 ≤ N ≤ 20 1≤N≤20 1≤N≤20 输入样例 3输出样例 52. 我的尝试 1. 卡特兰数 直接…

分布式锁的详细解释

什么是分布式锁 分布式锁是一种用于协调分布式系统中多个进程或线程之间访问共享资源的机制。在分布式系统中&#xff0c;多个进程或线程可能同时竞争访问某个共享资源&#xff0c;为了避免并发访问导致的数据不一致或冲突&#xff0c;需要使用分布式锁来保证资源的独占性。 分…

5.1.1、【AI技术新纪元:Spring AI解码】Openai chat

OpenAI 聊天 Spring AI 支持由 OpenAI 开发的人工智能语言模型 ChatGPT。ChatGPT 在激发人们对 AI 驱动文本生成的兴趣方面发挥了重要作用,这归功于它创建的行业领先的文本生成模型和嵌入技术。 先决条件 您需要与 OpenAI 创建一个 API 来访问 ChatGPT 模型。在 OpenAI 注册页…

第十三届蓝桥杯省赛CC++ 研究生组

蓝桥杯2022年第十三届省赛真题-裁纸刀 蓝桥杯2022年第十三届省赛真题-灭鼠先锋 蓝桥杯2022年第十三届省赛真题-质因数个数 求个数&#xff0c;则只需要计数即可。求啥算啥&#xff0c;尽量不要搞多余操作 蓝桥杯2022年第十三届省赛真题-选数异或 蓝桥杯2022年第十三届省赛真题…

深入理解并优化Android中的文件描述符(FD)

一、文件描述符&#xff08;FD&#xff09;概述 文件描述符&#xff08;File Descriptor&#xff0c;简称FD&#xff09;是Unix和类Unix操作系统&#xff08;包括Android&#xff09;中的一个关键概念。它是一个非负整数&#xff0c;用于标识操作系统分配的文件或其他输入/输出…

内网渗透学习-环境搭建

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;下方为空白区域。…

el-input添加keyup事件无响应

<el-input type"password" v-model"formData.password" keyup.enter"onSubmit"></el-input>使用 .native修饰符 有时&#xff0c;Vue 组件内部可能会处理某些事件&#xff0c;导致你不能直接在组件上监听这些事件。 在这种情况下&am…

【代码随想录Day28】

Day 28 回溯 Part04 今日任务 93.复原IP地址78.子集90.子集II 代码实现 今天的题出乎意料的简单 复原IP地址&#xff0c;和昨天的分割回文串比较像 一个是分割&#xff0c;一个是判断ip有效 List<String> result new ArrayList<>();List<String> path …

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

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

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

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

从基础入门到学穿C++

前言知识 C简介 C是一门什么样的语言&#xff0c;它与C语言有着什么样的关系&#xff1f; C语言是结构化和模块化的语言&#xff0c;适合处理较小规模的程序。对于复杂的问题&#xff0c;规模较大的程序&#xff0c;需要高度的抽象和建模时&#xff0c;C语言则不合适。为了解…

代理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; 学业成绩是初中分班的首要…

HarmonyOS 网络请求工具库封装,直接无脑用!!!

前言 HarmonyOS 原生网络请求的用法比较麻烦&#xff0c;还是有必要封装下&#xff0c;先看它的原生写法&#xff1a; // 引入包名 import http from ohos.net.http;// 每一个httpRequest对应一个HTTP请求任务&#xff0c;不可复用 let httpRequest http.createHttp(); // 用…

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

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