Vue自定义指令原来这么简单

Vue自定义指令原来这么简单看完这篇轻松掌握 Vue 自定义指令 助你面试 solo 面试官 项目中帮你处理更多难解情况 学习 Vue 指令原来这么简单

本篇学习目标

  1. 能够了解组件进阶知识
  2. 能够掌握自定义指令创建和使用
  3. 能够完成tabbar案例的开发

1. 组件进阶

1.0 组件进阶 – 动态组件

目标: 多个组件使用同一个挂载点,并动态切换,这就是动态组件

需求: 完成一个注册功能页面, 2个按钮切换, 一个填写注册信息, 一个填写用户简介信息

  1. 准备被切换的 – UserName.vue / UserInfo.vue 2个组件
  2. 引入到UseDynamic.vue注册
  3. 准备变量来承载要显示的”组件名”
  4. 设置挂载点, 使用is属性来设置要显示哪个组件
  5. 点击按钮 – 修改comName变量里的”组件名”
 
    

在App.vue – 引入01_UseDynamic.vue并使用显示

总结: vue内置component组件, 配合is属性, 设置要显示的组件名字

1.1 组件进阶 – 组件缓存

目标: 组件切换会导致组件被频繁销毁和重新创建, 性能不高

使用Vue内置的keep-alive组件, 可以让包裹的组件保存在内存中不被销毁

演示1: 可以先给UserName.vue和UserInfo.vue 注册created和destroyed生命周期事件, 观察创建和销毁过程

演示2: 使用keep-alive内置的vue组件, 让动态组件缓存而不是销毁

语法:

​ Vue内置的keep-alive组件 包起来要频繁切换的组件

02_UseDynamic.vue

 
  

补充生命周期:

  • activated – 激活
  • deactivated – 失去激活状态

总结: keep-alive可以提高组件的性能, 内部包裹的标签不会被销毁和重新创建, 触发激活和非激活的生命周期方法

1.2 组件进阶 – 激活和非激活

目标: 被缓存的组件不再创建和销毁, 而是激活和非激活

补充2个钩子方法名:

​ activated – 激活时触发

​ deactivated – 失去激活状态触发

1.3 组件进阶 – 组件插槽

目标: 用于实现组件的内容分发, 通过 slot 标签, 可以接收到写在组件标签内的内容

vue提供组件插槽能力, 允许开发者在封装组件时,把不确定的部分定义为插槽

语法口诀:

  1. 组件内用占位
  2. 使用组件时夹着的地方, 传入标签替换slot

03/Pannel.vue – 组件(直接复制)

 
     

views/03_UserSlot.vue – 使用组件(直接复制)

框: 在这个基础重复使用组件

 
     

views/03_UseSlot.vue – 组件插槽使用

 
    

总结: 组件内容分发技术, slot占位, 使用组件时传入替换slot位置的标签

1.4 组件进阶 – 插槽默认内容

目标: 如果外面不给传, 想给个默认显示内容

口诀: 夹着内容默认显示内容, 如果不给插槽slot传东西, 则使用夹着的内容在原地显示

 
  
    默认内容 
   

1.5 组件进阶 – 具名插槽

目标: 当一个组件内有2处以上需要外部传入标签的地方

传入的标签可以分别派发给不同的slot位置

要求: v-slot一般用跟template标签使用 (template是html5新出标签内容模板元素, 不会渲染到页面上, 一般被vue解析内部标签)

components/04/Pannel.vue – 留下具名slot

 
   

views/04_UseSlot.vue使用

 
    

v-slot可以简化成#使用

v-bind可以省略成: v-on: 可以省略成@ 那么v-slot: 可以简化成#

总结: slot的name属性起插槽名, 使用组件时, template配合#插槽名传入具体标签

1.6 组件进阶 – 作用域插槽

目标: 子组件里值, 在给插槽赋值时在父组件环境下使用

复习: 插槽内slot中显示默认内容

例子: 默认内容在子组件中, 但是父亲在给插槽传值, 想要改变插槽显示的默认内容

口诀:

  1. 子组件, 在slot上绑定属性和子组件内的值
  2. 使用组件, 传入自定义标签, 用template和v-slot=“自定义变量名”
  3. scope变量名自动绑定slot上所有属性和值

components/05/Pannel.vue – 定义组件, 和具名插槽, 给slot绑定属性和值

 
    

