Vue技巧大揭秘:自定义指令的力量与应用

引言

自定义指令就像是给予开发者的一把魔法钥匙,它能够打开DOM操作的新世界,按我的理解就是把对DOM操作的逻辑进行封装

全局注册与局部注册

全局注册

定义: 全局注册意味着自定义指令在Vue实例创建之前通过Vue.directive()方法注册,一旦注册,就可以在任意组件的模板中使用该指令。

适用场景

  • 当自定义指令在多个组件中都需要使用时。
  • 当你想要创建一个全局可用的行为,比如一个自定义的拖拽指令。

优点

  • 方便:只需注册一次,就可以在任何组件中使用。
  • 管理简单:所有指令集中注册,便于维护。

缺点

  • 不灵活:全局注册的指令在所有组件中都是可用的,可能会导致不必要的全局污染。
  • 不可配置:所有使用该指令的地方都会受到相同的指令行为影响。

局部注册

定义: 局部注册是在单个组件内部通过组件的directives选项注册的,这意味着指令只在该组件内有效。

适用场景

  • 当自定义指令只在一个组件内使用时。
  • 当你需要为不同的组件提供不同行为的指令时。

优点

  • 灵活:可以根据组件的需要注册和使用指令。
  • 可配置:可以在不同的组件中为同一个指令提供不同的配置。
  • 避免污染:不会影响到其他组件,减少了全局空间的占用。

缺点

  • 重复:如果在多个组件中使用相同的指令,需要在每个组件中重复注册。
  • 维护难度:指令分散在各个组件中,随着项目规模的增大,可能会导致维护难度增加。

实际操作示范

全局注册(与new Vue同级)

如何进行全局注册

全局注册是在Vue实例化之前进行的,通常在入口文件中,如main.jsapp.js。使用Vue.directive()方法来注册一个全局指令。

代码示例和步骤说明
// main.js
import Vue from 'vue';
import App from './App.vue';

// 全局注册自定义指令
Vue.directive('focus', {
  // 钩子函数
  inserted: function (el) {
    el.focus();
  }
});

new Vue({
  el: '#app',
  render: h => h(App)
});

在上面的代码中,我们定义了一个名为focus的全局指令,它在元素插入到DOM后自动聚焦该元素。

全局注册的应用场景

全局注册适用于那些在多个组件中都需要使用的指令,比如一个控制输入框自动聚焦的指令。

<!-- 在任何组件中使用全局注册的指令 -->
<template>
  <input v-focus>
</template>

局部注册(与methods同级)

如何进行局部注册

局部注册是在组件内部进行的,通过在组件的directives选项中定义指令。

代码示例和步骤说明
// 局部注册自定义指令
export default {
  name: 'MyComponent',
  directives: {
    'local-focus': {
      // 钩子函数
      inserted: function (el) {
        el.focus();
      }
    }
  },
  // ...
};

在上面的代码中,我们定义了一个名为local-focus的局部指令,它仅在MyComponent组件内部有效。

局部注册的应用场景

局部注册适用于那些仅在一个组件内部使用的指令,比如一个特定组件的特定行为。

<!-- 在组件内部使用局部注册的指令 -->
<template>
  <input v-local-focus>
</template>

指令的钩子函数

Vue中的自定义指令提供了几个钩子函数,这些函数允许你在不同阶段操纵DOM,或对DOM进行一些操作。以下是自定义指令的钩子函数,以及它们的作用和使用场景:

  1. bind: 在指令第一次绑定到元素上时调用。在这里可以进行一次性的初始化设置,例如添加事件监听器或初始化样式。

  2. inserted: 被绑定元素插入父节点时调用(仅保证父节点存在,但不一定已被插入文档中)。例如,如果你需要访问元素的尺寸或位置,这个钩子很有用。

  3. update: 所在组件的VNode更新时调用,但是可能发生在其子VNode更新之前。当你需要根据组件的状态更新指令的行为时,可以使用这个钩子。

  4. componentUpdated(updata的补充版本): 在指令所在组件的VNode及其子VNode全部更新后调用。如果你需要在所有子组件都更新后执行某些操作,这个钩子很有用。

  5. unbind: 只调用一次,指令与元素解绑时调用。在这里可以进行一些清理工作,例如移除事件监听器。

使用场景示例

  • bind: 设置初始样式或绑定事件监听器。
  • inserted: 自动聚焦输入框或执行与元素位置相关的操作。
  • update: 根据组件状态更新元素样式或内容。
  • componentUpdated: 在所有子组件更新后执行某些操作,例如调整滚动位置。
  • unbind: 清理工作,例如移除事件监听器或取消定时器。

代码示例

Vue.directive('example', {
  bind: function (el, binding, vnode) {
    // 初始化操作
  },
  inserted: function (el, binding, vnode) {
    // 元素插入父节点后的操作
  },
  update: function (el, binding, vnode, oldVnode) {
    // 组件更新时的操作
  },
  componentUpdated: function (el, binding, vnode, oldVnode) {
    // 组件及其子组件更新完成后的操作
  },
  unbind: function (el, binding, vnode) {
    // 指令解绑时的操作
  }
});

