当前位置:   article > 正文

那些关于DOM的常见Hook封装(一)_useclickaway

useclickaway

本文是深入浅出 ahooks 源码系列文章的第十四篇,这个系列的目标主要有以下几点:

  • 加深对 React hooks 的理解。

  • 学习如何抽象自定义 hooks。构建属于自己的 React hooks 工具库。

  • 培养阅读学习源码的习惯,工具库是一个对源码阅读不错的选择。

上一篇我们探讨了 ahooks 对 DOM 类 Hooks 使用规范,以及源码中是如何去做处理的。接下来我们就针对关于 DOM 的各个 Hook 封装进行解读。

useEventListener

优雅的使用 addEventListener。

我们先来看看 addEventListener 的定义,以下来自 MDN 文档:

EventTarget.addEventListener() 方法将指定的监听器注册到 EventTarget 上,当该对象触发指定的事件时,指定的回调函数就会被执行。

这里的 EventTarget 可以是一个文档上的元素 Element,Document和Window 或者任何其他支持事件的对象 (比如 XMLHttpRequest)。

我们看 useEventListener 函数 TypeScript 定义,通过类型重载,它对 Element、Document、Window 等元素以及其事件名称和回调参数都做了定义。

  1. function useEventListener<K extends keyof HTMLElementEventMap>(
  2.   eventName: K,
  3.   handler: (ev: HTMLElementEventMap[K]) => void,
  4.   options?: Options<HTMLElement>,
  5. ): void;
  6. function useEventListener<K extends keyof ElementEventMap>(
  7.   eventName: K,
  8.   handler: (ev: ElementEventMap[K]) => void,
  9.   options?: Options<Element>,
  10. ): void;
  11. function useEventListener<K extends keyof DocumentEventMap>(
  12.   eventName: K,
  13.   handler: (ev: DocumentEventMap[K]) => void,
  14.   options?: Options<Document>,
  15. ): void;
  16. function useEventListener<K extends keyof WindowEventMap>(
  17.   eventName: K,
  18.   handler: (ev: WindowEventMap[K]) => void,
  19.   options?: Options<Window>,
  20. ): void;
  21. function useEventListener(eventName: string, handler: noop, optionsOptions): void;

内部代码比较简单:

  • 判断是否支持 addEventListener,支持则将参数进行传递。可以留意注释中的几个参数的作用,当做复习,这里不展开细说。

  • useEffect 的返回逻辑,也就是组件卸载的时候,会自动清除事件监听器,避免产生内存泄露。

  1. function useEventListener(
  2.   // 事件名称
  3.   eventName: string,
  4.   // 处理函数
  5.   handler: noop,
  6.   // 设置
  7.   optionsOptions = {},
  8. ) {
  9.   const handlerRef = useLatest(handler);
  10.   useEffectWithTarget(
  11.     () => {
  12.       const targetElement = getTargetElement(options.target, window);
  13.       if (!targetElement?.addEventListener) {
  14.         return;
  15.       }
  16.       const eventListener = (event: Event) => {
  17.         return handlerRef.current(event);
  18.       };
  19.       // 监听事件
  20.       targetElement.addEventListener(eventName, eventListener, {
  21.         // listener 会在该类型的事件捕获阶段传播到该 EventTarget 时触发。
  22.         capture: options.capture,
  23.         // listener 在添加之后最多只调用一次。如果是 true,listener 会在其被调用之后自动移除。
  24.         once: options.once,
  25.         // 设置为 true 时,表示 listener 永远不会调用 preventDefault() 。如果 listener 仍然调用了这个函数,客户端将会忽略它并抛出一个控制台警告
  26.         passive: options.passive,
  27.       });
  28.       // 移除事件
  29.       return () => {
  30.         targetElement.removeEventListener(eventName, eventListener, {
  31.           capture: options.capture,
  32.         });
  33.       };
  34.     },
  35.     [eventName, options.capture, options.once, options.passive],
  36.     options.target,
  37.   );
  38. }

useClickAway

监听目标元素外的点击事件。

提到这个的应用场景,应该是模态框,点击外部阴影部分,自动关闭的场景。那这里它是怎么实现的呢?

首先它支持传递 DOM 节点或者 Ref,并且是支持数组方式。事件默认是支持 click,开发者可以自行传递并支持数组方式。

  1. export default function useClickAway<T extends Event = Event>(
  2.   // 触发函数
  3.   onClickAway: (event: T) => void,
  4.   // DOM 节点或者 Ref,支持数组
  5.   target: BasicTarget | BasicTarget[],
  6.   // 指定需要监听的事件,支持数组
  7.   eventName: string | string[] = 'click',
  8. ) {
  9. }