views/05_UseSlot.vue

 
    

总结: 组件内变量绑定在slot上, 然后使用组件v-slot=“变量” 变量上就会绑定slot身上属性和值

1.7 组件进阶 – 作用域插槽使用场景

目标: 了解作用域插槽使用场景, 自定义组件内标签+内容

案例: 封装一个表格组件, 在表格组件内循环产生单元格

准备MyTable.vue组件 – 内置表格, 传入数组循环铺设页面, 把对象每个内容显示在单元格里

准备UseTable.vue – 准备数据传入给MyTable.vue使用

components/06/MyTable.vue – 模板(直接复制)

 
    

views/06_UseTable.vue – 准备数据, 传入给MyTable.vue组件里循环使用

list: [ { 
    name: "小传同学", age: 18, headImgUrl: "http://yun.itheima.com/Upload/./Images//603f2d.jpg", }, { 
    name: "小黑同学", age: 25, headImgUrl: "http://yun.itheima.com/Upload/./Images//6040b101a18ef.jpg", }, { 
    name: "智慧同学", age: 21, headImgUrl: "http://yun.itheima.com/Upload/./Images//603e0142e535f.jpg", }, ], 

正确做法:

​ 在MyTable.vue的td中准备占位, 但是外面需要把图片地址赋予给src属性,所以在slot上把obj数据绑定

components/06/MyTable.vue – 正确代码

 
    

​ 在UseTable使用MyTable的时候, template上v-slot绑定变量, 传入img组件设置图片地址

 
     

总结: 插槽可以自定义标签, 作用域插槽可以把组件内的值取出来自定义内容

2. 自定义指令

自定义指令文档

除了核心功能默认内置的指令 (v-modelv-show),Vue 也允许注册自定义指令。 v-xxx

html+css的复用的主要形式是组件

你需要对普通 DOM 元素进行底层操作,这时候就会用到自定义指令

2.0 自定义指令-注册

目标: 获取标签, 扩展额外的功能

局部注册和使用

07_UseDirective.vue – 只能在当前组件.vue文件中使用

 
     

全局注册

在main.js用 Vue.directive()方法来进行注册, 以后随便哪个.vue文件里都可以直接用v-fofo指令

// 全局指令 - 到处"直接"使用 Vue.directive("gfocus", { 
    inserted(el) { 
    el.focus() // 触发标签的事件方法 } }) 

总结: 全局注册自定义指令, 哪里都能用, 局部注册, 只能在当前vue文件里用

2.1 自定义指令-传值

目标: 使用自定义指令, 传入一个值

需求: 定义color指令-传入一个颜色, 给标签设置文字颜色

main.js定义处修改一下

// 目标: 自定义指令传值 Vue.directive('color', { 
    inserted(el, binding) { 
    el.style.color = binding.value }, update(el, binding) { 
    el.style.color = binding.value } }) 

Direct.vue处更改一下

修改文字颜色

总结: v-xxx, 自定义指令, 获取原生DOM, 自定义操作

3. 案例-tabbar

知识点:

  • 组件封装
  • 动态组件
  • keep-alive
  • 作用域插槽
  • 自定义指令

3.0 案例-tabbar-初始化项目

  • 需求: 从0新建项目, 拆分组件, 创建使用

组件分析:

  • 组件拆分:
    • MyHeader.vue – 复用之前的
    • MyTabBar.vue – 底部导航
    • MyTable.vue – 封装表格
  • 三个页面
    • -MyGoodsList.vue – 商品页
    • MyGoodsSearch.vue – 搜索页
    • -MyUserInfo.vue – 用户信息页

思路分析:

​ ①: vue create tabbar-demo

​ ②: yarn add less less-loader@5.0.0 -D

​ ③: yarn add bootstrap axios 并在main.js 引入和全局属性

​ ④: 根据需求-创建需要的页面组件

​ ⑤: 把昨天购物车案例-封装的MyHeader.vue文件复制过来复用

​ ⑥: 从App.vue – 引入组织相关标签

新建工程:

vue create tabbar-demo yarn add less less-loader@5.0.0 -D yarn add bootstrap axios 

在main.js中引入bootStrap.css和字体图标样式

import "bootstrap/dist/css/bootstrap.css" import "./assets/fonts/iconfont.css" 

创建/复制如下文件

