Vue: 事件修饰符, 键盘事件, 鼠标事件,计算属性

news2025/4/12 17:52:41

目录

事件修饰符

阻止默认事件

阻止冒泡

允许触发一次

捕获模式

self

passive

键盘事件

keyup & keydown

按键别名

注意tab

注意系统按键

自定义按键

鼠标事件 

简介

鼠标焦点事件

计算属性

差值语法实现

methods实现 

computed实现

get()

set()

总结 

计算属性简写


事件修饰符

        使用v-on, 表示即将发生的事件, 例如点击事件click, 下面是一个例子:

    <div id="app">
        <h5 v-on:click="showInfo">点击我</h5>
        <a href="https://www.baidu.com" v-on:click="showInfo">点我</a>
    </div>

        v-on 可以简写为: @

    <div id="app">
        <h5 @click="showInfo">点击我</h5>
        <a href="https://www.baidu.com" @click="showInfo">点我</a>
    </div>

阻止默认事件

        点击a标签, 会执行click中的showInfo方法, 随后还会执行a中的默认跳转行为, 如果不想这类行为事件发生, 可以使用事件修饰符prevent来阻止进行:

        <a href="https://www.baidu.com" @click.prevent="showInfo">点我</a>

        当然可以在Vue实例中定义方法的时候, 传入event, 然后调用event的preventDefault方法:

    new Vue({
        el : '#app',
        data : {
            name : '王者荣耀'
        },
        methods : {
            showInfo(e) {
                e.preventDefault()
                alert('你好')
            }
        }
    })

阻止冒泡

        既然是点击, 那么如果一个区域里面存在两个点击事件. 例如:

<body>
    <div id="app">
        <div @click="testDiv" id="out">
            <a href="https://www.baidu.com" @click.prevent="showInfo">点我</a>
        </div>
    </div>
    <style>
        #out {
            width: 200px;
            height: 200px;
            border: solid blue 1px;
        }
    </style>
</body>
<script>
    new Vue({
        el : '#app',
        data : {
            name : '王者荣耀'
        },
        methods : {
            showInfo(e) {
                e.preventDefault()
                alert('你好')
            },
            testDiv() {
                alert('这是最外层的div')
            }
        }
    })
</script>

 点击a标签那么就会出现:

        首先出现:

        然后出现:  

        发现, 本来我的本意只想点击这个a标签, 但是因为层叠关系(冒泡), 就导致点击了a的同时, 也点击了div标签, 所以为了阻止这种情况, 使用event方法: event.stopPropagation() >>>

    new Vue({
        el : '#app',
        data : {
            name : '王者荣耀'
        },
        methods : {
            showInfo(e) {
                e.preventDefault()
                e.stopPropagation();
                alert('你好')
            },
            testDiv(e) {
                alert('这是最外层的div')
            }
        }
    })

        简写, 在事件后面添加事件修饰符stop, 例如:

            <a href="https://www.baidu.com" @click.prevent.stop="showInfo">点我</a>

允许触发一次

        使用修饰符once :

            <a href="https://www.baidu.com" @click.once="showInfo">点我</a>

事件只触发一次

捕获模式

        事件执行, 先开始捕获, 由外到内, 捕获完之后开始执行, 从内到外执行, 也就是冒泡, 现在更改为捕获状态就执行, 也就是添加.capture, 例如@click.capture = "xxx"

self

        @click.self, 只有在event.target 是本身的时候, 才会触发点击事件

passive

        事件立即执行, 无序等待前置条件: .passive

键盘事件

keyup & keydown

  • @keyup
  • @keydown

这里使用keyup作为案例:

<body>
    <div id="app">
        <p>欢迎来到{{name}}</p>
        <input type="text" placeholder="按下回车提示输入" @keyup="showInfo">
    </div>
</body>

<script>
    new Vue({
        el :'#app',
        data : {
            name : 'world'
        },
        methods : {
            showInfo(e) {
                console.log(e.target.value);
            }
        }
    })
</script>

