Vue自定义指令:clickoutside(点击指定DOM元素外触发)

1,856 阅读1分钟

1. 背景

在业务中写了个tooltip组件,起初需求只是简单的进入页面显示,3s后隐藏。后组件在其他场景中使用时,需要使用者自定义隐藏时机,并在tooltip隐藏时执行回调callback

此时就想到了element中,例如selectpopover等组件都会在用户点击组件外时执行隐藏,于是进行学习研究。

2. 实现指令

使用 vue自定义指令,在指令内 为document绑定touchstart事件

//outside.js

const ctx = "@@clickoutsideContext";

export default {
  bind(el, binding, vnode) {
    const ele = el;
    const documentHandler = (e) => {
      if (!vnode.context || ele.contains(e.target)) {
	    return false;
      }
      // 调用指令回调
      if (binding.expression) {
        vnode.context[el[ctx].methodName](e);
      } else {
        el[ctx].bindingFn(e);
      }
    }
    // 将方法添加到ele
    ele[ctx] = {
      documentHandler,
      methodName: binding.expression,
      bindingFn: binding.value,
    };

	setTimeout(() => {
	  document.addEventListener('touchstart', documentHandler); // 为document绑定事件
	});
  },
  update(el, binding) {
    const ele = el;
    ele[ctx].methodName = binding.expression;
    ele[ctx].bindingFn = binding.value;
  },
  unbind(el) {
	document.removeEventListener('touchstart', el[ctx].documentHandler); // 解绑
	delete el[ctx];
  },
};

3. 使用指令

// demo.vue

<div v-clickoutside="handleOutside"></div>

import outside from './outside';

export default {
    directives: { clickoutside: outside },
    methods: {
        handleOutside(e) {},
    },
};