从昨天案例中-直接复制过来components/MyHeader.vue

components/MyTabBar.vue

views/MyGoodsList.vue

views/MyGoodsSearch.vue

views/MyUserInfo.vue

components/MyTable.vue

3.1 案例-tabbar-底部封装

目标: 实现MyTabBar.vue组件

在这里插入图片描述

  • 需求: 把底部导航也灵活封装起来

分析:

​ ①: 基本标签+样式(md里复制)

​ ②: 为tabbar组件指定数据源

​ ③: 数据源最少2个, 最多5个(validator)

​ ④: 从App.vue给MyTabBar.vue传入底部导航的数据

​ ⑤: MyTabBar.vue中循环展示

App.vue-数组准备

tabList: [ { 
    iconText: "icon-shangpinliebiao", text: "商品列表", componentName: "MyGoodsList" }, { 
    iconText: "icon-sousuo", text: "商品搜索", componentName: "MyGoodsSearch" }, { 
    iconText: "icon-user", text: "我的信息", componentName: "MyUserInfo" } ] 

MyTabBar.vue – 标签模板

 
     

MyTabBar.vue正确代码(不可复制)

 
    

不要忘了把tabList数组从App.vue -> MyTabBar.vue

3.2 案例-tabbar-底部高亮

目标: 点击底部导航实现高亮效果

  • 需求: 点击底部实现高亮效果

分析:

​ ①: 绑定点击事件, 获取点击的索引

​ ②: 循环的标签设置动态class, 遍历的索引, 和点击保存的索引比较, 相同则高亮

MyTabBar.vue(正确代码)

 
    

3.3 案例-tabbar-组件切换

目的: 点击底部导航, 切换页面组件显示

需求: 点击底部切换组件

分析:

​ ①: 底部导航传出动态组件名字符串到App.vue

​ ②: 切换动态组件is属性的值为要显示的组件名

效果演示:

在这里插入图片描述

补充: 给内容div.app- 设置上下内边距

App.vue - 引入并注册

 
     

MyTabBar.vue - 点击传递过来组件名

methods: { 
    btn(index, theObj) { 
    this.selIndex = index; // 点谁, 就把谁的索引值保存起来 this.$emit("changeCom", theObj.componentName); // 要切换的组件名传App.vue }, }, 

3.4 案例-tabbar-商品列表

目标: 为MyGoodsList页面, 准备表格组件MyTable.vue-铺设展示数据

  • 需求: 商品列表铺设页面

分析:

​ ①: 封装MyTable.vue – 准备标签和样式

​ ②: axios在MyGoodsList.vue请求数据回来

​ ③: 请求地址: https://www.escook.cn/api/goods

​ ④: 传入MyTable.vue中循环数据显示

​ ⑤: 给删除按钮添加bootstrap的样式: btn btn-danger btn-sm

MyTable.vue - 准备table整个表格标签和样式(可复制)

 
     

使用axios请求数据, 把表格页面铺设出来

main.js - 注册axios配置默认地址

import axios from "axios"; axios.defaults.baseURL = "https://www.escook.cn"; 

MyGoodsList.vue - 使用axios请求数据, 把数据传入给MyTable.vue里循环铺设

 
    

MyTable.vue里正确代码(不可复制)

 
     

3.5_案例-tabbar-商品表格-插槽

目标: 使用插槽技术, 和作用域插槽技术, 给MyTable.vue组件, 自定义列标题, 自定义表格内容

  • 需求: 允许用户自定义表格头和表格单元格内容

分析:

​ ①: 把MyTable.vue里准备slot

​ ②: 使用MyTable组件时传入具体标签

步骤:

  1. 提高组件复用性和灵活性, 把表格列标题thead部分预留标签, 设置name属性
  2. 使用MyTable.vue时, 传入列标题标签
  3. 表格内容td部分也可以让组件使用者自定义, 也给tbody下tr内留好标签和name属性名
  4. 使用插槽需要用到插槽内的obj对象上的数据, 使用作用域插槽技术

在这里插入图片描述

MyTable.vue - 留好具名插槽

 
    

MyGoodsList.vue 使用

 
     

3.6 案例-tabbar-商品表格-tags微标

目标: 把单元格里的标签, tags徽章铺设下

  • 需求: 标签列自定义显示

分析:

​ ①: 插槽里传入的td单元格

​ ②: 自定义span标签的循环展示-给予样式

bootstrap徽章: https://v4.bootcss.com/docs/components/badge/

MyGoodsList.vue - 插槽

 { 
   { str }}  

下面额外添加样式

 

3.7 案例-tabbar-商品表格-删除功能

目标: 点击删除对应这条数据

  • 需求: 点击删除按钮删除数据

分析:

​ ①: 删除按钮绑定点击事件

​ ②: 作用域插槽绑定id值出来

​ ③: 传给删除方法, 删除MyGoodsList.vue里数组里数据

提示: id在MyTable.vue里, 但是MyGoodsList.vue里要使用, 而且在插槽位置, 使用作用域插槽已经把整个obj对象(包含id)带出来了

MyTable.vue

 
   
  1. MyGoodsList.vue - 注册点击事件
 

​ 2. my-goods-list.vue 根据 id 删除

removeBtn(id){ let index = this.list.findIndex(obj => obj.id === id) this.list.splice(index, 1) }, 

3.8 案例-tabbar-添加tab

目标: 实现点击tab按钮, 出现输入框自动获取焦点, 失去焦点关闭input, 回车新增tag, esc清空内容

  • 需求1: 点击Tab, 按钮消失, 输入框出现
  • 需求2: 输入框自动聚焦
  • 需求3: 失去焦点, 输入框消失, 按钮出
  • 需求4: 监测input回车, 无数据拦截
  • 需求5: 监测input取消, 清空数据
  • 需求6: 监测input回车, 有数据添加

3.8.0 点击按钮消失, 输入框出现

MyGoodsList.vue - 标签位置添加

注意: 每个tab按钮和input都是独立变量控制, 那么直接在row身上的属性控制即可

  

3.8.1 input自动获取焦点

main.js - 定义全局自定义指令

// 全局指令 Vue.directive("focus", { 
    inserted(el){ 
    el.focus() } }) 

