作者:lzg9527 https://juejin.cn/post/6906028995133833230

在 Vue,除了核心功能默认内置的指令 ( v-model 和 v-show ),Vue 也允许注册自定义指令。它的作用价值在于当开发人员在某些场景下需要对普通 DOM 元素进行操作。

Vue 自定义指令有全局注册和局部注册两种方式。先来看看注册全局指令的方式,通过 Vue.directive( id, [definition] ) 方式注册全局指令。然后在入口文件中进行 Vue.use() 调用。

批量注册指令,新建 directives/index.js 文件

import copy from './copy'import longpress from './longpress'// 自定义指令const directives = {  copy,  longpress,}

export default {  install(Vue) {    Object.keys(directives).forEach((key) => {      Vue.directive(key, directives[key])    })  },}

main.js 引入并调用

import Vue from 'vue'import Directives from './JS/directives'Vue.use(Directives)

指令定义函数提供了几个钩子函数(可选):

  • bind: 只调用一次,指令第一次绑定到元素时调用,可以定义一个在绑定时执行一次的初始化动作。
  • inserted: 被绑定元素插入父节点时调用(父节点存在即可调用,不必存在于 document 中)。
  • update: 被绑定元素所在的模板更新时调用,而不论绑定值是否变化。通过比较更新前后的绑定值。
  • componentUpdated: 被绑定元素所在模板完成一次更新周期时调用。
  • unbind: 只调用一次, 指令与元素解绑时调用。

下面分享几个实用的 Vue 自定义指令

  • 复制粘贴指令 v-copy
  • 长按指令 v-longpress
  • 输入框防抖指令 v-debounce
  • 禁止表情及特殊字符 v-emoji
  • 图片懒加载 v-LazyLoad
  • 权限校验指令 v-premission
  • 实现页面水印 v-waterMarker
  • 拖拽指令 v-draggable

v-copy

需求:实现一键复制文本内容,用于鼠标右键粘贴。

思路:

  1. 动态创建 textarea 标签,并设置 readOnly 属性及移出可视区域
  2. 将要复制的值赋给 textarea 标签的 value 属性,并插入到 body
  3. 选中值 textarea 并复制
  4. body 中插入的 textarea 移除
  5. 在第一次调用时绑定事件,在解绑时移除事件
