【学习记录24】vue3自定义指令

一、在单vue文件中直接使用

1、html部分

<template><divstyle="height: 100%;"v-loading="loading"><ul><li v-for="item in data">{{item}} - {{item * 2}}</li></ul></div>
</template>

 2、js部分

<script setup>import {ref} from 'vue'// loading图片路径import imgSrc from '@/views/loading.gif'const data = ref(1)// 定义loading初始值为true,插入loadingconst loading = ref(true)// 单页面自定义指令直接使用v开头驼峰命名变量,如下const vLoading = {// dom加载完,在使用v-loading指令的html节点里添加dommounted (el, binding) {const div =  document.createElement('div')div.className = 'loading'const img = document.createElement('img')img.src = imgSrcimg.width = 40div.appendChild(img)el.appendChild(div)},updated(el, binding) {// 当值loading绑定的值为false的时候删除domif (!binding.value) {const loadingDom = document.querySelector('.loading')el.removeChild(loadingDom)}}}// 模拟异步数据,2秒后loading值为falsesetTimeout(() => {loading.value = falsedata.value = 50}, 2000)
</script>

3、实现效果

二、全局注册使用

1、html部分

<template><divstyle="height: 100%;"v-loading="loading"><ul><li v-for="item in data">{{item}} - {{item * 2}}</li></ul></div>
</template>

2、js部分

在components下创建loading文件夹,在loading文件夹里创建directive.js 

// direcitve.jsimport imgSrc from "@/views/loading.gif";const loadingDirective = {mounted(el, binding) {const div = document.createElement('div')div.className = 'loading'const img = document.createElement('img')img.src = imgSrcimg.width = 40div.appendChild(img)el.appendChild(div)},updated(el, binding) {if (!binding.value) {const loadingDom = document.querySelector('.loading')el.removeChild(loadingDom)}}
}
export default loadingDirective

在main.js中全局注册指令

// 引入loading
import loadingDirective from './components/loading/directive'let app = createApp(App)app.use(ElementPlus).use(router)
app.directive('loading', loadingDirective) // 全局注册
app.mount('#app')

三、使用vue组件文件实现自定义指令

1、在components下创建loading文件夹,在loading文件夹里创建directive.js

2、在loading文件夹里创建loading.vue

3、在loading文件夹里放入一张GIF图(loading.gif)

 

 1、loading.vue文件源码

// loading.vue<template><div class="loading"><div class="loading-content"><img src="./loading.gif" width="24" height="24" alt=""><p class="desc">{{title}}</p></div></div>
</template><script setup>// vue3 写法import { ref, defineExpose } from 'vue'const title = ref('正在加载...')const setTitle = (t) => {title.value = t}// 导出setTitle方法defineExpose({setTitle})// vue2 写法// export default {//   name: 'loading',//   data() {//     return {//       title: '正在加载...'//     }//   },//   methods: {//     setTitle(title) {//       this.title = title//     }//   }// }
</script><style scoped lang="scss">.loading {position: absolute;top: 50%;left: 50%;transform: translate3d(-50%, -50%, 0);.loading-content {text-align: center;.desc {line-height: 20px;font-size: $font-size-small;color: $color-text-l;}}}
</style>

2、js部分(directive.js) 

