Vue 3 自定义指令

核心概念

自定义指令用于封装底层 DOM 操作,主要应用场景:

  • 直接操作 DOM(聚焦、滚动等)
  • 集成第三方 DOM 库
  • 添加特殊行为(权限控制、拖拽等)

一、指令生命周期钩子(7个)

javascript 复制代码
const myDirective = {
  // 元素属性/事件监听应用前调用
  created(el, binding, vnode) {},
  
  // 元素插入父节点前调用
  beforeMount(el, binding, vnode) {},
  
  // 元素挂载后调用(最常用)
  mounted(el, binding, vnode) {},
  
  // 父组件更新前调用
  beforeUpdate(el, binding, vnode, prevVnode) {},
  
  // 父组件及子组件更新后调用
  updated(el, binding, vnode, prevVnode) {},
  
  // 父组件卸载前调用
  beforeUnmount(el, binding, vnode) {},
  
  // 指令解绑且父组件卸载后调用
  unmounted(el, binding, vnode) {}
}

二、钩子函数参数详解

  1. el: 指令绑定的 DOM 元素

  2. binding : 包含以下属性的对象

    javascript 复制代码
    {
      value:       当前指令值(v-my-dir="value"),
      oldValue:    旧值(仅 beforeUpdate/updated 可用),
      arg:         参数(v-my-dir:arg),
      modifiers:   修饰符对象(v-my-dir.mod1.mod2 → { mod1: true, mod2: true }),
      instance:    组件实例,
      dir:         指令定义对象
    }
  3. vnode: 当前虚拟节点

  4. prevVnode: 更新前的虚拟节点(仅 beforeUpdate/updated)


三、注册方式

1. 全局注册
javascript 复制代码
// main.js
import { createApp } from 'vue'

const app = createApp(App)
app.directive('focus', {
  mounted(el) {
    el.focus()
  }
})
2. 局部注册
javascript 复制代码
// Component.vue
export default {
  directives: {
    highlight: {
      mounted(el, binding) {
        el.style.backgroundColor = binding.value || 'yellow'
      }
    }
  }
}

四、使用示例

1. 基础用法
html 复制代码
<input v-focus>
<button v-permission="'admin'">管理员按钮</button>
2. 带参数
html 复制代码
<div v-pin:top="50">固定在顶部50px处</div>
javascript 复制代码
app.directive('pin', {
  mounted(el, binding) {
    el.style.position = 'fixed'
    el.style[binding.arg || 'top'] = `${binding.value}px`
  }
})
3. 带修饰符
html 复制代码
<button v-click.once="log">仅触发一次</button>
javascript 复制代码
app.directive('click', {
  mounted(el, binding) {
    const handler = binding.value
    let clicked = false
    
    el.addEventListener('click', () => {
      if (binding.modifiers.once && clicked) return
      handler()
      clicked = true
    })
  }
})

五、完整功能示例

1. 权限控制指令
javascript 复制代码
app.directive('permission', {
  mounted(el, binding) {
    const { value, modifiers } = binding
    const userRoles = ['admin', 'editor'] // 实际从store获取
    
    if (!value.some(role => userRoles.includes(role))) {
      if (modifiers.hide) {
        el.style.display = 'none'
      } else {
        el.disabled = true
        el.title = '无权限操作'
      }
    }
  }
})
html 复制代码
<button v-permission="['admin']">管理员操作</button>
<button v-permission.hide="['super-admin']">超级管理员操作</button>
2. 拖拽指令
javascript 复制代码
app.directive('drag', {
  mounted(el) {
    let pos1 = 0, pos2 = 0, pos3 = 0, pos4 = 0
    
    const dragMouseDown = (e) => {
      e.preventDefault()
      pos3 = e.clientX
      pos4 = e.clientY
      document.onmouseup = closeDragElement
      document.onmousemove = elementDrag
    }
    
    const elementDrag = (e) => {
      e.preventDefault()
      pos1 = pos3 - e.clientX
      pos2 = pos4 - e.clientY
      pos3 = e.clientX
      pos4 = e.clientY
      el.style.top = (el.offsetTop - pos2) + "px"
      el.style.left = (el.offsetLeft - pos1) + "px"
    }
    
    const closeDragElement = () => {
      document.onmouseup = null
      document.onmousemove = null
    }
    
    el.onmousedown = dragMouseDown
  }
})
html 复制代码
<div v-drag style="position:absolute">可拖拽元素</div>
3. 防抖指令
javascript 复制代码
app.directive('debounce', {
  mounted(el, binding) {
    const [callback, delay = 300] = binding.value
    let timer = null
    
    el.addEventListener('click', () => {
      clearTimeout(timer)
      timer = setTimeout(() => callback(), delay)
    })
  }
})
html 复制代码
<button v-debounce="[submit, 500]">防抖提交</button>

六、最佳实践

  1. 优先使用组件:非 DOM 操作优先考虑组件

  2. 解绑事件 :在 unmounted 中移除事件监听

    javascript 复制代码
    unmounted(el) {
      el.removeEventListener('click', el._clickHandler)
    }
  3. 动态指令参数

    html 复制代码
    <div v-move:[direction]="100"></div>
  4. 指令复用:通过参数化实现通用指令

  5. 性能优化 :避免在 updated 中执行重操作


七、注意事项

  1. 组件可能使用 inheritAttrs: false 影响指令
  2. 指令优先级高于组件生命周期
  3. 在 SSR 中避免 DOM 操作
  4. 动态参数变化会触发 beforeUpdate/updated
相关推荐
岁月宁静2 小时前
深度定制:在 Vue 3.5 应用中集成流式 AI 写作助手的实践
前端·vue.js·人工智能
百锦再3 小时前
Vue Scoped样式混淆问题详解与解决方案
java·前端·javascript·数据库·vue.js·学习·.net
Sheldon一蓑烟雨任平生4 小时前
Vue3 表单输入绑定
vue.js·vue3·v-model·vue3 表单输入绑定·表单输入绑定·input和change区别·vue3 双向数据绑定
YUELEI1186 小时前
Vue 安装依赖的集合和小知识
javascript·vue.js·ecmascript
前端付豪7 小时前
万事从 todolist 开始
前端·vue.js·前端框架
华仔啊8 小时前
别再纠结Pinia和Vuex了!一篇文章彻底搞懂区别与选择
前端·vue.js
月弦笙音9 小时前
【Vue3】Keep-Alive 深度解析
前端·vue.js·源码阅读
咖啡の猫10 小时前
Vue 实例生命周期
前端·vue.js·okhttp
JNU freshman11 小时前
vue 之 import 的语法
前端·javascript·vue.js
剑亦未配妥11 小时前
Vue 2 响应式系统常见问题与解决方案(包含_demo以下划线开头命名的变量导致响应式丢失问题)
前端·javascript·vue.js