v-clickoutside的源码分析及改进

1.前言

最近遇到一个需求,公司需要做一个类似facebook的搜索选择组件,我打算用el-input和el-cascader-panel结合设计。在设计过程中参考了el-cascader的源码,其中的v-clickoutside自定义指令蛮值得研究的,所以写篇文章记录下。

2.基础知识点

(1)v-directive

vue文档已经写的很清楚了,这里只贴网址:vue中文文档-自定义指令

(2)v-clickoutside

先放element-ui中的源码

//element-ui/src/utils/clickoutside.js
import Vue from 'vue';
import { on } from 'element-ui/src/utils/dom';

const nodeList = [];
const ctx = '@@clickoutsideContext';

let startClick;
let seed = 0;

!Vue.prototype.$isServer && on(document, 'mousedown', e => (startClick = e));

!Vue.prototype.$isServer && on(document, 'mouseup', e => {
  nodeList.forEach(node => node[ctx].documentHandler(e, startClick));
});

function createDocumentHandler(el, binding, vnode) {
  return function(mouseup = {}, mousedown = {}) {
    if (!vnode ||
      !vnode.context ||
      !mouseup.target ||
      !mousedown.target ||
      el.contains(mouseup.target) ||
      el.contains(mousedown.target) ||
      el === mouseup.target ||
      (vnode.context.popperElm &&
      (vnode.context.popperElm.contains(mouseup.target) ||
      vnode.context.popperElm.contains(mousedown.target)))) return;

    if (binding.expression &&
      el[ctx].methodName &&
      vnode.context[el[ctx].methodName]) {
      vnode.context[el[ctx].methodName]();
    } else {
      el[ctx].bindingFn && el[ctx].bindingFn();
    }
  };
}

/**
 * v-clickoutside
 * @desc 点击元素外面才会触发的事件
 * @example
 * ```vue
 * <div v-element-clickoutside="handleClose">
 * ```
 */
export default {
  bind(el, binding, vnode) {
    nodeList.push(el);
    const id = seed++;
    el[ctx] = {
      id,
      documentHandler: createDocumentHandler(el, binding, vnode),
      methodName: binding.expression,
      bindingFn: binding.value
    };
  },

  update(el, binding, vnode) {
    el[ctx].documentHandler = createDocumentHandler(el, binding, vnode);
    el[ctx].methodName = binding.expression;
    el[ctx].bindingFn = binding.value;
  },

  unbind(el) {
    let len = nodeList.length;

    for (let i = 0; i < len; i++) {
      if (nodeList[i][ctx].id === el[ctx].id) {
        nodeList.splice(i, 1);
        break;
      }
    }
    delete el[ctx];
  }
};
//element-ui/src/utils/dom
export const on = (function() {
  if (!isServer && document.addEventListener) {
    return function(element, event, handler) {
      if (element && event && handler) {
        element.addEventListener(event, handler, false);
      }
    };
  } else {
    return function(element, event, handler) {
      if (element && event && handler) {
        element.attachEvent('on' + event, handler);
      }
    };
  }
})();

v-clickoutside用于处理目标节点外的点击事件,最主要的是下拉框等展开内容的关闭。

这道程序使用了订阅发布的设计模式,先通过自定义的on方法(兼容addEventListener和attachEvent两个绑定方法的函数),在document中绑定mouseup和mousedown事件,两个事件分别记录鼠标按下和松开时所在的节点,之后与目标节点进行对比,如果点击元素是目标节点或者被目标节点包含在内,则触发对应的执行函数。

在执行bind周期函数时,先把该元素存放在nodeList中,且在元素中赋值一个名为“@@clickoutsideContext”的属性,这个对象里面分别存放id(用于标记元素,方便unbind周期函数中把元素从nodeList中剔除),documentHandler(存放执行函数),methodName(binding.expression),bindingFn(binding.value)。

当执行到update周期函数时,会对el中的ctx属性进行更新。执行unbind函数时,会根据el中的ctx属性中记录的id去剔除本对象。

createDocumentHandler函数会返回一个能通过闭包访问到mouseup(鼠标按下时所在的元素)和mousedown(鼠标松开时所在的元素)的函数,该函数执行时会按照以下条件判断v-clickoutside中绑定的函数是否执行:

  1. !vnode || !vnode.context || !mouseup.target || !mousedown.target :判断vnode和vnode.context等目标是否存在
  2. el.contains(mouseup.target) || el.contains(mousedown.target) || el === mouseup.target:判断当前目标节点是鼠标松开时所在的节点,或者是否包含鼠标点击或者鼠标松开时所在的元素
  3. (vnode.context.popperElm &(vnode.context.popperElm.contains(mouseup.target) || vnode.context.popperElm.contains(mousedown.target))):判断虚拟节点vnode中的popperElm,也就是是否存在悬浮的组件上。

当符合上面任一条件时,函数就会return,否则执行所绑定目标节点中存放在名为“@@clickoutsideContext”属性的执行方法。

3.源码的改进

分析:个人觉得el[ctx]这个设计的存在可能是便于其他作用域访问当前el所绑定的执行函数而设计的。基于我的情况,主要围绕把el[ctx]为出发点,把nodeList改成了以Map的(key,value)作保存,改进如下:

let clickInEvent
let nodeEventRecorder = new Map()

document.addEventListener('mousedown', e => (clickInEvent = e))
document.addEventListener('mouseup', e => {
  nodeEventRecorder.forEach((value) => {
    value(e, clickInEvent)
  })
})

function createHandler (el, binding, vnode) {
  return function (mouseup = {}, mousedown = {}) {
    if (!vnode ||
      !vnode.context ||
      !mouseup.target ||
      el.contains(mouseup.target) ||
      el.contains(mousedown.target) ||
      el === mouseup.target ||
      (vnode.context.popperElm &&
        (
          vnode.context.popperElm.contains(mouseup.target) ||
          vnode.context.popperElm.contains(mousedown.target)
        )
      )
    ) {
      return
    }
    if (binding.expression && vnode.context[binding.expression]) {
      vnode.context[binding.expression]()
    } else {
      if (typeof (binding.value) === 'function') {
        binding.value()
      } else {
        throw new Error('value should be a function')
      }
    }
  }
}

let directive = {
  bind (el, binding, vnode) {
    nodeEventRecorder.set(el, createHandler(el, binding, vnode))
  },
  update (el, binding, vnode) {
    nodeEventRecorder.set(el, createHandler(el, binding, vnode))
  },
  unbind (el) {
    nodeEventRecorder.delete(el)
  }
}

有人会问为什么不用Object作为储存映射的容器类型,那是因为Object的key只能是字符串或者Symbol类型的数据。而Map的Key可以是任何数据类型,包括节点。

用Map替换后也不存在el[ctx]和seed++这类变量了。在unbind中执行删除操作时也不需要遍历列表这么麻烦了。

实现的效果如下:

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: v-clickoutsideVue.js 框架中的一个指令,用于在点击元素外部时触发绑定的事件。它可以用于实现点击其他地方关闭弹出框等功能。在使用 v-clickoutside 指令时,需要将其绑定到一个具有相应事件的元素上,并将要触发的事件作为参数传入。当用户在元素外部点击时,该事件就会被触发。 ### 回答2: v-clickoutside是一个在Vue.js中用于处理点击元素外部事件的自定义指令。这个指令可以在需要监听点击元素外部的情况下使用,比如下拉菜单或者模态框等。 使用v-clickoutside非常简单,只需要将该指令应用在需要监听的元素上即可。当别的元素被点击时,会触发指定的回调函数。这样就可以方便地处理点击元素外部的事件,比如关闭下拉菜单或者模态框。 为了更好地理解v-clickoutside的原理,下面是一个示例: 在Vue组件中,我们有一个下拉菜单,我们希望当点击菜单以外的区域时,菜单会关闭。 我们可以在菜单的根元素上添加v-clickoutside指令,然后定义一个对应的事件处理函数。当点击菜单以外的区域时,这个事件处理函数会被调用。 在Vue组件中,我们可以这样使用v-clickoutside指令: ```html <template> <div v-clickoutside="closeMenu"> <button @click="toggleMenu">打开菜单</button> <ul v-show="isOpen"> <li>选项1</li> <li>选项2</li> <li>选项3</li> </ul> </div> </template> <script> export default { data() { return { isOpen: false }; }, methods: { toggleMenu() { this.isOpen = !this.isOpen; }, closeMenu() { this.isOpen = false; } } }; </script> ``` 在上面的代码中,v-clickoutside指令被应用在包含菜单的div元素上。当点击div元素以外的区域时,会调用closeMenu方法,从而关闭菜单。 总结来说,v-clickoutside是一个用于处理点击元素外部事件的Vue.js自定义指令。通过应用这个指令,我们可以方便地监听并处理点击元素外部的事件,从而实现某些功能,比如关闭下拉菜单或者模态框。 ### 回答3: v-clickoutside是一个Vue.js指令,用于在点击元素外部时执行特定的操作。在Vue项目中使用v-clickoutside可以监听整个页面的点击事件,并判断点击的目标是否在指定的元素外部。 使用v-clickoutside指令有两个步骤。首先,在Vue组件中,需要导入v-clickoutside指令的定义,可以通过在组件中引入import语句来完成。其次,需要在需要使用v-clickoutside的元素上添加v-clickoutside指令,并将需要执行的方法作为参数传入该指令。 当点击元素外部时,v-clickoutside指令会触发绑定的方法,并将点击的事件对象作为参数传入执行的方法中。通过这种方式,我们可以在点击元素外部时执行特定的操作,比如关闭下拉菜单、弹出框等。 v-clickoutside的实现原理是通过在Vue组件的生命周期钩子函数中,将元素绑定的点击事件进行监听,并通过事件委托的方式判断事件点击的目标是否在指定的元素外部。如果是,则执行绑定的方法。 v-clickoutside的使用可以大大简化我们在处理点击元素外部事件时的代码逻辑,提高代码的可读性和可维护性。同时,它也提供了一个更好的用户体验,当用户点击元素外部时,页面会有相应的反馈或动作,以增强用户的交互体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值