指令的参数和修饰符 

 

参数(Argument)

指令参数通常用于指定指令的行为或选项。在指令中使用参数时,需要在指令名称后面用冒号(:)分隔。例如,v-my-directive:arg中的arg就是参数。

使用场景
  • 当你需要根据不同的条件执行不同的指令逻辑时。
  • 当你需要传递一个值或配置项给指令时。
示例

假设我们有一个自定义指令v-tooltip,用于显示工具提示,我们可以通过参数指定工具提示的方向:

<p v-tooltip:"top">鼠标悬停显示顶部工具提示</p>
<p v-tooltip:"bottom">鼠标悬停显示底部工具提示</p>

在指令定义中,我们可以通过binding.arg获取参数值:

Vue.directive('tooltip', {
  bind: function (el, binding) {
    // 根据参数设置工具提示的方向
    el.style[binding.arg] = '10px';
  }
});

修饰符(Modifier)

修饰符是以点(.)开头的特殊标记,用于指示指令应该以特殊方式绑定。修饰符可以改变指令的行为,或者为指令提供额外的信息。

使用场景
  • 当你需要为指令提供额外的配置选项时。
  • 当你需要改变指令的默认行为时。
示例

假设我们有一个自定义指令v-click-outside,用于在点击元素外部时触发事件。我们可以通过修饰符来指定是否阻止事件冒泡:

<div v-click-outside.stop>点击外部时触发事件,并阻止事件冒泡</div>

在指令定义中,我们可以通过binding.modifiers获取修饰符对象:

Vue.directive('click-outside', {
  bind: function (el, binding) {
    // 为元素添加点击事件监听器
    el.addEventListener('click', function (event) {
      // 如果有修饰符@stop,则阻止事件冒泡
      if (binding.modifiers.stop) {
        event.stopPropagation();
      }
      // 执行指令逻辑
    });
  }
});

 总结

Vue自定义指令允许开发者封装DOM操作逻辑,全局和局部注册提供了灵活性。指令钩子在不同生命周期操作DOM,参数和修饰符增强了指令的定制能力。这些特性有助于创建高效、可维护的自定义指令。
创作不易,您的每一个点赞和评论都是我创作的动力!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/765737.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

在非 antd pro 项目中使用 umi OpenAPI

大家好&#xff0c;我是松柏。自从跟着鱼皮哥使用了ant design pro中的OpenAPI插件之后&#xff0c;我已经无法忍受自己写请求后端接口的方法了&#xff0c;所以这篇文章记录一下如何在非ant design pro项目中使用OpenAPI。 安装依赖 首先我们需要安装包umijs/openapi&#x…

java面试课程-SpringIOC部分源码解析

1.SpringIOC的refresh源码解析 核心&#xff1a; 核心使用的是&#xff1a; 需要完成配置类的解析&#xff0c;各种BeanFactoryProcessor的注册。还有写国际化配置的初始化。Web容器的内部构造。 上面几个方法是refresh方法的内容。注意可以与applicationContext里的内容一起…

Profibus DP主站转Modbus网关连接智能化电表通讯

Profibus DP主站转Modbus网关&#xff08;XD-MDPBM20&#xff09;&#xff0c;是实现不同工业通信协议之间互联互通的设备&#xff0c;主要将Profibus DP协议转换为Modbus协议&#xff0c;实现数据的双向传输。通过Profibus DP主站转Modbus网关&#xff08;XD-MDPBM20&#xff…

大Excel表格76M,电脑16G内存打不开,内存利用率100%虚拟内存占用16G还是卡死提示内存不足,如何才能查看里面内容?

环境: Excel2016 问题描述: 大Excel表格76M,电脑16G内存打不开,内存利用率100%虚拟内存占用16G还是卡死提示内存不足,如何才能查看里面内容? 解决方案: 遇到这种情况,说明Excel文件非常大,超出了你当前计算机配置的处理能力。以下是一些解决方法,帮助你尝试打开或…

代码随想录-Day45

198. 打家劫舍 你是一个专业的小偷&#xff0c;计划偷窃沿街的房屋。每间房内都藏有一定的现金&#xff0c;影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统&#xff0c;如果两间相邻的房屋在同一晚上被小偷闯入&#xff0c;系统会自动报警。 给定一个代表每个…

编译原理2

推导和短语 推导 推导过程中&#xff0c;每一步推导都是对句型的 最右非终结符 进行替换&#xff0c;最右推导(规范推导)&#xff1b; 短语 用 β 替换 A&#xff0c;则 β 就是 关于A 的一个短语&#xff1b; 直接短语是短语范围内的一步推导&#xff1b; 直接短语可能不…

基于python的随机森林回归预测+贝叶斯优化超参数前后训练效果对比

目录 1.导入必要的库 2.导入数据与数据预处理 3.查看数据分布 4.特征选择 5.模型建立与训练 6.训练集预测结果 7.模型评估 8.预测新数据 9.贝叶斯优化超参数 1.导入必要的库 # 导入所需的库 from sklearn.model_selection import cross_val_score import pandas as …

Sentinel实现区分来源