MyGoodsList.vue - 使用 v-focus指令

3.8.2 input失去焦点关闭input

监听input失去焦点事件, 让input消失

@blur="scope.row.inputVisible = false" 

3.8.3 input回车新增tag

监听input的回车事件, 如果无数据拦截代码

@keydown.enter="enterFn(scope.row)" 

事件处理函数

enterFn(obj){ 
    // 回车 // console.log(obj.inputValue); if (obj.inputValue.trim().length === 0) { 
    alert('请输入数据') return } obj.tags.push(obj.inputValue) // 表单里的字符串状态tags数组 obj.inputValue = "" } 

3.8.4 input框esc清空内容

@keydown.esc="scope.row.inputValue = ''" 

今日总结

  1. 动态组件的使用步骤
  2. 组件缓存使用步骤和作用
  3. 组件插槽默认使用
  4. 插槽默认显示的内容
  5. 多个插槽时, 具名插槽如何使用
  6. 作用域插槽如何使用以及目的
  7. 自定义命令如何使用
  8. 跟随视频完成tabbar案例

面试题

1. vue中solt的使用方式,以及solt作用域插槽的用法

<current-user> <template v-slot:default="slotProps"> { 
   { 
    slotProps.user.firstName }} </template> </current-user> // current-user组件, user属性和值, 绑定给slotProps上 <span> <slot v-bind:user="user"> { 
   { 
    user.lastName }} </slot> </span> 

扩展阅读: https://cn.vuejs.org/v2/guide/components-slots.html (了解即可, 一般用不上)

2. 跟keep-alive有关的生命周期是哪些?(必会)

1)前言:在开发Vue项目的时候,大部分组件是没必要多次渲染的,所以Vue提供了一个内置组件keep-alive来缓存组件内部状态,避免重新渲染,在开发Vue项目的时候,大部分组件是没必要多次渲染的,所以Vue提供了一个内置组件keep-alive来缓存组件内部状态,避免重新渲染

2)生命周期函数:在被keep-alive包含的组件/路由中,会多出两个生命周期的钩子:activated 与 deactivated。

1、activated钩子:在在组件第一次渲染时会被调用,之后在每次缓存组件被激活时调用。

2、Activated钩子调用时机: 第一次进入缓存路由/组件,在mounted后面,beforeRouteEnter守卫传给 next 的回调函数之前调用,并且给因为组件被缓存了,再次进入缓存路由、组件时,不会触发这些钩子函数,beforeCreate created beforeMount mounted 都不会触发