// directive.js// 引入vue方法createApp
import {createApp} from 'vue'
// 引入添加dom和删除dom的方法
import {addClass, removeClass} from '@/assets/js/dom'const relativeCls = 'g-relative'
const loadingDirective = {mounted (el, binding) {// 创建一个loading的vue实例const app = createApp(Loading)// 挂载loading.vue 到div DOM上const instance = app.mount(document.createElement('div'))// 把instance挂到要用指令的element下el.instance = instanceconst title = binding.arg// 如果传了title就重新设置title的值if (typeof title !== 'undefined') {instance.setTitle(title)}// 指令绑定的值为true把自定义的vue实例下的dom节点$el添加到el下if (binding.value) {append(el)}},// 指令绑定的值更新以后updated (el, binding) {const title = binding.argif (typeof title !== 'undefined') {el.instance.setTitle(title)}if (binding.value !== binding.oldValue) {// 指令绑定的值为true添加指令dom,否则删除指令的dombinding.value ? append(el) : remove(el)}}
}function append(el) {const style = getComputedStyle(el)// 如果要绑定的dom没有定位就添加一个有定位的classiif (!['absolute', 'fixed', 'relative'].includes(style.position)) {addClass(el, relativeCls)}el.appendChild(el.instance.$el)
}function remove(el) {removeClass(el, relativeCls)el.removeChild(el.instance.$el)
}export default loadingDirective

 3、@/assets/js/dom.js源码

export function addClass(el, className) {if (!el.classList.contains(className)) {el.classList.add(className)}
}export function removeClass(el, className) {el.classList.remove(className)
}

4、main.js源码 

// 引入loading
import loadingDirective from './components/loading/directive'let app = createApp(App)app.use(ElementPlus).use(router)
app.directive('loading', loadingDirective) // 全局注册
app.mount('#app')

四、进阶(有多个自定义指令)

1、封装一个通用的js

把自定义组件的方法拎出来单独弄一个js文件,我习惯放在src/assets/js/create-my-like-directive.js

// create-my-like-directive.jsimport { createApp } from 'vue'
import { addClass, removeClass } from '@/assets/js/dom'const relativeCls = 'g-relative'export default function createMyLikeDirective (Comp) { // 改动的地方,变成可传参的方法return {mounted (el, binding) {const app = createApp(Comp)  // 传入变动参数const instance = app.mount(document.createElement('div'))const name = Comp.nameif (!el[name]) {el[name] = {}}// 把实例挂载到dom的name下,防止多个自定义指令互相影响干扰出现bugel[name].instance = instanceconst title = binding.argif (title) {instance.setTitle(title)}if (binding.value) {append(el)}},updated (el, binding) {const title = binding.argconst name = Comp.nameif (title) {el[name].instance.setTitle(title)}if (binding.value !== binding.oldValue) {binding.value ? append(el) : remove(el)}}}function append (el) {const style = getComputedStyle(el)const name = Comp.nameif (!['absolute', 'fixed', 'relative'].includes(style.position)) {addClass(el, relativeCls)}el.appendChild(el[name].instance.$el)}function remove (el) {const name = Comp.nameremoveClass(el, relativeCls)el.removeChild(el[name].instance.$el)}
}

2、directive.js修改

import Loading from './loading.vue'
import createMyLikeDirective from '@/assets/js/create-my-like-directive'
// 如果有不同的自定义好的vue文件,Loading变为别的vue文件即可
const loadingDirective = createMyLikeDirective(Loading)export default loadingDirective

 3、main.js修改

// 引入loading
import loadingDirective from './components/loading/directive'// 引入其他的
import AAAA from './components/AAAA/directive'
import BBBB from './components/BBBB/directive'let app = createApp(App)app.use(ElementPlus).use(router)
app.directive('loading', loadingDirective) // 全局注册
app.directive('aaaa', AAAA) // 全局注册
app.directive('bbbb', BBBB) // 全局注册
app.mount('#app')

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

YOLOV7剪枝流程

YOLOV7剪枝流程 1、训练 1&#xff09;划分数据集进行训练前的准备&#xff0c;按正常的划分流程即可 2&#xff09;修改train.py文件 第一次处在参数列表里添加剪枝的参数&#xff0c;正常训练时设置为False&#xff0c;剪枝后微调时设置为True parser.add_argument(--pr…

Linux安装ossutil工具且在Jenkins中执行shell脚本下载文件

测试中遇到想通过Jenkins下载OSS桶上的文件&#xff0c;要先在linux上安装ossutil工具&#xff0c;记录安装过程如下&#xff1a; 一、下载安装ossutil&#xff0c;使用命令 1.下载&#xff1a;wget https://gosspublic.alicdn.com/ossutil/1.7.13/ossutil64 2.一定要赋权限…

不同波段的热红外相机特点与应用

热红外相机通常分为三个主要波段&#xff1a; 长波红外 (LWIR) 波段&#xff1a;这个波段的范围大约在8-14微米。长波红外相机主要用于工业、安全监控和夜视设备。这个波段的特点是对温度变化非常敏感&#xff0c;能够在没有任何光源的情况下工作。 中波红外 (MWIR) 波段&…

使用的uview 微信高版本 头像昵称填写能力

<template><view><button class"cu-btn block bg-blue margin-tb-sm lg" tap"wxGetUserInfo">一键登录</button><view><!-- 提示窗示例 --><u-popup :show"show" background-color"#fff">&…

小程序开发实战案例五 | 小程序如何嵌入H5页面

在接入小程序过程中会遇到需要将 H5 页面集成到小程序中情况&#xff0c;今天我们就来聊一聊怎么把 H5 页面塞到小程序中。 本篇文章将会从下面这几个方面来介绍&#xff1a; 小程序承载页面的前期准备小程序如何承载 H5小程序和 H5 页面如何通讯小程序和 H5 页面的相互跳转 小…

Ubuntu重启后进入initramfs导致无法开机

今晚&#xff0c;我的电脑意外关机&#xff0c;重新开机后打开了虚拟机后出现initramfs&#xff0c;一直无法开机。该虚拟机使用的是 vm17,系统是ubuntu20, 解决方案 使用如下命令查看和识别磁盘、分区或文件系统的信息 在initramfs后面输入 fsck /dev/sdb4 ,即修复上面损坏的…

WINCC读写EXCEL-VBS

原创 RENHQ WINCC 关于VBS操作EXCEL的文档不管在论坛上还是在网上&#xff0c;相关的脚本已经很多&#xff0c;但是依然有很多人在问这个问题&#xff0c;于是把我以前在论坛上发的一个集合帖子的脚本拿来&#xff0c;重新开个帖子&#xff0c;如果再有人问的话&#xff0c;可…

kali下-MSF-ftp_login模块破解FTP账号及密码

一、环境准备 两台设备在同一个网络内 一台kali系统&#xff1a;192.168.10.128 一台winserver2016&#xff1a;192.168.10.132 二、MSF介绍 metasploit 全称是The Metasploit Framework&#xff0c;又称MSF&#xff0c;是Kali 内置的一款渗透测试框架&#xff0c;也是全球…

MySQL中锁的概述

按照锁的粒度来分可分为&#xff1a;全局锁&#xff08;锁住当前数据库的所有数据表&#xff09;&#xff0c;表级锁&#xff08;锁住对应的数据表&#xff09;&#xff0c;行级锁&#xff08;每次锁住对应的行数据&#xff09; 加全局锁&#xff1a;flush tables with read lo…

基于springboot+vue的图书个性化推荐系统(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容&#xff1a;毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目背景…

线程同步--生产者消费者模型

文章目录 一.条件变量pthread线程库提供的条件变量操作 二.生产者消费者模型生产者消费者模型的高效性基于环形队列实现生产者消费者模型中的数据容器 一.条件变量 条件变量是线程间共享的全局变量,线程间可以通过条件变量进行同步控制条件变量的使用必须依赖于互斥锁以确保线…

数据结构之栈的基本操作

该顺序栈涉及到了存储整型数据的顺序栈还有存储字符型数据的顺序栈 实现的功能有&#xff1a;入栈、出栈、判断是否为空栈、求栈的长度、清空栈、销毁栈、得到栈顶元素 此外根据上述功能&#xff0c;编写了数值转换&#xff08;十进制转化八进制&#xff09;方法、括号匹配方法…

服务器变矿机,该如何应对?

开始 恶意的挖矿程序会导致服务器cpu的异常占用&#xff0c;很让人讨厌。起初&#xff0c;我只是使用top命令显示出占用cpu不正常的进程&#xff0c;发现其中一个进程占用了百分之九十九点几&#xff0c;然后通过kill -9 <PID>命令干掉它。但总是过不了几天&#xff0c;…

深度学习:探索人工智能的前沿

1. 引言 1.1 人工智能的演进 人工智能&#xff08;Artificial Intelligence&#xff0c;简称AI&#xff09;是一门研究如何使计算机能够执行通常需要人类智能的任务的领域。从早期的符号推理到现代的深度学习&#xff0c;人工智能经历了漫长的发展过程。 20世纪50年代&#xff…

MySQL 基于创建时间进行RANGE分区

MySQL是一款广泛使用的关系型数据库。在MySQL中&#xff0c;大量数据场景提高查询效率是非常关键的&#xff0c;所以&#xff0c;对数据表进行分区是一个很好的选择。 在创建分区表之前&#xff0c;需要了解一下MySQL分区的基本概念。MySQL分区可以将一个大表分成多个小表&…

K8s 网关选型血泪史

Sealos 公有云几乎打爆了市面上所有主流的开源网关&#xff0c;本文可以给大家很好的避坑&#xff0c;在网关选型方面做一些参考。 Sealos Cloud 的复杂场景 Sealos 公有云上线以来&#xff0c;用户呈爆发式增长&#xff0c;目前总共注册用户 8.7w&#xff0c;每个用户都去创…

FFmpeg之AVFormat

文章目录 一、概述二、解封装流程三、重要结构体3.1、AVFormatContext3.2、AVInputFormat3.3、AVOutputFormat3.4、AVStream 四、重要函数分析4.1、avformat_alloc_context4.2、avformat_open_input4.2.1、init_input4.2.2、av_probe_input_format2 4.3、avformat_find_stream_…

Flutter开发之蓝牙链接传输数据

本文使用的是flutter_blue_plus插件来实现链接蓝牙之后&#xff0c;和设备直接实现数据互相传输的功能。 1、配置蓝牙权限 iOS权限设置<key>NSBluetoothAlwaysUsageDescription</key><string>App需要您的同意,才能访问蓝牙,进行设备连接,数据通讯服务</…

LeetCode 热题 100 | 双指针(上)

目录 1 283. 移动零 2 11. 盛最多水的容器 3 15. 三数之和 菜鸟做题第一周&#xff0c;语言是 C 1 283. 移动零 解题思路&#xff1a; 两个指针一前一后遍历数组前者永远指向 0&#xff0c;后者永远在寻找非 0 数的路上后者找到一个非 0 数就和前者进行一个数值交换 …

Qt应用开发(安卓篇)——Hello Qt On Android

一、前言 这一篇从实际出发&#xff0c;讲述如何创建、编译和部署Qt On Android项目。 二、ADB调试 ADB的全称为Android Debug Bridge&#xff0c;就是起到调试桥的作用&#xff0c;主要用于连接计算机与Android 设备&#xff0c;以便进行调试和数据传输。ADB 可以实现以下主要…