在前端开发中,我们需要经常和用于交互。
这个时候,我们就必须监听用户发生的时间,比如点击、拖拽、键盘事件等等
在Vue中如何监听事件呢?使用v-on指令
v-on介绍
作用:绑定事件监听器
缩写:@
预期:Function | Inline Statement | Object
参数:event
下面,我们就具体来学习v-on的使用。
这里,我们用一个监听按钮的点击事件,来简单看看v-on的使用
下面的代码中,
我们使用了v-on:click="counter++”
另外,我们可以将事件指向一个在methods中定义的函数
注:v-on也有对应的语法糖:
v-on:click可以写成@click
当通过methods中定义方法,以供@click调用时,需要注意参数问题:
情况一:如果该方法不需要额外参数,那么方法后的()可以不添加。
但是注意:如果方法本身中有一个参数,那么会默认将原生事件event参数传递进去
在某些情况下,我们拿到event的目的可能是进行一些事件处理。
Vue提供了修饰符来帮助我们方便的处理一些事件:
v-if、v-else-if、v-else
条件渲染案例:
我们来做一个简单的小案例:
用户再登录时,可以切换使用用户账号登录还是邮箱地址登录。
类似如下情景:
案例中存在的小问题:
小问题:
如果我们在有输入内容的情况下,切换了类型,我们会发现文字依然显示之前的输入的内容。
但是按道理讲,我们应该切换到另外一个input元素中了。
在另一个input元素中,我们并没有输入内容。
为什么会出现这个问题呢?
问题解答:
这是因为Vue在进行DOM渲染时,出于性能考虑,会尽可能的复用已经存在的元素,而不是重新创建新的元素。
在上面的案例中,Vue内部会发现原来的input元素不再使用,直接作为else中的input来使用了。
解决方案:
v-show的用法和v-if非常相似,也用于决定一个元素是否渲染
v-if和v-show对比
v-if和v-show都可以决定一个元素是否渲染,那么开发中我们如何选择呢?
开发中如何选择呢?
当我们有一组数据需要进行渲染时,我们就可以使用v-for来完成。
v-for的语法类似于JavaScript中的for循环。
格式如下:item in items
的形式。
我们来看一个简单的案例:
如果在遍历的过程中不需要使用索引值
v-for="item in names"
依次从names中取出item,并且在元素的内容中,我们可以使用Mustache语法,来使用item
如果在遍历的过程中,我们需要拿到元素在数组中的索引值呢?
语法格式:v-for=(item, index) in items
其中的index就代表了取出的item在原数组的索引值。
v-for可以用户遍历对象:
比如某个对象中存储着你的个人信息,我们希望以列表的形式显示出来。
因为Vue是响应式的,所以当数据发生变化时,Vue会自动检测数据变化,视图会发生对应的更新。
Vue中包含了一组观察数组编译的方法,使用它们改变数组也会触发视图的更新。
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Titletitle>
head>
<body>
<div id="app">
<ul>
<li v-for="item in letters">{{item}}li>
ul>
<button @click="btnClick">按钮button>
div>
<script src="../js/vue.js">script>
<script>
const app = new Vue({
el: '#app',
data: {
letters: ['A','B','C','D','E']
},
methods:{
btnClick() {
// 0注意:通过索引值修改数组中的元素(无法做到响应式,数组的内容改变时页面内容不会响应式变化)
// this.letters[0] = 'bbbb'
/*以下几种改变数组的方法是响应式的*/
// 1.push方法
// this.letters.push('AAA');
// this.letters.push('AAA','bbbb');
// 2.pop():删除数组中的最后一个元素
// this.letters.pop();
// 3.shift():删除数组中的第一个元素
// this.letters.shift();
// 4.unshift():在数组最前面添加元素
// this.letters.unshift('aaaa');
// this.letters.unshift('aaaa','BBBB');
// 5.
// splice的作用:删除元素/插入元素/替换元素
// 删除元素:第二个参数传入 你要删除几个元素(如果没有传参数,就删除后面所有的元素)
// 替换元素:第二个参数,表示我们要替换几个元素,后面的参数是用来替换前面的元素
// 插入元素:第二个参数,传入0,并且后面跟上要插入的元素
// splice(start):
// this.letters.splice(2,2,'ab','cz');
// this.letters.splice(2,0,'X','Y')
// 6.sort()
// this.letters.sort()
// 7.reverse()
// this.letters.reverse()
// set(要修改的对象,索引值,修改后的值)
Vue.set(this.letters,0,'bbbbbbbb')
}
}
})
script>
body>
html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Titletitle>
<link rel="stylesheet" href="style.css">
head>
<body>
<div id="app">
<div v-if="books.length">
<table>
<thead>
<tr>
<th>th>
<th>书籍名称th>
<th>出版日期th>
<th>价格th>
<th>购买数量th>
<th>操作th>
tr>
thead>
<tbody>
<tr v-for="(item,index) in books">
<td>{{item.id}}td>
<td>{{item.name}}td>
<td>{{item.date}}td>
<td>{{item.price | showPrice}}td>
<td>
<button @click="decrement(index)" v-bind:disabled="item.count <= 1">-button>
{{item.count}}
<button @click="increment(index)">+button>
td>
<td>
<button @click="removeHandle(index)">移除button>
td>
tr>
tbody>
table>
<h2>总价格:{{totalPrice | showPrice}}h2>
div>
<h2 v-else>购物车为空!h2>
div>
<script src="../js/vue.js">script>
<script src="main.js">script>
body>
html>
table {
border: 1px solid #e9e9e9;
border-collapse: collapse;
border-spacing: 0;
}
th, td {
padding: 8px 16px;
border: 1px solid #e9e9e9;
text-align: center;
}
th {
background-color: #f7f7f7;
color: #5c6b77;
font-weight: 600;
}
const app = new Vue({
el:'#app',
data:{
books: [
{
id: 1,
name: '《Python入门》',
date: '2020-10',
price: 85.00,
count: 36
},
{
id: 2,
name: '《Vue从入门到精通》',
date: '2020-10',
price: 99.00,
count: 22
},
{
id: 3,
name: '《算法导论》',
date: '2020-08',
price: 32.00,
count: 10
},
{
id: 4,
name: '《软件工程》',
date: '2020-02',
price: 105.00,
count: 50
}
]
},
methods: {
getFinalPrice(price) {
return '¥' + price.toFixed(2);
},
increment(index) {
console.log('increment',index);
this.books[index].count++;
},
decrement(index) {
console.log('decrement',index);
this.books[index].count--;
},
removeHandle(index) {
this.books.splice(index,1);
}
},
filters: {
showPrice(price) {
return '¥' + price.toFixed(2);
}
},
computed: {
totalPrice() {
/*// 1. 普通的for循环
let totalPrice = 0;
for (let i = 0; i < this.books.length; i++) {
totalPrice += this.books[i].price * this.books[i].count;
}
return totalPrice;*/
/*// 2. for(let i in this..books)
let totalPrice = 0;
for (let i in this.books) {
totalPrice += this.books[i].price * this.books[i].count;
}
return totalPrice;*/
/*// 3. for(let item of this.books)
let totalPrice = 0;
for (let item of this.books) {
totalPrice += item.price * item.count;
}
return totalPrice;*/
// 4.reduce
return this.books.reduce(function (preValue,book) {
return preValue + book.price * book.count
},0)
}
}
})
// 编程范式:命令式编程/声明式编程
// 编程范式:面向对象编程(第一公民:对象)/函数式编程(第一公民:函数)
// filter /map /reduce
// filter 中的回调函数有一个要求:必须返回一个boolean值
// true :当返回true时,函数内部会自动将这次回调的n加入到心得数组中
// false:当返回false时,函数内部会过滤掉这次的n
const nums = [10,20,111,222,444,40,50]
/*let total = nums.filter(function (n) {
return n < 100;
}).map(function (n) {
return n * 2;
}).reduce(function (preValue,n) {
return preValue + n;
},0)
console.log(total);*/
let total = nums.filter(n => n < 100)
.map(n => n * 2)
.reduce((preValue,n) => preValue + n)
console.log(total);
/*
// 1. filter函数的使用
// 10, 20, 40, 50
let newNums = nums.filter(function (n) {
return n < 100;
})
console.log(newNums);
// 2. map函数的使用
// 20,40,80,100
let new2Nums = newNums.map(function (n) {
return n*2;
})
console.log(new2Nums);
// 3. reduce函数的使用
// educe函数的作用:对数组中所有的内容进行汇总
/!*
// new2Nums.reduce(function (preValue,n) {
// return 100;
// },0)
// 第一次:preValue=0,n=20
// 第二次:preValue=100,n=40
// 第三次:preValue=100,n=80
// 第四次:preValue=100,n=100
*!/
let total = new2Nums.reduce(function (preValue,n) {
return preValue + n;
},0)
// 第一次:preValue=0,n=20
// 第二次:preValue=20,n=40
// 第三次:preValue=60,n=80
// 第四次:preValue=140,n=100
// 第五次:preValue + n=240
console.log(total);
*/
表单控件在实际开发中是非常常见的。
特别是对于用户信息的提交,需要大量的表单。
Vue中使用v-model
指令来实现表单元素和数据的双向绑定。
案例的解析:
v-model其实是一个语法糖,它的背后本质上是包含两个操作:
<input type="text" v-model="message">
等同于
<input type="text" v-bind:value="message"
v-on:input="message = $event.target.value">
复选框分为两种情况:单个勾选框和多个勾选框
单个勾选框:
v-model即为布尔值。
此时input的value并不影响v-model的值。
多个复选框:
当是多个复选框时,因为可以选中多个,所以对应的data中属性是一个数组。
当选中某一个时,就会将input的value添加到数组中。
和checkbox一样,select也分单选和多选两种情况。
单选:只能选中一个值。
初看Vue官方值绑定的时候,我很疑惑:what the hell is that?
但是仔细阅读之后,发现很简单,就是动态的给value赋值而已:
我们前面的value中的值,可以回头去看一下,都是在定义input的时候直接给定的。
但是真实开发中,这些input的值可能是从网络获取或定义在data中的。
所以我们可以通过v-bind:value动态的给value绑定值。
这不就是v-bind吗?
这不就是v-bind在input中的应用吗?搞的我看了很久,搞不清他想讲什么。
这里不再给出对应的代码,因为会用v-bind,就会值绑定的应用了。
默认情况下,v-model默认是在input事件中同步输入框的数据的。 也就是说,一旦有数据发生改变对应的data中的数据就会自动发生改变。
lazy修饰符可以让数据在失去焦点或者回车时才会更新
默认情况下,在输入框中无论我们输入的是字母还是数字,都会被当做字符串类型进行处理。
但是如果我们希望处理的是数字类型,那么最好直接将内容当做数字处理。
number修饰符可以让在输入框中输入的内容自动转成数字类型
如果输入的内容首尾有很多空格,通常我们希望将其去除
trim修饰符可以过滤内容左右两边的空格