按下a建., 输入一个a, 这个时候控制台还没有输入, 但是松开a键之后, 控制台输出:

再按下一个a, 松开之后, 控制台输出 两个a :

keydown的用法类似, 这里不一一列举 

相同的keyup和keydown都有事件修饰符:

  • enter,  键盘输入之后并不会触发事件, 只有在键入的值是enter(回车)的时候, 才会触发事件, 例如:
        <div id="app">
            <p>欢迎来到{{name}}</p>
            <input type="text" placeholder="按下回车提示输入" @keyup.enter="showInfo">
        </div>
    
        new Vue({
            el :'#app',
            data : {
                name : 'world'
            },
            methods : {
                showInfo(e) {
                    console.log(e.target.value);
                }
            }
        })

    输入一个a并不会触发事件showInfo, 但是在输入a后,再按一个回车, 控制台就输出了一个a :

按键别名

  • 相同的还有很多其他的键盘事件修饰符, 这些别的修饰符是我们vue中常用的按键的别名:

    • 回车 : enter

    • 删除 : delete

    • 空格 : space

    • 退出 : esc

    • 换行 : tab

    • 上 : up

    • 下 : down

    • 左 : left

    • 右 : right

vue暂时没有提供别的按键别名, 可以使用按键原始的key进行绑定

系统按键修饰符:

  • ctrl
  • alt
  • shift
  • meta (win)

        系统修饰键配合keyup使用, 按下修饰键的同时, 再按下其他键, 随后释放其他键, 事件才被触发 : 解释 >>  因为同时按下之后, keyup是属于松开才触发事件, 但是现在除了ctrl等键之外, 其他的一些按键同样被按下了, 那么他们之间就有一个释放的顺序, 所以现在只有其他键先释放, 这个keyup状态才结束, 事件执行

        配合keydown使用, 直接按下之后就可以正常触发事件

注意tab

        有下面的例子:

<body>
    <div id="app">
        <p>欢迎来到{{name}}</p>
        <input type="text" placeholder="按下回车提示输入" @keyup.tab="showInfo">
    </div>
</body>

<script>
    new Vue({
        el :'#app',
        data : {
            name : 'world'
        },
        methods : {
            showInfo(e) {
                console.log(e.target.value);
            }
        }
    })
</script>

        这是一个使用键盘事件, 并且使用tab修饰这个键盘事件 :

        我们输入aaa, 然后按下tab:

        发现这个玩意没有输出到控制台:

        多按几次就会发现, 他并不是没有作用, 反而像是被什么锁定了一样, 不能在input中输入, :

        多按下几次就会发现, 他会在浏览器上切换焦点

        这个焦点被切走了. 那为什么没有触发事件??

        分析: 细心的你一定会发现, 这个tab是按下立即就生效的, 但是我们定义的键盘事件是keyup, 按下tab之后, 焦点立即调走, 还没来得及松开焦点就已经从输入框中切走, 那么vm也就无法捕捉到这个键入tab的行为, 更加捕获不到tab松开.

        改进 : 所以这个时候, 我们就可以使用keydown事件

注意系统按键

        同按键别名中的说明, 如果直接使用@keyup.ctrl .....

        比如同时按下ctrl + y, 然后松开y之后, 触发事件

        还有一个需要注意的是, 你可以指定松开哪一个键执行事件:

        @keyup.ctrl.y, 这个时候, 只有同时按下ctrl + y, 然后松开y才有用, 按下ctrl + z或者是其他的除y之外的按键都是没有用的. 

自定义按键

Vue.config.keyCodes.huiche = 13;

        <input type="text" placeholder="按下回车提示输入" @keyup.huihce="showInfo">

鼠标事件 

简介

        除了键盘事件, 那么肯定有鼠标事件的啦, 看看下面这几个鼠标事件的例子:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <script src="../js/vue.js"></script>