const copy = {  bind(el, { value }) {    el.$value = value    el.handler = () => {      if (!el.$value) {        // 值为空的时候,给出提示。可根据项目UI仔细设计        console.log('无复制内容')        return      }      // 动态创建 textarea 标签      const textarea = document.createElement('textarea')      // 将该 textarea 设为 readonly 防止 iOS 下自动唤起键盘,同时将 textarea 移出可视区域      textarea.readOnly = 'readonly'      textarea.style.position = 'absolute'      textarea.style.left = '-9999px'      // 将要 copy 的值赋给 textarea 标签的 value 属性      textarea.value = el.$value      // 将 textarea 插入到 body 中      document.body.appendChild(textarea)      // 选中值并复制      textarea.select()      const result = document.execCommand('Copy')      if (result) {        console.log('复制成功') // 可根据项目UI仔细设计      }      document.body.removeChild(textarea)    }    // 绑定点击事件,就是所谓的一键 copy 啦    el.addEventListener('click', el.handler)  },  // 当传进来的值更新的时候触发  componentUpdated(el, { value }) {    el.$value = value  },  // 指令与元素解绑的时候,移除事件绑定  unbind(el) {    el.removeEventListener('click', el.handler)  },}

export default copy

使用:给 Dom 加上 v-copy 及复制的文本即可

<template>  <button v-copy="copyText">复制button>template>

<script>export default {    data() {return {copyText: 'a copy directives',      }    },  }script>

v-longpress

需求:实现长按,用户需要按下并按住按钮几秒钟,触发相应的事件

思路:

  1. 创建一个计时器, 2 秒后执行函数
  2. 当用户按下按钮时触发 mousedown 事件,启动计时器;用户松开按钮时调用mouseout 事件。
  3. 如果 mouseup 事件 2 秒内被触发,就清除计时器,当作一个普通的点击事件
  4. 如果计时器没有在 2 秒内清除,则判定为一次长按,可以执行关联的函数。
  5. 在移动端要考虑 touchstarttouchend 事件
const longpress = {  bind: function (el, binding, vNode) {    if (typeof binding.value !== 'function') {      throw 'callback must be a function'    }    // 定义变量    let pressTimer = null    // 创建计时器( 2秒后执行函数 )    let start = (e) => {      if (e.type === 'click' && e.button !== 0) {        return      }      if (pressTimer === null) {        pressTimer = setTimeout(() => {          handler()        }, 2000)      }    }    // 取消计时器    let cancel = (e) => {      if (pressTimer !== null) {        clearTimeout(pressTimer)        pressTimer = null      }    }    // 运行函数    const handler = (e) => {      binding.value(e)    }    // 添加事件监听器    el.addEventListener('mousedown', start)    el.addEventListener('touchstart', start)    // 取消计时器    el.addEventListener('click', cancel)    el.addEventListener('mouseout', cancel)    el.addEventListener('touchend', cancel)    el.addEventListener('touchcancel', cancel)  },  // 当传进来的值更新的时候触发  componentUpdated(el, { value }) {    el.$value = value  },  // 指令与元素解绑的时候,移除事件绑定  unbind(el) {    el.removeEventListener('click', el.handler)  },}

export default longpress

使用:给 Dom 加上 v-longpress 及回调函数即可

<template>  <button v-longpress="longpress">长按button>template>

<script>export default {methods: {    longpress () {      alert('长按指令生效')    }  }}

v-debounce

背景:在开发中,有些提交保存按钮有时候会在短时间内被点击多次,这样就会多次重复请求后端接口,造成数据的混乱,比如新增表单的提交按钮,多次点击就会新增多条重复的数据。

需求:防止按钮在短时间内被多次点击,使用防抖函数限制规定时间内只能点击一次。

思路:

  1. 定义一个延迟执行的方法,如果在延迟时间内再调用该方法,则重新计算执行时间。
  2. 将时间绑定在 click 方法上。
const debounce = {  inserted: function (el, binding) {    let timer    el.addEventListener('keyup', () => {      if (timer) {        clearTimeout(timer)      }      timer = setTimeout(() => {        binding.value()      }, 1000)    })  },}

export default debounce

使用:给 Dom 加上 v-debounce 及回调函数即可

<template>  <button v-debounce="debounceClick">防抖button>template>

<script>export default {methods: {    debounceClick () {console.log('只触发一次')    }  }}

v-emoji

背景:开发中遇到的表单输入,往往会有对输入内容的限制,比如不能输入表情和特殊字符,只能输入数字或字母等。

我们常规方法是在每一个表单的 on-change 事件上做处理。

<template>  <input type="text" v-model="note" @change="vaidateEmoji" />template>

<script>export default {methods: {      vaidateEmoji() {var reg = /[^\u4E00-\u9FA5|\d|\a-zA-Z|\r\n\s,.?!,。?!…—&$=()-+/*{}[\]]|\s/gthis.note = this.note.replace(reg, '')      },    },  }script>

这样代码量比较大而且不好维护,所以我们需要自定义一个指令来解决这问题。

需求:根据正则表达式,设计自定义处理表单输入规则的指令,下面以禁止输入表情和特殊字符为例。

let findEle = (parent, type) => {  return parent.tagName.toLowerCase() === type ? parent : parent.querySelector(type)}

const trigger = (el, type) => {  const e = document.createEvent('HTMLEvents')  e.initEvent(type, true, true)  el.dispatchEvent(e)}

const emoji = {  bind: function (el, binding, vnode) {    // 正则规则可根据需求自定义    var regRule = /[^\u4E00-\u9FA5|\d|\a-zA-Z|\r\n\s,.?!,。?!…—&$=()-+/*{}[\]]|\s/g    let $inp = findEle(el, 'input')    el.$inp = $inp    $inp.handle = function () {      let val = $inp.value      $inp.value = val.replace(regRule, '')

      trigger($inp, 'input')    }    $inp.addEventListener('keyup', $inp.handle)  },  unbind: function (el) {    el.$inp.removeEventListener('keyup', el.$inp.handle)  },}

export default emoji

使用:将需要校验的输入框加上 v-emoji 即可

<template>  <input type="text" v-model="note" v-emoji />template>

v-LazyLoad

背景:在类电商类项目,往往存在大量的图片,如 banner 广告图,菜单导航图,美团等商家列表头图等。图片众多以及图片体积过大往往会影响页面加载速度,造成不良的用户体验,所以进行图片懒加载优化势在必行。

需求:实现一个图片懒加载指令,只加载浏览器可见区域的图片。

思路:

  1. 图片懒加载的原理主要是判断当前图片是否到了可视区域这一核心逻辑实现的
  2. 拿到所有的图片 Dom ,遍历每个图片判断当前图片是否到了可视区范围内
  3. 如果到了就设置图片的 src 属性,否则显示默认图片

图片懒加载有两种方式可以实现,一是绑定 srcoll 事件进行监听,二是使用 IntersectionObserver 判断图片是否到了可视区域,但是有浏览器兼容性问题。

下面封装一个懒加载指令兼容两种方法,判断浏览器是否支持 IntersectionObserver API,如果支持就使用 IntersectionObserver 实现懒加载,否则则使用 srcoll 事件监听 + 节流的方法实现。

const LazyLoad = {  // install方法  install(Vue, options) {    const defaultSrc = options.default    Vue.directive('lazy', {      bind(el, binding) {        LazyLoad.init(el, binding.value, defaultSrc)      },      inserted(el) {        if (IntersectionObserver) {          LazyLoad.observe(el)        } else {          LazyLoad.listenerScroll(el)        }      },    })  },  // 初始化  init(el, val, def) {    el.setAttribute('src', val)    el.setAttribute('src', def)  },  // 利用IntersectionObserver监听el  observe(el) {    var io = new IntersectionObserver((entries) => {      const realSrc = el.dataset.src      if (entries[0].isIntersecting) {        if (realSrc) {          el.src = realSrc          el.removeAttribute('src')        }      }    })    io.observe(el)  },  // 监听scroll事件  listenerScroll(el) {    const handler = LazyLoad.throttle(LazyLoad.load, 300)    LazyLoad.load(el)    window.addEventListener('scroll', () => {      handler(el)    })  },  // 加载真实图片  load(el) {    const windowHeight = document.documentElement.clientHeight    const elTop = el.getBoundingClientRect().top    const elBtm = el.getBoundingClientRect().bottom    const realSrc = el.dataset.src    if (elTop - windowHeight 0 && elBtm > 0) {      if (realSrc) {        el.src = realSrc        el.removeAttribute('src')      }    }  },  // 节流  throttle(fn, delay) {    let timer    let prevTime    return function (...args) {      const currTime = Date.now()      const context = this      if (!prevTime) prevTime = currTime      clearTimeout(timer)

      if (currTime - prevTime > delay) {        prevTime = currTime        fn.apply(context, args)        clearTimeout(timer)        return      }

      timer = setTimeout(function () {        prevTime = Date.now()        timer = null        fn.apply(context, args)      }, delay)    }  },}

export default LazyLoad

使用,将组件内  标签的 src 换成 v-LazyLoad

<img v-LazyLoad="xxx.jpg" />

v-permission

背景:在一些后台管理系统,我们可能需要根据用户角色进行一些操作权限的判断,很多时候我们都是粗暴地给一个元素添加 v-if / v-show 来进行显示隐藏,但如果判断条件繁琐且多个地方需要判断,这种方式的代码不仅不优雅而且冗余。针对这种情况,我们可以通过全局自定义指令来处理。

需求:自定义一个权限指令,对需要权限判断的 Dom 进行显示隐藏。

思路:

  1. 自定义一个权限数组
  2. 判断用户的权限是否在这个数组内,如果是则显示,否则则移除 Dom
function checkArray(key) {  let arr = ['1', '2', '3', '4']  let index = arr.indexOf(key)  if (index > -1) {    return true // 有权限  } else {    return false // 无权限  }}

const permission = {  inserted: function (el, binding) {    let permission = binding.value // 获取到 v-permission的值    if (permission) {      let hasPermission = checkArray(permission)      if (!hasPermission) {        // 没有权限 移除Dom元素        el.parentNode && el.parentNode.removeChild(el)      }    }  },}

export default permission

使用:给 v-permission 赋值判断即可

<div class="btns">

  <button v-permission="'1'">权限按钮1button>

  <button v-permission="'10'">权限按钮2button>div>

vue-waterMarker

需求:给整个页面添加背景水印

思路:

  1. 使用 canvas 特性生成 base64 格式的图片文件,设置其字体大小,颜色等。
  2. 将其设置为背景图片,从而实现页面或组件水印效果
function addWaterMarker(str, parentNode, font, textColor) {  // 水印文字,父元素,字体,文字颜色  var can = document.createElement('canvas')  parentNode.appendChild(can)  can.width = 200  can.height = 150  can.style.display = 'none'  var cans = can.getContext('2d')  cans.rotate((-20 * Math.PI) / 180)  cans.font = font || '16px Microsoft JhengHei'  cans.fillStyle = textColor || 'rgba(180, 180, 180, 0.3)'  cans.textAlign = 'left'  cans.textBaseline = 'Middle'  cans.fillText(str, can.width / 10, can.height / 2)  parentNode.style.backgroundImage = 'url(' + can.toDataURL('image/png') + ')'}

const waterMarker = {  bind: function (el, binding) {    addWaterMarker(binding.value.text, el, binding.value.font, binding.value.textColor)  },}

export default waterMarker

使用,设置水印文案,颜色,字体大小即可

<template>  <div v-waterMarker="{text:'lzg版权所有',textColor:'rgba(180, 180, 180, 0.4)'}">div>template>

v-draggable

需求:实现一个拖拽指令,可在页面可视区域任意拖拽元素。

思路:

  1. 设置需要拖拽的元素为相对定位,其父元素为绝对定位。
  2. 鼠标按下(onmousedown)时记录目标元素当前的 lefttop 值。
  3. 鼠标移动(onmousemove)时计算每次移动的横向距离和纵向距离的变化值,并改变元素的 lefttop
  4. 鼠标松开(onmouseup)时完成一次拖拽
const draggable = {  inserted: function (el) {    el.style.cursor = 'move'    el.onmousedown = function (e) {      let disx = e.pageX - el.offsetLeft      let disy = e.pageY - el.offsetTop      document.onmousemove = function (e) {        let x = e.pageX - disx        let y = e.pageY - disy        let maxX = document.body.clientWidth - parseInt(window.getComputedStyle(el).width)        let maxY = document.body.clientHeight - parseInt(window.getComputedStyle(el).height)        if (x 0) {          x = 0        } else if (x > maxX) {          x = maxX        }

        if (y 0) {          y = 0        } else if (y > maxY) {          y = maxY        }

        el.style.left = x + 'px'        el.style.top = y + 'px'      }      document.onmouseup = function () {        document.onmousemove = document.onmouseup = null      }    }  },}export default draggable

使用: 在 Dom 上加上 v-draggable 即可

<template>  <div class="el-dialog" v-draggable>div>template>

clientsideevents能定义几个click事件_分享8个非常实用的Vue自定义指令相关推荐

  1. pyqt5 treeview鼠标右键菜单事件_【动手实践】使用 Vue 自定义指令实现右键菜单...

    本文来自于 神奇的程序员 前言 浏览器里右键时会有一个默认的菜单,在我的开源项目中正好有自定义右键菜单的需求,在npm库找了下与之相关的包,发现都是以组件形式实现的,感觉那种做法太过繁琐. 于是,我就 ...

  2. vue鼠标右键自定义菜单_使用Vue自定义指令实现右键菜单

    前言 浏览器里右键时会有一个默认的菜单,在我的开源项目中正好有自定义右键菜单的需求,在npm库找了下与之相关的包,发现都是以组件形式实现的,感觉那种做法太过繁琐. 于是,我就想着能不能像vue的内置指 ...

  3. vuejs 指令封装 button 加载效果_这些Vue自定义指令,让你的项目开发爽到爆

    受 AngularJS 的启发,Vue 内置了一些非常有用的指令(比如v-html 和 v-once等),每个指令都有自身的用途.完整的指令列表可以在这里查看. 这还没完,更棒的是可以开发自定义指令. ...

  4. vue 离开页面事件_【必看】58 道 Vue 常见面试题集锦,涵盖入门到精通,自测 Vue 掌握程度...

    △ 是新朋友吗?记得先点web前端学习圈关注我哦- 1.vue优点? 答: 轻量级框架:只关注视图层,是一个构建数据的视图集合,大小只有几十 kb : 简单易学:国人开发,中文文档,不存在语言障碍 , ...

  5. Vue 自定义指令上报 Google Analytics 事件统计

    发现问题 一般前端开发离不开数据统计,我们经常需要接入统计服务以方便运营,例如现在需要统计一个按钮 <template><button @click="handleClic ...

  6. 5e怎么绑定一键跳投_一个 Vue 自定义指令实现一键 Copy的功能

    话不多说先看效果,这个效果是用在真实项目中的实际效果哈: 指令是啥? 按照惯例,先请出官方的解释: 指令 (Directives) 是带有 v- 前缀的特殊特性.指令特性的值预期是单个 JavaScr ...

  7. vue 图片拖动加载 类似于地图_前端性能优化之图片懒加载(附vue自定义指令)...

    作者:lzg9527 链接:https://juejin.cn/post/6903774214780616718 在类电商类项目,往往存在大量的图片,如 banner 广告图,菜单导航图,美团等商家列 ...

  8. vue自定义指令---处理加载图片失败时出现的碎图,onerror事件

    目录 一.自定义指令 1.局部注册和使用 2.全局注册和使用 二.自定义指令处理图片加载失败(碎图) 一.自定义指令 vue中除v-model.v-show等内置指令之外,还允许注册自定义指令,获取D ...

  9. mysql 查看等待事件_分享两个实用SQL--查看故障时间等待事件、问题sql及会话访问次数...

    分享两个SQL,这两个SQL只是基础原型,大家自己自定义条件去做一些调整.主要是在前面讲过的dba_hist_active_sess_history相关sql上总结出来的比较实用的sql. DBA_H ...

最新文章

  1. Linux ekho
  2. NMHDR 与 NM_LISTVIEW,NM_TREEVIEW 的关系
  3. 剑指Offer(Java实现)删除链表中重复的结点
  4. 成功解决 “this version of pandas is incompatible with numpy < 1.15.4\n“ImportError: this version of pand
  5. python 中的路径. ./ .. ../和相对路径的写法
  6. hibernate报错 net.sf.json.util.CycleDetectionStrategy$StrictionStrategyRepeatedReferenceAsObject
  7. javascript文本对象
  8. Ubuntu 配置vsftpd实现FTP服务器
  9. flink其他可选api
  10. 做一个小淘气轮廓--文章和论文专辑
  11. 收集java精确截取字符串
  12. Python爬虫入门教程 68-100 一招解决爬虫程序卡死问题
  13. 《黑匣子思维:我们如何更理性地犯错》ipad部分
  14. 对比 iOS 14.5 最强更新功能,Android 系统做得怎么样?
  15. 【C/C++】【NOI】雇佣兵代码及分析理解
  16. 计算机课高数怎么过,我是大一新鸟,学软件,想向大虾们讨教几手,我有高数,英语,法律.计算机,4门课,我该怎么学!介绍越详细越好,...
  17. 最新消息:@live.com@live.cn邮箱正式开放注册了!
  18. 大一python期末必考题_Python期末复习材料
  19. JS 判断手机端跳转
  20. NFT 有哪些不同类型

热门文章

  1. 201521123035《Java程序设计》第十周实验总结
  2. input美化上传按钮美化
  3. hive函数参考手册
  4. 中文依存句法分析概述及应用
  5. DirectX SDK (June 2010)安装遇到Error Code S1023
  6. Color Table
  7. 在线 SQL 数据库环境
  8. ubuntu16.04.3 ntp服务
  9. bat shell 命令行中 21 的含义
  10. 【MySQL-5.7】日期与时间类型