vue3中自定义指令使用

Vue 3 自定义指令详细使用指南

1. 基本结构

一个完整的自定义指令包含以下可选钩子函数:

javascript 复制代码
const myDirective = {
  // 指令第一次绑定到元素时调用(只调用一次)
  created(el, binding, vnode, prevVnode) {},
  
  // 元素插入父 DOM 前调用
  beforeMount(el, binding, vnode, prevVnode) {},
  
  // 元素插入父 DOM 后调用
  mounted(el, binding, vnode, prevVnode) {},
  
  // 父组件更新前调用
  beforeUpdate(el, binding, vnode, prevVnode) {},
  
  // 父组件和子组件都更新后调用
  updated(el, binding, vnode, prevVnode) {},
  
  // 父组件卸载前调用
  beforeUnmount(el, binding, vnode, prevVnode) {},
  
  // 父组件卸载后调用
  unmounted(el, binding, vnode, prevVnode) {}
}

2.钩子函数参数

3.注册方式

3.1全局注册

javascript 复制代码
// main.js
import { createApp } from 'vue'
import App from './App.vue'

const app = createApp(App)

app.directive('focus', {
  mounted(el) {
    el.focus()
  }
})

app.mount('#app')

3.2 局部注册

javascript 复制代码
// 在组件中
export default {
  directives: {
    focus: {
      mounted(el) {
        el.focus()
      }
    }
  }
}

4.使用方式

5.实用示例

5.1权限控制指令

javascript 复制代码
app.directive('permission', {
  mounted(el, binding) {
    const userPermissions = ['edit', 'view'] // 通常从store获取
    if (!userPermissions.includes(binding.value)) {
      el.style.display = 'none'
      // 或者直接移除元素
      // el.parentNode.removeChild(el)
    }
  }
})

5.2 防抖点击指令

javascript 复制代码
app.directive('debounce-click', {
  mounted(el, binding) {
    const delay = binding.value || 500
    let timer = null
    
    el.addEventListener('click', () => {
      if (timer) clearTimeout(timer)
      timer = setTimeout(() => {
        binding.arg && binding.arg()
      }, delay)
    })
  },
  unmounted(el) {
    el.removeEventListener('click')
  }
})

5.3 拖拽指令

javascript 复制代码
app.directive('drag', {
  mounted(el) {
    let isDragging = false
    let offsetX, offsetY
    
    el.style.position = 'absolute'
    el.style.cursor = 'move'
    
    el.addEventListener('mousedown', (e) => {
      isDragging = true
      offsetX = e.clientX - el.getBoundingClientRect().left
      offsetY = e.clientY - el.getBoundingClientRect().top
    })
    
    document.addEventListener('mousemove', (e) => {
      if (!isDragging) return
      el.style.left = e.clientX - offsetX + 'px'
      el.style.top = e.clientY - offsetY + 'px'
    })
    
    document.addEventListener('mouseup', () => {
      isDragging = false
    })
  }
})

5.4 图片懒加载指令

javascript 复制代码
app.directive('lazy', {
  mounted(el, binding) {
    const observer = new IntersectionObserver((entries) => {
      entries.forEach(entry => {
        if (entry.isIntersecting) {
          el.src = binding.value
          observer.unobserve(el)
        }
      })
    })
    observer.observe(el)
  }
})

6.注意事项

在 setup() 中定义的指令必须以 v 开头(如 vFocus),才能在模板中作为 v-focus 使用
指令必须从 setup() 中返回才能在模板中使用
对于需要清理的资源(如事件监听器),确保在 unmounted 钩子中进行清理
指令的 binding 参数是响应式的,但 el 不是
在 <script setup> 中定义的指令会自动暴露给模板,无需显式返回

相关推荐
有味道的男人12 分钟前
如何使用招标网API获取项目详情?
java·服务器·前端
qq_4061761413 分钟前
深入剖析JS中的XSS与CSRF漏洞:原理、攻击与防御全指南
服务器·开发语言·前端·javascript
RFCEO14 分钟前
HTML编程 课程六、:HTML5 新增多媒体标签
前端·html·html5·多媒体标签·嵌入音频、视频、动画
yanyu-yaya22 分钟前
速学兼复习之vue3章节4
前端·vue.js·前端框架
Mr-Wanter26 分钟前
vue 数据反显时数字/字母不换行导致的样式问题
前端·javascript·vue.js
梁萌33 分钟前
vue项目从npm升级为pnpm
前端·npm·node.js
修己xj34 分钟前
CSS魔法:对话生成器与奔驰骏马的创意实现
前端·css
qq_124987075334 分钟前
基于Java的心理测试系统的设计与实现(源码+论文+部署+安装)
java·开发语言·vue.js·spring boot·计算机毕设·计算机毕业设计
daols881 小时前
vue表格 vxe-table 如何实现键盘导航时,按回车健向右移动,并到最后一行时按回车自动新增一行
vue.js·vxe-table
琹箐1 小时前
Cursor 无法使用prettier格式化
前端