</head>
<body>
    <div id="app">
        <div id="testDiv" 
        @click="testClick" 
        @dblclick="testDBClick" 
        @mouseenter="testMouseEnter" 
        @mouseleave="testMouseLeave" 
        @mousedown="testMouseDown"
        @mouseup="testMouseUp"
        @mousemove="testMouseMove"
        >
        {{content}}
        </div>  
        <br>
        <hr>
    </div>
</body>
<script>
    var vue = new Vue({
        el : '#app',
        data : {
            content: '',
            value : null
        },
        methods : {
            testClick() { // 单机
                this.content =  '鼠标单击';
            },
            testDBClick() { // 鼠标双击
                this.content = '鼠标双击';
            },
            testMouseEnter() { // 鼠标进入事件
                this.content = '鼠标进入';
            },
            testMouseLeave() { // 鼠标离开事件
                this.content = '鼠标离开';
            }, testMouseDown() { // 鼠标按下事件
                this.content = '鼠标按下';
            }, testMouseUp() {
                this.content = '鼠标弹起';
            }, testMouseMove(event) { // 任何事件都可以添加事件event对象
                this.content = event.offsetX + '---' + event.offsetY; // 只要是@的事件都可以使用event
            }, testFocus() {
                this.value ='获得焦点'
            }
            , testBlur() {
                this.value = '失去焦点'
            }
        }
    });
</script>
<style>
    #testDiv {
        width: 150px;
        height: 150px;
        border: 2px solid green;
        text-align: center;
        line-height: 100%;
    }
</style>
</html>

 单击

双击

离开div

 这里不一一展示

总结鼠标点击事件 :

鼠标事件所表示触发的前置条件
@click鼠标单机触发
@dblclick鼠标双击
@mouseenter鼠标进入
@mouseleave鼠标离开
@mousedown鼠标按下
@mouseup鼠标弹起
@mousemove鼠标移动

鼠标焦点事件

  • @focus : 鼠标聚焦目标以触发事件
  • @blur : 鼠标聚焦跳出目标, 触发事件

案例: 

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <script src="../js/vue.js"></script>
</head>
<body>
    <div id="app">
        <input type="text" @focus="testFocus" @blur="testBlur" v-model="value">
    </div>
</body>
<script>
    var vue = new Vue({
        el : '#app',
        data : {
            content: '',
            value : null
        },
        methods : { 
            testFocus() {
                this.value ='获得焦点'
            }
            , testBlur() {
                this.value = '失去焦点'
            }
        }
    });
</script>
</html>

首先不进行任何操作:

点击这个输入框, 触发聚焦事件 :

鼠标点击输入框之外的内容:

单词解释: 

        blur也就是鼠标模糊不清, 也就是失去焦点. 

计算属性

差值语法实现

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <script src="../js/vue.js"></script>
</head>
<body>
    <div id="root">
        姓 : <input type="text" v-model="firstName"><br>
        名 : <input type="text" v-model="lastName"> <br>
        姓名 : <span>{{firstName}}{{lastName}}</span>
    </div>
</body>
<script>
    new Vue({
        el :'#root',
        data : {
            firstName : '张',
            lastName : '三'
        }
    })
</script>
</html>

methods实现 

        {{}}的里面写的是差值表达式, 所以这里直接使用方法名 + ()的形式, 表示调用这个函数的返回值

<body>
    <div id="root">
        姓 : <input type="text" v-model="firstName"><br>
        名 : <input type="text" v-model="lastName"> <br>
        姓名 : <span>{{fullName()}}</span>
    </div>
</body>
<script>
    new Vue({
        el: '#root',
        data: {
            firstName: '张',
            lastName: '三'
        },methods  :{
            fullName() {
                return '博子妹妹';
            }
        }
    })
</script>

        这里将'博子妹妹' 修改为 this.firstName + '-' + lastName的形式, 即可联动显示, 因为是双向绑定, 修改 姓 之后, 就会修改data中的值, 然后vm捕捉到这个修改, 然后用到这个name的地方就会更新 . 以后任何修改data的数据操作之后, vm都会重新解析一次.

computed实现

