1. 背景
在业务中写了个tooltip
组件,起初需求只是简单的进入页面显示,3s后隐藏。后组件在其他场景中使用时,需要使用者自定义隐藏时机,并在tooltip
隐藏时执行回调callback
。
此时就想到了element
中,例如select
、popover
等组件都会在用户点击组件外时执行隐藏,于是进行学习研究。
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) {},
},
};