要区分来源就要写代码实现RequestOriginParser接口 &#xff0c;注意是要实现com.alibaba.csp.sentinel.adapter.servlet.callback.RequestOriginParser 接口&#xff0c;别搞错接口了。 MyRequestOriginParser.java package com.codex.terry.sentinel.origin;import com.ali…

STM32mp157aaa按键中断实验

效果图&#xff1a; 源码&#xff1a; #include "key.h" void hal_key1_rcc_gpio_init() {// 使能GPIOF组RCC->MP_AHB4ENSETR | (0x1 << 5);// 设置引脚位输入模式GPIOF->MODER & (~(0X3 << 18));GPIOF->MODER & (~(0X3 << 16))…

当Matplotlib遇见SciencePlots

分享一个Matplotlib扩展工具SciencePlots&#xff0c;一行代码绘制science、nature、ieee等要求的图形。 安装 安装SciencePlots # 直接从PyPI安装 pip install SciencePlots 安装latex 如果latex未安装&#xff0c;会报错&#xff1a;RuntimeError: Failed to process st…

【QT开发】乒乓球碰撞反弹demo

在编写代码时&#xff0c;无意弄出来了一个这个东西&#xff0c;觉得挺有意思的记录一下&#xff0c;类似乒乓球在矩形内一直运动碰撞反弹demo 头文件 #ifndef MYPROJECT_H #define MYPROJECT_H#include <QMainWindow> #include <QPainter> #include "form.…

【区块链+基础设施】国家健康医疗大数据科创平台 | FISCO BCOS应用案例

在医疗领域&#xff0c;疾病数据合法合规共享是亟待解决的难题。一方面&#xff0c;当一家医院对患者实施治疗后&#xff0c;若患者转到其 他医院就医&#xff0c;该医院就无法判断诊疗手段是否有效。另一方面&#xff0c;医疗数据属于个人敏感数据&#xff0c;一旦被泄露或被恶…

前端开发中的常见问题及解决方法

前端开发是一个充满挑战和乐趣的领域。然而&#xff0c;在开发过程中&#xff0c;开发者常常会遇到各种各样的问题。本文将介绍一些前端开发中常用或者经常遇到的问题&#xff0c;并提供相应的解决方法&#xff0c;帮助你提高开发效率和解决问题的能力。 一. 页面布局问题 问题…

ArcTs布局入门03——层叠布局(Stack)

如果你也对鸿蒙开发感兴趣&#xff0c;加入“Harmony自习室”吧&#xff01; 扫描下面的二维码关注公众号。 1、概述 叠布局&#xff08;StackLayout&#xff09;用于在屏幕上预留一块区域来显示组件中的元素&#xff0c;提供元素可以重叠的布局。层叠布局通过Stack容器组件实…

机械拆装-基于Unity-装配功能的实现

目录 1. 装配场景的相机控制 2. 鼠标拖拽和旋转功能的实现 2.1 鼠标拖拽 2.2 物体旋转 3. 零件与装配位置的对应关系 4. 轴向装配的准备位置 5. 装配顺序的实现 5.1 标签提示 5.2 定义一个变量记录步骤数值 1. 装配场景的相机控制 开始装配功能时&#xff0c;需要将相机调…

k8s公网集群安装(1.23.0)

网上搜到的公网搭建k8s都不太一致, 要么说的太复杂, 要么镜像无法下载, 所以写了一个简洁版,小白也能一次搭建成功 使用的都是centos7,k8s版本为1.23.0 使用二台机器搭建的, 三台也是一样的思路1.所有节点分别设置对应主机名 hostnamectl set-hostname master hostnamectl set…

QT4-QT5(6)-const char* QString 乱码转换

我简单粗暴的给出个结论&#xff1a; QString GBK编码正常&#xff0c;可以转UTF-8编码&#xff0c;但会有少量乱码。 const char* 编码就不要转编码&#xff0c;转哪个都是乱码。 UTF-8.cpp 下 1.QString GBK->UTF-8 2.const char * GBK->UTF-8 const char *…

ViewBinding的使用(因为kotlin-android-extensions插件的淘汰)

书籍&#xff1a; 《第一行代码 Android》第三版 开发环境&#xff1a; Android Studio Jellyfish | 2023.3.1 问题&#xff1a; 3.2.4在Activity中使用Toast章节中使用到了kotlin-android-extensions插件,但是该插件已经淘汰,根据网上了解,目前使用了新的技术VewBinding替…

Shiro框架

入门概述 1 shiro是什么? Apache Shiro 是一个功能强大且易于使用的 Java 安全(权限)框架。Shiro 可以完成&#xff1a;认证、授权、加密、会话管理、与 Web 集成、缓存 等。借助 Shiro 您可以快速轻松地保护任何应用程序——从最小的移动应用程序到最大的 Web 和企业应用程…

Spring之spring的单例bean是线程安全的吗

Spring单例bean是线程安全的吗&#xff1f; 不是线程安全的。 1、Bean的作用域 Service Scope("singleton") public class UserServiceImpl implements UserService{ } singleton &#xff08;默认&#xff09;&#xff1a;bean在每个Spring IOC容器中只有一个实例…
最新文章