get()

        不用方法, 也不使用插值语法来表示, 那么就使用vm实例中的computed字段来实现,

        computed, 本就有计算的意思:

       在computed的字段中定义fullName, 然后使用这个fullName去计算出 全名, 使用的时候, 可以像data中的数据那样去访问这个fullName, 例如: {{fullName}}, 访问这个fullName, vm会调用computed中的fullName的get()方法, 然后将其返回值最为fullName的值.

    new Vue({
        el: '#root',
        data: {
            firstName: '张',
            lastName: '三'
        },
        computed : {
            fullName : { // 这个fullName依然在vm身上
                // get() : 当有人读取fullName的时候, 就会调用get方法, 返回值就会被当做fullName的值
                get() {
                    return this.firstName + this.lastName;
                }
            }
        }
    })

         但是这个get方法中不能直接 return firstName + lastName,因为他访问不到data中的数据,但是可以通过vm实例对象来访问.

<body>
    <div id="root">
        姓 : <input type="text" v-model="firstName"><br>
        名 : <input type="text" v-model="lastName"> <br>
        姓名 : <span>{{fullName}}</span>
    </div>
</body>
<script>
    new Vue({
        el: '#root',
        data: {
            firstName: '张',
            lastName: '三'
        },
        computed : {
            fullName : { // 这个fullName依然在vm身上
                // get() : 当有人读取fullName的时候, 就会调用get方法, 返回值就会被当做fullName的值
                get() {
                    return this.firstName + this.lastName;
                }
            }
        }
    })
</script>

思考> computed中的属性的get 会被调用??

        解释: 初次读取computed中的属性的时候,对应的get方法会被调用, 由于调用之后, 其返回值被缓存, 下次读取就不会调用get方法, 其次就是所依赖的数据被修改的时候, 就上面例子而言, 如果输入框中的姓被修改的时候, get方法就会被重新调用.

        fullName虽然是个对象, 但是不能直接使用: {{fullName.get()}}. 

set()

        有get, 那么必然有set() :

computed : {
            fullName : { // 这个fullName依然在vm身上
                // get() : 当有人读取fullName的时候, 就会调用get方法, 返回值就会被当做fullName的值
                get() {
                    return this.firstName + this.lastName;
                },
                set (value) {
                    const arr = value.split('-');
                    this.firstName = arr[0];
                    this.lastName = arr[1];
                }
            }
        }

        我们讲到, 如果调用了set方法, 那么就会修改firstName和lastName属性, 那么vm就会捕获到这个修改, 从而重新更新使用到这两个属性的地方, 同时set方法也捕获到了这个修改, 然后将引用这个fullName的地方进行更新:

总结 

计算属性:
        1.定义:要用的属性不存在,要通过已有属性计算得来。

如果计算属性中存在在vm外部定义的变量, 那么确实是可以像已有的属性那样显示出来, 但是vm无法捕捉到这个外部定义的变量的变化.

        2.原理: 底层借助了objcet.defineproperty方法提供的getter和setter。
        3.get函数什么时候执行?
                (1).初次读取时会执行一次。
                (2).当依赖的数据发生改变时会被再次调用。
        4.优势:与methods实现相比,内部有缓存机制 (复用) ,效率更高,调试方便。
        5.备注:
                1.计算属性最终会出现在vm上,直接读取使用即可。
                2.如果计算属性要被修改,那必须写set函数去响应修改,且set中要引起计算时依赖的数据发生变化

计算属性简写

计算属性中的每一个属性都是当做对象处理:

        对象中的每个方法使用逗号分割, 但是实际上这个set方法一般都用不到, 所以直接将这个fullName定义为一个方法 :

    const vm = new  Vue({
        el: '#root',
        data: {
            firstName: '张',
            lastName: '三'
        },
        computed : {
            fullName : function(){ // 这个fullName依然在vm身上
                    return this.firstName + this.lastName;
            }
        }
    })

         更甚将其改为:

            fullName(){ // 这个fullName依然在vm身上
                    return this.firstName + this.lastName;
            }

        注意这里简写的前置条件是, 我不会去调用set方法, 如果调用set方法:

[vue警告]:计算属性, “FulName”分配给了它,但它没有set方法. 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于AM62x的ARM+FPGA+Codesys低成本软PLC解决方案

GPMC并口简介 GPMC(General Purpose Memory Controller)是TI处理器特有的通用存储器控制器接口&#xff0c;支持8/16bit数据位宽&#xff0c;支持128MB访问空间&#xff0c;最高时钟速率133MHz。GPMC是AM62x、AM64x、AM437x、AM335x、AM57x等处理器专用于与外部存储器设备的接口…

IP地理位置定位技术基本原理

IP地理位置定位技术的基本原理是基于IP地址的特性。每个IP地址在网络中都有一个与之对应的地理位置信息&#xff0c;这是通过IP地址数据库来确定的。这个数据库由ISP&#xff08;Internet Service Provider&#xff09;或其它一些机构维护&#xff0c;其中包含了每个IP地址的地…

openGauss学习笔记-179 openGauss 数据库运维-逻辑复制-发布订阅

文章目录 openGauss学习笔记-179 openGauss 数据库运维-逻辑复制-发布订阅179.1 发布179.2 订阅179.3 冲突处理179.4 限制179.5 架构179.6 监控179.7 安全性179.8 配置设置179.9 快速设置 openGauss学习笔记-179 openGauss 数据库运维-逻辑复制-发布订阅 发布和订阅基于逻辑复…

Java核心知识点1-java和c++区别、隐式和显示类型转换

java和c区别 java通过虚拟机实现跨平台特性&#xff0c;但c依赖于特定的平台。java没有指针&#xff0c;它的引用可以理解为安全指针&#xff0c;而c和c一样具有指针。java支持自动垃圾回收&#xff0c;而c需要手动回收。java不支持多重继承&#xff0c;只能通过实现多个接口来…

Java集合/泛型篇----第一篇

系列文章目录 文章目录 系列文章目录前言一、ArrayList和linkedList的区别二、HashMap和HashTable的区别三、Collection包结构,与Collections的区别四、泛型常用特点前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站…

swing快速入门(三十二)消息对话框

注释很详细&#xff0c;直接上代码 上一篇 新增内容 1.自定义对话框前列图标 2.消息对话框的若干种形式 package swing21_30;import javax.swing.*; import java.awt.*; import java.awt.event.ActionEvent;public class swing_test_30 {// 定义一个JFrameJFrame jFrame n…

自动驾驶学习笔记(二十三)——车辆控制模型

#Apollo开发者# 学习课程的传送门如下&#xff0c;当您也准备学习自动驾驶时&#xff0c;可以和我一同前往&#xff1a; 《自动驾驶新人之旅》免费课程—> 传送门 《Apollo开放平台9.0专项技术公开课》免费报名—>传送门 文章目录 前言 运动学模型 动力学模型 总结…

2023年03月20日_对李开复3月20日线下媒体会的解读

最近这个AI大模型 因为GPT4.0 ChatGPT 文心一言等等这些事情呢 一下子就被推到了风口浪尖 我们也做了来介绍相关的进展 国内呢也不断有一些大佬开始下场 包括王慧文、张朝阳、李彦宏什么的 都开始说自己要搞AI大模型 就在昨天呢 创新工厂的董事长兼CEO李开复 也发朋友…

oracle-存储结构

文件包括 控制文件.ctl、数据文件.dbf、日志文件.log这三类放在存储上。 参数文件&#xff1a;空间的划分&#xff0c;进程的选用&#xff08;.ora&#xff09; oracle启动的时候需要读一下&#xff0c;数据库启动后&#xff0c;参数文件并不关闭&#xff0c;但即使文件丢了&a…

IntelliJ IDEA常用快捷键

【1】创建内容&#xff08;新建&#xff09;&#xff1a;altinsert 【2】main方法&#xff1a;psvm 【3】输出语句&#xff1a;sout 【4】复制行&#xff1a;ctrld 【5】删除行&#xff1a;ctrly&#xff08;很多编辑器ctrly是前进操作&#xff0c;如果选择 Delete Line&…