1、deactivated钩子:组件被停用(离开路由)时调用。

2、deactivated钩子调用时机:使用keep-alive就不会调用beforeDestroy(组件销毁前钩子)和destroyed(组件销毁),因为组件没被销毁,被缓存起来了,这个钩子可以看作beforeDestroy的替代,如果你缓存了组件,要在组件销毁的的时候做一些事情,可以放在这个钩子里,组件内的离开当前路由钩子beforeRouteLeave => 路由前置守卫 beforeEach =>全局后置钩子afterEach => deactivated 离开缓存组件 => activated 进入缓存组件(如果你进入的也是缓存路由)

3. 自定义指令(v-check、v-focus)的方法有哪些?它有哪些钩子函数?还有哪些钩子函数参数?(必会)

​ 全局定义指令:在vue对象的directive方法里面有两个参数,一个是指令名称,另外一个是函数。组件内定义指令:directives

​ 钩子函数:bind(绑定事件触发)、inserted(节点插入的时候触发)、update(组件内相关更新)

​ 钩子函数参数:el、binding

4. is这个特性你有用过吗?主要用在哪些方面?(高薪常问)

 1)动态组件

​ , componentName可以是在本页面已经注册的局部组件名和全局组件名,也可以是一个组件的选项对象。 当控制componentName改变时就可以动态切换选择组件。

 2)is的用法

​ 有些HTML元素,诸如 <ul><ol><table><select>,对于哪些元素可以出现在其内部是有严格限制的。 ​ 而有些HTML元素,诸如 <li><tr><option>,只能出现在其它某些特定的元素内部。 
<ul><card-list></card-list></ul> 

​ 所以上面会被作为无效的内容提升到外部,并导致最终渲染结果出错。应该这么写:

<ul><li is="cardList"></li></ul> 
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/199879.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月20日 下午12:00
下一篇 2026年3月20日 下午12:01


相关推荐

  • TimerTask倒计时

    TimerTask倒计时引导页 TimerTask 倒计时 packagecom umeng soexample importandroi content Intent importandroi support v7 app AppCompatAct importandroi os Bundle importandroi widget ImageView importandroi

    2026年3月17日
    2
  • 90后潮汕小伙打造的AI公司:获得阿里腾讯投资,估值超100亿美元

    90后潮汕小伙打造的AI公司:获得阿里腾讯投资,估值超100亿美元

    2026年3月12日
    3
  • MATLAB fmincon函数 进阶资料(磕盐记录)

    MATLAB fmincon函数 进阶资料(磕盐记录)初级资料 Matlab 求解非线性规划 fmincon 函数的用法总结完备资料 MathWork

    2026年3月17日
    1
  • 不装了、摊牌了,我们要搞事情

    不装了、摊牌了,我们要搞事情一起在技术的海洋里狗刨~

    2022年7月25日
    13
  • 外汇mt4和mt5的区别_鑫圣金业mt4平台下载

    外汇mt4和mt5的区别_鑫圣金业mt4平台下载这两个交易平台之间存在巨大差异。让我们看看它们之间的显着差异。那么让我们看看mt4与mt5之间的差异。mt4和mt5的下载方式差别不大,都可以在https://www.qiejf.cn/下载和安装。主要是在功能上有区别,下面详细来讲解一下。MT4和MT5交易平台的区别:  MT4仅提供外汇交易,但另一方面,MT5使交易者可以访问货币以外的差价合约、股票和期货。  这取决于交易者决定交易什么,并在此基础上,他们可以选择他们的交易平台。MT4始终是我的首要任务。它简单、灵活,让我能够根据自

    2022年8月15日
    5
  • JMM内存模型介绍「建议收藏」

    JMM内存模型介绍「建议收藏」一、JMM的定义1.什么是JMM《Java虚拟机规范》中曾试图定义一种“Java内存模型”(JavaMemoryModel简称JMM)来屏蔽各种硬件和操作系统的内存访问差异,以实现让Java程序在各种平台下都能达到一致的内存访问效果。Java内存模型是一种抽象的概念,并不真实存在,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(包括实例字段,静态字段和构成数组对象的元素)的访问方式。JMM是围绕原子性,有序性、可见性展开。2.主内存与工作内存Java内存模型的主要目的是定义程

    2025年7月14日
    6

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号