然后内部通过 document.addEventListener 监听事件。组件卸载的时候清除事件监听。

  1. // 事件列表
  2. const eventNames = Array.isArray(eventName) ? eventName : [eventName];
  3. // document.addEventListener 监听事件,通过事件代理的方式知道目标节点
  4. eventNames.forEach((event) => document.addEventListener(event, handler));
  5. return () => {
  6.   eventNames.forEach((event) => document.removeEventListener(event, handler));
  7. };

最后看 handler 函数,通过 event.target 获取到触发事件的对象 (某个 DOM 元素) 的引用,判断假如不在传入的 target 列表中,则触发定义好的 onClickAway 函数。

  1. const handler = (event: any=> {
  2.   const targets = Array.isArray(target) ? target : [target];
  3.   if (
  4.     // 判断点击的 DOM Target 是否在定义的 DOM 元素(列表)中
  5.     targets.some((item) => {
  6.       const targetElement = getTargetElement(item);
  7.       return !targetElement || targetElement.contains(event.target);
  8.     })
  9.   ) {
  10.     return;
  11.   }
  12.   // 触发点击事件
  13.   onClickAwayRef.current(event);
  14. };

小结一下,useClickAway 就是使用了事件代理的方式,通过 document 监听事件,判断触发事件的 DOM 元素是否在 target 列表中,从而决定是否要触发定义好的函数。

useEventTarget

常见表单控件(通过 e.target.value 获取表单值) 的 onChange 跟 value 逻辑封装,支持自定义值转换和重置功能。

直接看代码,比较简单,其实就是监听表单的 onChange 事件,拿到值后更新 value 值,更新的逻辑支持自定义。

  1. function useEventTarget<T, U = T>(options?: Options<T, U>) {
  2.   const { initialValue, transformer } = options || {};
  3.   const [value, setValue] = useState(initialValue);
  4.   // 自定义转换函数
  5.   const transformerRef = useLatest(transformer);
  6.   const reset = useCallback(() => setValue(initialValue), []);
  7.   const onChange = useCallback((e: EventTarget<U>=> {
  8.     // 获取 e.target.value 的值,并进行设置
  9.     const _value = e.target.value;
  10.     if (isFunction(transformerRef.current)) {
  11.       return setValue(transformerRef.current(_value));
  12.     }
  13.     // no transformer => U and T should be the same
  14.     return setValue(_value as unknown as T);
  15.   }, []);
  16.   return [
  17.     value,
  18.     {
  19.       onChange,
  20.       reset,
  21.     },
  22.   ] as const;
  23. }

useTitle

用于设置页面标题。

这个页面标题指的是浏览器 Tab 中展示的。通过 document.title 设置。

代码非常简单,一看就会:

  1. function useTitle(title: stringoptionsOptions = DEFAULT_OPTIONS) {
  2.   const titleRef = useRef(isBrowser ? document.title : '');
  3.   useEffect(() => {
  4.     document.title = title;
  5.   }, [title]);
  6.   useUnmount(() => {
  7.     // 组件卸载后,恢复上一次的 title
  8.     if (options.restoreOnUnmount) {
  9.       document.title = titleRef.current;
  10.     }
  11.   });
  12. }

useFavicon

设置页面的 favicon。

favicon 指的是页面 Tab 的这个 ICON。

原理是通过 link 标签设置 favicon。

  1. const useFavicon = (href: string=> {
  2.   useEffect(() => {
  3.     if (!href) return;
  4.     const cutUrl = href.split('.');
  5.     const imgSuffix = cutUrl[cutUrl.length - 1].toLocaleUpperCase() as ImgTypes;
  6.     const link: HTMLLinkElement =
  7.       document.querySelector("link[rel*='icon']") || document.createElement('link');
  8.     // 用于定义链接的内容的类型。
  9.     link.type = ImgTypeMap[imgSuffix];
  10.     // 指定被链接资源的URL。
  11.     link.href = href;
  12.     // 此属性命名链接文档与当前文档的关系。
  13.     link.rel = 'shortcut icon';
  14.     document.getElementsByTagName('head')[0].appendChild(link);
  15.   }, [href]);
  16. };
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/275061
推荐阅读
相关标签
  

闽ICP备14008679号