APP端网络测试与弱网模拟!

当前APP网络环境比较复杂&#xff0c;网络制式有2G、3G、4G网络&#xff0c;还有越来越多的公共Wi-Fi。不同的网络环境和网络制式的差异&#xff0c;都会对用户使用app造成一定影响。另外&#xff0c;当前app使用场景多变&#xff0c;如进地铁、上公交、进电梯等&#xff0c;使…

【Maven】报错合集

问题1&#xff1a;com.github.everit-org.json-schema:org.everit.json.schema:pom:1.12.1 failed to transfer from http://localhost:8081/repository/maven-public/ during a previous attempt 发现原来是maven的settings.xml文件配置出现了问题。首先是之前maven进阶学习时…

如何解决msvcp110.dll丢失问题,多种解决方法分享

在计算机使用过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是“msvcp110.dll丢失”&#xff0c;msvcp110.dll是Microsoft Visual C 2012 Redistributable的一个组件&#xff0c;它提供了许多用于运行程序的函数和类库。当这个文件丢失或损坏时&#xff…

大一c语言期末复习题

第11题&#xff1a; 答案&#xff1a; #include<stdio.h> #include<string.h> int main() {char a[100];char b[100];int n 0, i 0, j 0;fgets(a, sizeof(a), stdin); //输入一串字符存给asize_t len1 strlen(a); //一开始该字符串的长度if (len1 &g…

【Android12】Android Framework系列---tombstone墓碑生成机制

tombstone墓碑生成机制 Android中程序在运行时会遇到各种各样的问题&#xff0c;相应的就会产生各种异常信号&#xff0c;比如常见的异常信号 Singal 11&#xff1a;Segmentation fault表示无效的地址进行了操作&#xff0c;比如内存越界、空指针调用等。 Android中在进程(主要…

华为hcia之ipv6实验手册

R3: dhcp enable ipv6 dhcpv6 pool test address prefix 2000:23::/64 excluded-address 2000:23::2 dns-server 2000:23::2 interface GigabitEthernet0/0/0 ipv6 enable ipv6 address 2000:12::2/64 ipv6 address auto link-local undo ipv6 nd ra halt //无状态配置 inter…

基于Java网上点餐系统设计与实现

博主介绍&#xff1a; ✌至今服务客户已经1000、专注于Java技术领域、项目定制、技术答疑、开发工具、毕业项目实战 ✌ &#x1f345; 文末获取源码联系 &#x1f345; &#x1f447;&#x1f3fb; 精彩专栏 推荐订阅 &#x1f447;&#x1f3fb; 不然下次找不到 Java项目精品实…

Jmeter吞吐量控制器总结

吞吐量控制器(Throughput Controller) 场景&#xff1a; 在同一个线程组里, 有10个并发, 7个做A业务, 3个做B业务,要模拟这种场景,可以通过吞吐量模拟器来实现。 添加吞吐量控制器 用法1: Percent Executions 在一个线程组内分别建立两个吞吐量控制器, 分别放业务A和业务B …

行车记录仪变清晰,变高清的办法一定要收藏

有时候我们会发现行车记录仪拍摄的视频不够清晰&#xff0c;特别是出现事故需要视频为证的时候&#xff0c;如果视频太模糊&#xff0c;很难获得交警的支持&#xff0c;那么如何让行车记录仪拍摄的视频变得更加清晰呢&#xff1f; 小编给大家分享几个办法&#xff0c;建议收藏…

山西电力市场日前价格预测【2023-12-31】

日前价格预测 预测说明&#xff1a; 如上图所示&#xff0c;预测明日&#xff08;2023-12-31&#xff09;山西电力市场全天平均日前电价为445.23元/MWh。其中&#xff0c;最高日前电价为791.27元/MWh&#xff0c;预计出现在08:15。最低日前电价为270.52元/MWh&#xff0c;预计…