02. Directive
November 11, 2023About 8 min
Mustache
<div id="app">
<h2 v-once>{{message}}</h2>
</div>
<script src="https://unpkg.com/vue@3/dist/vue.global.js"></script>
<script>
const app = Vue.createApp({
data() {
return {
message: "Hello World!"
}
}
})
app.mount('#app')
</script>
Most Used Directives
v-once
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>vue study - v-once</title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!-- 数据绑定后,UI不会随着数据的变化而发生变化 -->
<h2 v-once>{{message}}</h2>
</div>
<script>
const app = new Vue({
el:"#app",
data:{
message:"hello world"
}
})
</script>
</body>
</html>
v-for
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<ul> <li v-for="item in names">{{item}}</li> </ul>
<ul> <li v-for="(item, index) in names">{{index+1}} {{item}}</li> </ul>
<ul> <li v-for="(value, key) in info">{{key}}:{{value}}</li></ul>
<ul> <li v-for="(value, key, index) in info">{{key}}:{{value}}:{{index}}</li> </ul>
<ul> <li v-for="item in items" :key="item">{{item}}</li> </ul>
<button @click="btnClick">btn</button>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
names: ['item1', 'item2', 'item3', 'item4'],
info: {
name: 'why',
age: 18,
height: 2.0
},
items: ['a', 'b', 'c', 'd'],
},
methods:{
btnClick(){
//1.push Yes
//this.items.push('f')
//2.modify data by index No.
//this.items[0] = 'aaaaa'
//yes for listening
//pop() //删除最后
//shift() //删除第一
//unshift()//最前面添加多个元素
//splice() //可删,可加,可替换
//sort()
//reverse()
//this.items.reverse()
//this.items.sort()
}
}
})
</script>
</body>
</html>
v-text
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>vue study - v-text</title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!-- 设置文本,若标签中存在文本,则会被message中内容覆盖 -->
<h2 v-text="message"></h2>
</div>
<script>
const app = new Vue({
el:"#app",
data:{
message:"hello world"
}
})
</script>
</body>
</html>
v-html
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>vue study - v-html</title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!-- 将绑定值以HTML代码处理 -->
<h2 v-html="url"></h2>
</div>
<script>
const app = new Vue({
el:"#app",
data:{
url: '<a href="https://www.baidu.com">Baidu</a>'
}
})
</script>
</body>
</html>
v-pre
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>vue study - list</title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!--作用:不进行解析,直接显示{{message}}-->
<h2 v-pre>{{message}}</h2>
</div>
<script>
const app = new Vue({
el:"#app",
data:{
message:"hello world"
}
})
</script>
</body>
</html>
v-cloak
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>vue study - list</title>
<script src="../js/vue.js"></script>
<style>
[v-cloak] {
display: none;
}
</style>
</head>
<body>
<!--
在vue解析前,div中有一个属性v-cloak,解析后删除该属性,因
此可通过css一起配合避免DOM闪动的情况(一般由于JS执行卡顿导致) -->
<div id="app" v-cloak>
<h2 v-once>{{message}}</h2>
<h2 v-pre>{{message}}</h2>
<h2>{{url}}</h2>
<h2 v-html="url"></h2>
<h2>{{firstName + ' ' + lastName}}</h2>
<h2>{{firstName}} {{lastName}}</h2>
<h2 v-once>{{counter*2}}</h2>
<ul>
<li v-for="item in movies">{{item}}</li>
</ul>
</div>
<script>
setTimeout(function(){
const app = new Vue({
el:"#app",
data:{
message:"hello world",
firstName: "xxxx",
lastName: "sssss",
counter : 1,
url: '<a href="https://www.baidu.com">Baidu</a>',
movies:["海王", "新机穿越","大话西游","盗梦空间"],
}
})},
1000)
</script>
</body>
</html>
v-if
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<h2 v-if="score>=90"> 优秀</h2>
<h2 v-else-if="score>=80">良好</h2>
<h2 v-else>合格</h2>
</div>
<script>
const app = new Vue({
el: '#app',
data: {
score: 70
}
})
</script>
</body>
</html>
v-show
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!-- v-show 若不显示时,知识在DOM中增加行内样式display: none -->
<span v-show="isUser">
<label for="username">用户账户</label>
<input id="username" placeholder="用户账户" key="username"></input>
</span>
<span v-show="isEmail">
<label for="email">用户邮箱</label>
<input id="email" placeholder="用户邮箱" key="email"></input>
</span>
<button @click="isUser = !isUser">切换登录</button>
</div>
<script>
const app = new Vue({
el: '#app',
data: {
isUser: true,
isEmail: true
}
})
</script>
</body>
</html>
v-bind
指令全写为v-bind:{被绑属性}=“被绑值” 也可简写为 :{被绑属性}=“被绑值”,主要作用对属性实现绑定。
Basic
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<script src="../js/vue.js"></script>
<title></title>
<style>
.active {
color: red;
}
</style>
</head>
<body>
<div id="app">
<img v-bind:src="imgURL"></img>
<img :src="imgURL"></img>
<h2 :class="{active: isActive, line: isLine}">{{message}}</h2>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "Hello world",
imgURL : "",
isActive: true,
isLine: false
}
})
</script>
</body>
</html>
Bind Object
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<script src="../js/vue.js"></script>
<title></title>
<style>
.active {
color: red;
}
</style>
</head>
<body>
<div id="app">
<h2 class="title" :class="{active: isActive, line: isLine}">{{message}}</h2>
<h2 class="title" :class="getClasses()">{{message}}</h2>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "Hello world",
isActive: true,
isLine: false
},
methods: {
btnClick: function(){
this.isActive = !this.isActive;
},
getClasses: function(){
return {active: this.isActive, line: this.isLine};
}
}
})
</script>
</body>
</html>
Bind Array
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<script src="../js/vue.js"></script>
<title></title>
<style>
.active {
color: red;
}
</style>
</head>
<body>
<div id="app">
<h2 class="title" :class="[active, line]">{{message}}</h2>
<button v-on:click="btnClick">点击</button>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "hello world"
isActive: true,
isLine: false,
active: "active",
line: "line"
},
methods: {
btnClick: function(){
this.isActive = !this.isActive;
}
}
})
</script>
</body>
</html>
Bind Style
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<script src="../js/vue.js"></script>
<title></title>
<style>
.active {
color: red;
}
</style>
</head>
<body>
<div id="app">
<h2 :style="{fontSize: '50px'}">{{message}}</h2>
<h2 :style="{fontSize: fsize}">{{message}}</h2>
<h2 :style="{fontSize: ifsize + 'px'}">{{message}}</h2>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "Hello world",
fsize: "50px",
ifsize: 100
},
methods: {
btnClick: function(){
this.isActive = !this.isActive;
},
getClasses: function(){
return {active: this.isActive, line: this.isLine};
}
}
})
</script>
</body>
</html>
Bind Style Array
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<script src="../js/vue.js"></script>
<title></title>
<style>
.active {
color: red;
}
</style>
</head>
<body>
<div id="app">
<h2 :style="[baseStyle, baseStyle1]">{{message}}</h2>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "Hello world",
baseStyle: {backgroundColor: 'red'},
baseStyle1: {fontSize: '50px'}
},
methods: {
btnClick: function(){
this.isActive = !this.isActive;
},
getClasses: function(){
return {active: this.isActive, line: this.isLine};
}
}
})
</script>
</body>
</html>
v-on
事件绑定指令,绑定语法为 v-on:事件=”被绑定方法”,或者简写为 @事件=”被绑定方法”
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<button v-on:click="btn1Click()"> click1</button>
<button @click="btn1Click"> click2</button>
<button @click="btn2Click()">click3</button>
<!--when the param is omitted, the param will be assigned with event object-->
<button @click="btn2Click"> click4</button>
<button @click="btn3Click(123, $event)"> click5</button>
<button @click="btn1Click()"> click6</button>
</div>
<script>
const app = new Vue({
el: "#app",
data: {},
methods: {
btn1Click(){
console.log('btn1Click')
},
btn2Click(param){
console.log(param)
},
btn3Click(abc, event){
console.log(abc)
console.log(event)
}
}
})
</script>
</body>
</html>
Event Handling
Most Used List
.stop
:阻止单击事件继续传播.prevent
:提交事件不再重载页面.capture
: 添加事件监听器时使用事件捕获模式,即内部元素触发的事件先在此处理,然后才交由内部元素进行处理。.self
: 只当在 event.target 是当前元素自身时触发处理函数,即事件不是从内部元素触发的。.once
:点击事件将只会触发一次
<!-- 样例代码 -->
<!-- 阻止单击事件继续传播 -->
<a v-on:click.stop="doThis"></a>
<!-- 提交事件不再重载页面 -->
<form v-on:submit.prevent="onSubmit"></form>
<!-- 修饰符可以串联 -->
<a v-on:click.stop.prevent="doThat"></a>
<!-- 只有修饰符 -->
<form v-on:submit.prevent></form>
<!-- 添加事件监听器时使用事件捕获模式 -->
<!-- 即内部元素触发的事件先在此处理,然后才交由内部元素进行处理 -->
<div v-on:click.capture="doThis">...</div>
<!-- 只当在 event.target 是当前元素自身时触发处理函数 -->
<!-- 即事件不是从内部元素触发的 -->
<div v-on:click.self="doThat">...</div>
model
主要用于表单数据的绑定,实现表单元素和数据的双向绑定。双向绑定的本质就是对UI进行输入监测,然后反向修改被绑定数据。
Basic
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<input type="text" v-model="message" >
<input type="text" :value="message" @input="valueChange">
<input type="text" :value="message" @input="message=$event.target.value">
{{message}}
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "hello world."
},
methods:{
valueChange(event){
this.message = event.target.value;
}
}
})
</script>
</body>
</html>
Radio
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<input type="radio" id="male2" value="男" name="11" v-model="sex">男</input>
<input type="radio" id="female2" value="女" name="11" v-model="sex">女</input>
</br>
<label for="male">
<input type="radio" id="male" value="男" v-model="gender">男</input>
</label>
<label for="female">
<input type="radio" id="female" value="女" v-model="gender">女</input>
</label>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
sex: "男",
gender: "男"
},
methods:{
valueChange(event){
this.message = event.target.value;
}
}
})
</script>
</body>
</html>
Checkbox
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!-- 01.单选框 -->
<label for="is">
<input type="checkbox" id="is" v-model="agree">同意</input>
</label>
</br>
<!-- 02. 多选框 -->
<label for="">
<input type="checkbox" value="足球" v-model="hobbies">足球</input>
<input type="checkbox" value="篮球" v-model="hobbies">篮球</input>
<input type="checkbox" value="乒乓球" v-model="hobbies">乒乓球</input>
<input type="checkbox" value="橄榄球" v-model="hobbies">橄榄球</input>
</label>
<h2>您的爱好:{{hobbies}}</h2>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
agree: false,
hobbies: []
},
methods:{
}
})
</script>
</body>
</html>
Dropbox
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<select name="abc" id="" v-model="fruit">
<option value="1">苹果</option>
<option value="2">香蕉</option>
<option value="3">草莓</option>
<option value="4">蓝莓</option>
<option value="5">榴莲</option>
</select>
<h2>您的选择为:{{fruit}}</h2>
<select name="abc" id="" v-model="fruits" multiple>
<option value="苹果">苹果</option>
<option value="香蕉">香蕉</option>
<option value="草莓">草莓</option>
<option value="蓝莓">蓝莓</option>
<option value="榴莲">榴莲</option>
</select>
<h2>您的选择为:{{fruits}}</h2>
<select name="abc" id="" v-model="fruits" multiple >
<option v-for="item in originalFruits" :value="item">{{item}}</option>
</select>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
agree: false,
fruit: [],
fruits: [],
originalFruits:["苹果","香蕉","草莓","蓝莓","榴莲"] ,
},
methods:{
}
})
</script>
</body>
</html>
v-model modifier
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title></title>
<script src="../js/vue.js"></script>
</head>
<body>
<div id="app">
<!-- 01. lazy -->
<input type="text" v-model.lazy="message">
{{message}}
</br>
<!-- 02. number -->
<input type="text" v-model.number="age">
{{age}}:{{typeof age}}
</br>
<!-- 03.trim -->
<input type="text" v-model.trim="name">
|{{name}}|
</br>
</div>
<script>
const app = new Vue({
el: "#app",
data: {
message: "Hello world",
age: 15,
name: "james"
},
methods:{
}
})
</script>
</body>
</html>