1. 前言

在实际开发时会有这样的需求:点击某一个按钮出现一个弹窗,然后点弹窗的其他区域时需要关闭弹窗,如果是点击的弹窗本身,除非是关闭操作,否则不关闭弹窗。本文将简单介绍几种这种需求的几种实现方案。

2. 实现

2.1. 通过失焦事件来实现

首先,失焦事件并不是表单元素的专利,通过添加tabindex可以可以让任意一个元素获取失焦事件。

什么是 tabindex

tabindex 属性可以有三个值:

  • tabindex="0" :元素可以获得焦点,并且可以通过键盘导航(通常是Tab键)到达。
  • tabindex="-1" :元素可以获得焦点,但不会通过键盘导航到达。这通常用于脚本中通过编程方式设置焦点。
  • tabindex 属性:元素默认不会获得焦点。

通过为div添加tabindex,我们可以很方便的判断是否点击了非div及其内部本元素:

2.2. 通过蒙版

其实蒙版是现在比较主流的一种方法,很多组件库都是用的这个方法,本质就是在要监测的元素下面加个蒙版元素,通过为蒙版添加点击事件来判断用户点击了外界。

这个方法有个缺点就是,对外部的第一次点击时无效的,不会触发外部元素的点击事件,也就说用户想要点击外部的某个按钮需要点击两次。

2.3. 直接遍历元素

众所周知,html的元素的结构为树结构,那么我们就可以通过对树的遍历来判断点击的这个元素是否在窗口中。

简单实现一下:

但是这样的代码太不优雅了,可以封装一个指令来优化:

 1import { Directive } from "vue";
 2const wm = new WeakMap<HTMLElement, Function>();
 3
 4const dfs = (ele: HTMLElement, target: any): boolean => {
 5  if (ele === target) {
 6    return true;
 7  }
 8
 9  return [...(ele.children || [])].some((item) => dfs(item as HTMLElement, target));
10};
11
12const vClickOut: Directive = {
13  mounted(box: HTMLElement, { value: cb }) {
14    const onClick = (et: Event) => {
15      const target = et.target;
16      const isClickOut = !dfs(box, target);
17      if (isClickOut && cb && typeof cb === "function") {
18        cb();
19      }
20    };
21    wm.set(box, onClick);
22    window.addEventListener("click", onClick);
23  },
24  beforeUnmount(el: HTMLElement) {
25    window.removeEventListener("click", wm.get(el) as any);
26  },
27};
28export default vClickOut;

使用:

2.4. 借助DOM API 中的contains方法

其实DOM API中有一个Node.contains() 方法,这个方法返回一个布尔值,表示一个节点是否是给定节点的后代,即该节点本身、其直接子节点(childNodes)、子节点的直接子节点等。

简单写个demo:

 1import { useEffect, useRef, useState } from "react";
 2import styles from "./index.module.scss";
 3
 4export default function TestPage() {
 5  const ref = useRef<HTMLDivElement>();
 6
 7  useEffect(() => {
 8    const onClick = (e: PointerEvent) => {
 9      const clickInner = ref.current.contains(e.target as HTMLElement);
10      console.log(clickInner ? "点击了内部" : "点击了外部");
11    };
12    window.addEventListener("click", onClick);
13    return () => {
14      window.removeEventListener("click", onClick);
15    };
16  }, []);
17
18  return (
19    <div className={styles["test-page-container"]}>
20      <div
21        ref={(el) => (ref.current = el as HTMLDivElement)}
22        style={{
23          width: 500,
24          height: 500,
25          backgroundColor: "ButtonFace",
26        }}
27        >
28        <div
29          style={{
30            width: 100,
31            height: 100,
32            backgroundColor: "red",
33          }}
34          >
35          test
36        </div>
37      </div>
38    </div>
39  );
40}

这个方法的兼容性也是相当的好:

3. 评论

直接监听全局的点击事件就行了,只要不是当前需要点击的元素就都是元素之外

直接 判断的元素.contains(e.target) 不就完事了

也可以监听最外层的根节点,通过冒泡来实现。
比如 ahooks 的 use-click-away

个人笔记记录 2021 ~ 2025