事件处理
监听事件
可以用 v-on
指令监听 DOM 事件来触发一些 JavaScript 代码。
示例:
<div id="example-1"> <button v-on:click="counter += 1">增加 1</button> <p>这个按钮被点击了 {{ counter }} 次。</p> </div> |
var example1 = new Vue({ el: ‘#example-1‘, data: { counter: 0 } }) |
结果:
增加 1
这个按钮被点击了 0 次。
方法事件处理器
许多事件处理的逻辑都很复杂,所以直接把 JavaScript 代码写在 v-on
指令中是不可行的。因此 v-on
可以接收一个定义的方法来调用。
示例:
<div id="example-2"> <!-- `greet` 是在下面定义的方法名 --> <button v-on:click="greet">Greet</button> </div> |
var example2 = new Vue({ el: ‘#example-2‘, data: { name: ‘Vue.js‘ }, // 在 `methods` 对象中定义方法 methods: { greet: function (event) { // `this` 在方法里指当前 Vue 实例 alert(‘Hello ‘ + this.name + ‘!‘) // `event` 是原生 DOM 事件 if (event) { alert(event.target.tagName) } } } }) // 也可以用 JavaScript 直接调用方法 example2.greet() // => ‘Hello Vue.js!‘ |
结果:
Greet
内联处理器里的方法
除了直接绑定到一个方法,也可以用内联 JavaScript 语句:
<div id="example-3"> <button v-on:click="say(‘hi‘)">Say hi</button> <button v-on:click="say(‘what‘)">Say what</button> </div> |
new Vue({ el: ‘#example-3‘, methods: { say: function (message) { alert(message) } } }) |
结果:
Say hi Say what
有时也需要在内联语句处理器中访问原生 DOM 事件。可以用特殊变量 $event
把它传入方法:
<button v-on:click="warn(‘Form cannot be submitted yet.‘, $event)"> Submit </button> |
// ... methods: { warn: function (message, event) { // 现在我们可以访问原生事件对象 if (event) event.preventDefault() alert(message) } } |
事件修饰符
在事件处理程序中调用 event.preventDefault()
或 event.stopPropagation()
是非常常见的需求。尽管我们可以在 methods 中轻松实现这点,但更好的方式是:methods 只有纯粹的数据逻辑,而不是去处理 DOM 事件细节。
为了解决这个问题,Vue.js 为 v-on
提供了事件修饰符。通过由点 (.) 表示的指令后缀来调用修饰符。
.stop
.prevent
.capture
.self
.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> <!-- 只当事件在该元素本身 (比如不是子元素) 触发时触发回调 --> <div v-on:click.self="doThat">...</div> |
使用修饰符时,顺序很重要;相应的代码会以同样的顺序产生。因此,用@click.prevent.self
会阻止所有的点击,而 @click.self.prevent
只会阻止元素上的点击。
2.1.4 新增
<!-- 点击事件将只会触发一次 --> <a v-on:click.once="doThis"></a> |
不像其它只能对原生的 DOM 事件起作用的修饰符,.once
修饰符还能被用到自定义的组件事件上。如果你还没有阅读关于组件的文档,现在大可不必担心。
键值修饰符
在监听键盘事件时,我们经常需要监测常见的键值。Vue 允许为 v-on
在监听键盘事件时添加关键修饰符:
<!-- 只有在 keyCode 是 13 时调用 vm.submit() --> <input v-on:keyup.13="submit"> |
记住所有的 keyCode 比较困难,所以 Vue 为最常用的按键提供了别名:
<!-- 同上 --> <input v-on:keyup.enter="submit"> <!-- 缩写语法 --> <input @keyup.enter="submit"> |
全部的按键别名:
.enter
.tab
.delete
(捕获“删除”和“退格”键).esc
.space
.up
.down
.left
.right
可以通过全局 config.keyCodes
对象自定义键值修饰符别名:
// 可以使用 v-on:keyup.f1 Vue.config.keyCodes.f1 = 112 |
自动匹配按键修饰符
2.5.0 新增
你也可以通过将它们转换到 kebab-case 来直接使用由 KeyboardEvent.key
暴露的任意有效按键名作为修饰符:
<input @keyup.page-down="onPageDown"> |
在上面的例子中,处理函数仅在 $event.key === ‘PageDown‘
时被调用。
极少数的键 (.esc
以及所有的方向键) 在 IE9 中有不同的 key
值, 如果你想支持 IE9,它们的内置别名应该是首选。
系统修饰键
2.1.0 新增
可以用如下修饰符开启鼠标或键盘事件监听,使在按键按下时发生响应。
.ctrl
.alt
.shift
.meta
注意:在 Mac 系统键盘上,meta 对应命令键 (?)。在 Windows 系统键盘 meta 对应 windows 徽标键 (?)。在 Sun 操作系统键盘上,meta 对应实心宝石键 (◆)。在其他特定键盘上,尤其在 MIT 和 Lisp 键盘及其后续,比如 Knight 键盘,space-cadet 键盘,meta 被标记为“META”。在 Symbolics 键盘上,meta 被标记为“META”或者“Meta”。
例如:
<!-- Alt + C --> <input @keyup.alt.67="clear"> <!-- Ctrl + Click --> <div @click.ctrl="doSomething">Do something</div> |
修饰键比正常的按键不同;修饰键和 keyup
事件一起用时,事件引发时必须按下正常的按键。换一种说法:如果要引发 keyup.ctrl
,必须按下 ctrl
时释放其他的按键;单单释放 ctrl
不会引发事件。
.exact
修饰符
2.5.0 新增
.exact
修饰符应与其他系统修饰符组合使用,以指示处理程序只在精确匹配该按键组合时触发。
<!-- 即使 Alt 或 Shift 被一同按下时也会触发 --> <button @click.ctrl="onClick">A</button> <!-- 只有在 Ctrl 被按下的时候触发 --> <button @click.ctrl.exact="onCtrlClick">A</button> |
鼠标按钮修饰符
2.1.0 新增
.left
.right
.middle
这些修饰符会限制处理程序监听特定的滑鼠按键。
为什么在 HTML 中监听事件?
你可能注意到这种事件监听的方式违背了关注点分离 (separation of concern) 传统理念。不必担心,因为所有的 Vue.js 事件处理方法和表达式都严格绑定在当前视图的 ViewModel 上,它不会导致任何维护上的困难。实际上,使用 v-on
有几个好处:
- 扫一眼 HTML 模板便能轻松定位在 JavaScript 代码里对应的方法。
- 因为你无须在 JavaScript 里手动绑定事件,你的 ViewModel 代码可以是非常纯粹的逻辑,和 DOM 完全解耦,更易于测试。
- 当一个 ViewModel 被销毁时,所有的事件处理器都会自动被删除。你无须担心如何自己清理它们。
表单输入绑定
基础用法
你可以用 v-model
指令在表单控件元素上创建双向数据绑定。它会根据控件类型自动选取正确的方法来更新元素。尽管有些神奇,但 v-model
本质上不过是语法糖,它负责监听用户的输入事件以更新数据,并特别处理一些极端的例子。
v-model
会忽略所有表单元素的 value
、checked
、selected
特性的初始值。因为它会选择 Vue 实例数据来作为具体的值。你应该通过 JavaScript 在组件的data
选项中声明初始值。
对于要求 IME (如中文、日语、韩语等) (IME 意为“输入法”)的语言,你会发现v-model
不会在 ime 输入中得到更新。如果你也想实现更新,请使用 input
事件。
文本
<input v-model="message" placeholder="edit me"> <p>Message is: {{ message }}</p> |
Message is:
多行文本
<span>Multiline message is:</span> <p style="white-space: pre-line;">{{ message }}</p> <br> <textarea v-model="message" placeholder="add multiple lines"></textarea> |
Multiline message is:
在文本区域插值 (<textarea></textarea>
) 并不会生效,应用 v-model
来代替。
复选框
单个勾选框,逻辑值:
<input type="checkbox" id="checkbox" v-model="checked"> <label for="checkbox">{{ checked }}</label> |
false
多个勾选框,绑定到同一个数组:
<div id=‘example-3‘> <input type="checkbox" id="jack" value="Jack" v-model="checkedNames"> <label for="jack">Jack</label> <input type="checkbox" id="john" value="John" v-model="checkedNames"> <label for="john">John</label> <input type="checkbox" id="mike" value="Mike" v-model="checkedNames"> <label for="mike">Mike</label> <br> <span>Checked names: {{ checkedNames }}</span> </div> |
new Vue({ el: ‘#example-3‘, data: { checkedNames: [] } }) |
Jack John Mike
Checked names: []
单选按钮
<div id="example-4"> <input type="radio" id="one" value="One" v-model="picked"> <label for="one">One</label> <br> <input type="radio" id="two" value="Two" v-model="picked"> <label for="two">Two</label> <br> <span>Picked: {{ picked }}</span> </div> |
new Vue({ el: ‘#example-4‘, data: { picked: ‘‘ } }) |
One
Two
Picked:
选择列表
单选列表:
<div id="example-5"> <select v-model="selected"> <option disabled value="">请选择</option> <option>A</option> <option>B</option> <option>C</option> </select> <span>Selected: {{ selected }}</span> </div> |
new Vue({ el: ‘...‘, data: { selected: ‘‘ } }) |
请选择
A
B
C
Selected:
如果 v-model
表达初始的值不匹配任何的选项,<select>
元素就会以”未选中”的状态渲染。在 iOS 中,这会使用户无法选择第一个选项,因为这样的情况下,iOS 不会引发 change 事件。因此,像以上提供 disabled 选项是建议的做法。
多选列表 (绑定到一个数组):
<div id="example-6"> <select v-model="selected" multiple style="width: 50px;"> <option>A</option> <option>B</option> <option>C</option> </select> <br> <span>Selected: {{ selected }}</span> </div> |
new Vue({ el: ‘#example-6‘, data: { selected: [] } }) |
A
B
C
Selected: []
动态选项,用 v-for
渲染:
<select v-model="selected"> <option v-for="option in options" v-bind:value="option.value"> {{ option.text }} </option> </select> <span>Selected: {{ selected }}</span> |
new Vue({ el: ‘...‘, data: { selected: ‘A‘, options: [ { text: ‘One‘, value: ‘A‘ }, { text: ‘Two‘, value: ‘B‘ }, { text: ‘Three‘, value: ‘C‘ } ] } }) |
One
Two
Three
Selected: A
值绑定
对于单选按钮,勾选框及选择列表选项,v-model
绑定的 value 通常是静态字符串 (对于勾选框是逻辑值):
<!-- 当选中时,`picked` 为字符串 "a" --> <input type="radio" v-model="picked" value="a"> <!-- `toggle` 为 true 或 false --> <input type="checkbox" v-model="toggle"> <!-- 当选中时,`selected` 为字符串 "abc" --> <select v-model="selected"> <option value="abc">ABC</option> </select> |
但是有时我们想绑定 value 到 Vue 实例的一个动态属性上,这时可以用 v-bind
实现,并且这个属性的值可以不是字符串。
复选框
<input type="checkbox" v-model="toggle" v-bind:true-value="a" v-bind:false-value="b" > |
// 当选中时 vm.toggle === vm.a // 当没有选中时 vm.toggle === vm.b |
单选按钮
<input type="radio" v-model="pick" v-bind:value="a"> |
// 当选中时 vm.pick === vm.a |
选择列表的选项
<select v-model="selected"> <!-- 内联对象字面量 --> <option v-bind:value="{ number: 123 }">123</option> </select> |
// 当选中时 typeof vm.selected // => ‘object‘ vm.selected.number // => 123 |
修饰符
.lazy
在默认情况下,v-model
在 input
事件中同步输入框的值与数据 (除了 上述 IME 部分),但你可以添加一个修饰符 lazy
,从而转变为在 change
事件中同步:
<!-- 在 "change" 而不是 "input" 事件中更新 --> <input v-model.lazy="msg" > |
.number
如果想自动将用户的输入值转为 Number 类型 (如果原值的转换结果为 NaN 则返回原值),可以添加一个修饰符 number
给 v-model
来处理输入值:
<input v-model.number="age" type="number"> |
这通常很有用,因为在 type="number"
时 HTML 中输入的值也总是会返回字符串类型。
.trim
如果要自动过滤用户输入的首尾空格,可以添加 trim
修饰符到 v-model
上过滤输入:
<input v-model.trim="msg"> |
v-model
与组件
如果你还不熟悉 Vue 的组件,跳过这里即可。
HTML 内建的 input 类型有时不能满足你的需求。还好,Vue 的组件系统允许你创建一个具有自定义行为可复用的 input 类型,这些 input 类型甚至可以和 v-model
一起使用!要了解更多,请参阅自定义 input 类型。
vue组件
什么是组件?
组件 (Component) 是 Vue.js 最强大的功能之一。组件可以扩展 HTML 元素,封装可重用的代码。在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功能。在有些情况下,组件也可以表现为用 is
特性进行了扩展的原生 HTML 元素。
使用组件
全局注册
我们已经知道,可以通过以下方式创建一个 Vue 实例:
new Vue({ el: ‘#some-element‘, // 选项 }) |
要注册一个全局组件,可以使用 Vue.component(tagName, options)
。例如:
Vue.component(‘my-component‘, { // 选项 }) |
请注意,对于自定义标签的命名 Vue.js 不强制遵循 W3C 规则 (小写,并且包含一个短杠),尽管这被认为是最佳实践。
组件在注册之后,便可以作为自定义元素 <my-component></my-component>
在一个实例的模板中使用。注意确保在初始化根实例之前注册组件:
<div id="example"> <my-component></my-component> </div> |
// 注册 Vue.component(‘my-component‘, { template: ‘<div>A custom component!</div>‘ }) // 创建根实例 new Vue({ el: ‘#example‘ }) |
渲染为:
<div id="example"> <div>A custom component!</div> </div> |
A custom component!
局部注册
你不必把每个组件都注册到全局。你可以通过某个 Vue 实例/组件的实例选项 components
注册仅在其作用域中可用的组件:
var Child = { template: ‘<div>A custom component!</div>‘ } new Vue({ // ... components: { // <my-component> 将只在父组件模板中可用 ‘my-component‘: Child } }) |
这种封装也适用于其它可注册的 Vue 功能,比如指令。
DOM 模板解析注意事项
当使用 DOM 作为模板时 (例如,使用 el
选项来把 Vue 实例挂载到一个已有内容的元素上),你会受到 HTML 本身的一些限制,因为 Vue 只有在浏览器解析、规范化模板之后才能获取其内容。尤其要注意,像 <ul>
、<ol>
、<table>
、<select>
这样的元素里允许包含的元素有限制,而另一些像 <option>
这样的元素只能出现在某些特定元素的内部。
在自定义组件中使用这些受限制的元素时会导致一些问题,例如:
<table> <my-row>...</my-row> </table> |
自定义组件 <my-row>
会被当作无效的内容,因此会导致错误的渲染结果。变通的方案是使用特殊的 is
特性:
<table> <tr is="my-row"></tr> </table> |
应当注意,如果使用来自以下来源之一的字符串模板,则没有这些限制:
<script type="text/x-template">
- JavaScript 内联模板字符串
.vue
组件
因此,请尽可能使用字符串模板。
data
必须是函数
构造 Vue 实例时传入的各种选项大多数都可以在组件里使用。只有一个例外:data
必须是函数。实际上,如果你这么做:
Vue.component(‘my-component‘, { template: ‘<span>{{ message }}</span>‘, data: { message: ‘hello‘ } }) |
那么 Vue 会停止运行,并在控制台发出警告,告诉你在组件实例中 data
必须是一个函数。但理解这种规则为何存在也是很有益处的,所以让我们先作个弊:
<div id="example-2"> <simple-counter></simple-counter> <simple-counter></simple-counter> <simple-counter></simple-counter> </div> |
var data = { counter: 0 } Vue.component(‘simple-counter‘, { template: ‘<button v-on:click="counter += 1">{{ counter }}</button>‘, // 技术上 data 的确是一个函数了,因此 Vue 不会警告, // 但是我们却给每个组件实例返回了同一个对象的引用 data: function () { return data } }) new Vue({ el: ‘#example-2‘ }) |
0 0 0
由于这三个组件实例共享了同一个 data
对象,因此递增一个 counter 会影响所有组件!这就错了。我们可以通过为每个组件返回全新的数据对象来修复这个问题:
data: function () { return { counter: 0 } } |
现在每个 counter 都有它自己内部的状态了:
0 0 0
组件组合
组件设计初衷就是要配合使用的,最常见的就是形成父子组件的关系:组件 A 在它的模板中使用了组件 B。它们之间必然需要相互通信:父组件可能要给子组件下发数据,子组件则可能要将它内部发生的事情告知父组件。然而,通过一个良好定义的接口来尽可能将父子组件解耦也是很重要的。这保证了每个组件的代码可以在相对隔离的环境中书写和理解,从而提高了其可维护性和复用性。
在 Vue 中,父子组件的关系可以总结为 prop 向下传递,事件向上传递。父组件通过 prop给子组件下发数据,子组件通过事件给父组件发送消息。看看它们是怎么工作的。
Prop
使用 Prop 传递数据
组件实例的作用域是孤立的。这意味着不能 (也不应该) 在子组件的模板内直接引用父组件的数据。父组件的数据需要通过 prop 才能下发到子组件中。
子组件要显式地用 props
选项声明它预期的数据:
Vue.component(‘child‘, { // 声明 props props: [‘message‘], // 就像 data 一样,prop 也可以在模板中使用 // 同样也可以在 vm 实例中通过 this.message 来使用 template: ‘<span>{{ message }}</span>‘ }) |
然后我们可以这样向它传入一个普通字符串:
<child message="hello!"></child> |
结果:
hello!
camelCase vs. kebab-case
HTML 特性是不区分大小写的。所以,当使用的不是字符串模板时,camelCase (驼峰式命名) 的 prop 需要转换为相对应的 kebab-case (短横线分隔式命名):
Vue.component(‘child‘, { // 在 JavaScript 中使用 camelCase props: [‘myMessage‘], template: ‘<span>{{ myMessage }}</span>‘ }) |
<!-- 在 HTML 中使用 kebab-case --> <child my-message="hello!"></child> |
如果你使用字符串模板,则没有这些限制。
动态 Prop
与绑定到任何普通的 HTML 特性相类似,我们可以用 v-bind
来动态地将 prop 绑定到父组件的数据。每当父组件的数据变化时,该变化也会传导给子组件:
<div> <input v-model="parentMsg"> <br> <child v-bind:my-message="parentMsg"></child> </div> |
你也可以使用 v-bind
的缩写语法:
<child :my-message="parentMsg"></child> |
结果:
Message from parent
如果你想把一个对象的所有属性作为 prop 进行传递,可以使用不带任何参数的 v-bind
(即用 v-bind
而不是 v-bind:prop-name
)。例如,已知一个 todo
对象:
todo: { text: ‘Learn Vue‘, isComplete: false } |
然后:
<todo-item v-bind="todo"></todo-item> |
将等价于:
<todo-item v-bind:text="todo.text" v-bind:is-complete="todo.isComplete" ></todo-item> |
字面量语法 vs 动态语法
初学者常犯的一个错误是使用字面量语法传递数值:
<!-- 传递了一个字符串 "1" --> <comp some-prop="1"></comp> |
因为它是一个字面量 prop,它的值是字符串 "1"
而不是一个数值。如果想传递一个真正的 JavaScript 数值,则需要使用 v-bind
,从而让它的值被当作 JavaScript 表达式计算:
<!-- 传递真正的数值 --> <comp v-bind:some-prop="1"></comp> |
单向数据流
Prop 是单向绑定的:当父组件的属性变化时,将传导给子组件,但是反过来不会。这是为了防止子组件无意间修改了父组件的状态,来避免应用的数据流变得难以理解。
另外,每次父组件更新时,子组件的所有 prop 都会更新为最新值。这意味着你不应该在子组件内部改变 prop。如果你这么做了,Vue 会在控制台给出警告。
在两种情况下,我们很容易忍不住想去修改 prop 中数据:
- Prop 作为初始值传入后,子组件想把它当作局部数据来用;
- Prop 作为原始数据传入,由子组件处理成其它数据输出。
对这两种情况,正确的应对方式是:
- 定义一个局部变量,并用 prop 的值初始化它:
props: [‘initialCounter‘],data: function () {
return { counter: this.initialCounter }
}
- 定义一个计算属性,处理 prop 的值并返回:
props: [‘size‘],computed: {
normalizedSize: function () {
return this.size.trim().toLowerCase()
}
}
注意在 JavaScript 中对象和数组是引用类型,指向同一个内存空间,如果 prop 是一个对象或数组,在子组件内部改变它会影响父组件的状态。
Prop 验证
我们可以为组件的 prop 指定验证规则。如果传入的数据不符合要求,Vue 会发出警告。这对于开发给他人使用的组件非常有用。
要指定验证规则,需要用对象的形式来定义 prop,而不能用字符串数组:
Vue.component(‘example‘, { props: { // 基础类型检测 (`null` 指允许任何类型) propA: Number, // 可能是多种类型 propB: [String, Number], // 必传且是字符串 propC: { type: String, required: true }, // 数值且有默认值 propD: { type: Number, default: 100 }, // 数组/对象的默认值应当由一个工厂函数返回 propE: { type: Object, default: function () { return { message: ‘hello‘ } } }, // 自定义验证函数 propF: { validator: function (value) { return value > 10 } } } }) |
type
可以是下面原生构造器:
- String
- Number
- Boolean
- Function
- Object
- Array
- Symbol
type
也可以是一个自定义构造器函数,使用 instanceof
检测。
当 prop 验证失败,Vue 会抛出警告 (如果使用的是开发版本)。注意 prop 会在组件实例创建之前进行校验,所以在 default
或 validator
函数里,诸如 data
、computed
或methods
等实例属性还无法使用。
非 Prop 特性
所谓非 prop 特性,就是指它可以直接传入组件,而不需要定义相应的 prop。
尽管为组件定义明确的 prop 是推荐的传参方式,组件的作者却并不总能预见到组件被使用的场景。所以,组件可以接收任意传入的特性,这些特性都会被添加到组件的根元素上。
例如,假设我们使用了第三方组件 bs-date-input
,它包含一个 Bootstrap 插件,该插件需要在 input
上添加 data-3d-date-picker
这个特性。这时可以把特性直接添加到组件上 (不需要事先定义 prop
):
<bs-date-input data-3d-date-picker="true"></bs-date-input> |
添加属性 data-3d-date-picker="true"
之后,它会被自动添加到 bs-date-input
的根元素上。
替换/合并现有的特性
假设这是 bs-date-input
的模板:
<input type="date" class="form-control"> |
为了给该日期选择器插件增加一个特殊的主题,我们可能需要增加一个特殊的 class,比如:
<bs-date-input data-3d-date-picker="true" class="date-picker-theme-dark" ></bs-date-input> |
在这个例子当中,我们定义了两个不同的 class
值:
form-control
,来自组件自身的模板date-picker-theme-dark
,来自父组件
对于多数特性来说,传递给组件的值会覆盖组件本身设定的值。即例如传递 type="large"
将会覆盖 type="date"
且有可能破坏该组件!所幸我们对待 class
和 style
特性会更聪明一些,这两个特性的值都会做合并 (merge) 操作,让最终生成的值为:form-control date-picker-theme-dark
。
自定义事件
我们知道,父组件使用 prop 传递数据给子组件。但子组件怎么跟父组件通信呢?这个时候 Vue 的自定义事件系统就派得上用场了。
使用 v-on
绑定自定义事件
每个 Vue 实例都实现了事件接口,即:
- 使用
$on(eventName)
监听事件 - 使用
$emit(eventName)
触发事件
Vue 的事件系统与浏览器的 EventTarget API 有所不同。尽管它们的运行起来类似,但是 $on
和 $emit
并不是addEventListener
和 dispatchEvent
的别名。
另外,父组件可以在使用子组件的地方直接用 v-on
来监听子组件触发的事件。
不能用 $on
侦听子组件释放的事件,而必须在模板里直接用 v-on
绑定,参见下面的例子。
下面是一个例子:
<div id="counter-event-example"> <p>{{ total }}</p> <button-counter v-on:increment="incrementTotal"></button-counter> <button-counter v-on:increment="incrementTotal"></button-counter> </div> |
Vue.component(‘button-counter‘, { template: ‘<button v-on:click="incrementCounter">{{ counter }}</button>‘, data: function () { return { counter: 0 } }, methods: { incrementCounter: function () { this.counter += 1 this.$emit(‘increment‘) } }, }) new Vue({ el: ‘#counter-event-example‘, data: { total: 0 }, methods: { incrementTotal: function () { this.total += 1 } } }) |
0
0 0
在本例中,子组件已经和它外部完全解耦了。它所做的只是报告自己的内部事件,因为父组件可能会关心这些事件。请注意这一点很重要。
给组件绑定原生事件
有时候,你可能想在某个组件的根元素上监听一个原生事件。可以使用 v-on
的修饰符.native
。例如:
<my-component v-on:click.native="doTheThing"></my-component> |
.sync
修饰符
2.3.0+
在一些情况下,我们可能会需要对一个 prop 进行“双向绑定”。事实上,这正是 Vue 1.x 中的.sync
修饰符所提供的功能。当一个子组件改变了一个带 .sync
的 prop 的值时,这个变化也会同步到父组件中所绑定的值。这很方便,但也会导致问题,因为它破坏了单向数据流。由于子组件改变 prop 的代码和普通的状态改动代码毫无区别,当光看子组件的代码时,你完全不知道它何时悄悄地改变了父组件的状态。这在 debug 复杂结构的应用时会带来很高的维护成本。
上面所说的正是我们在 2.0 中移除 .sync
的理由。但是在 2.0 发布之后的实际应用中,我们发现 .sync
还是有其适用之处,比如在开发可复用的组件库时。我们需要做的只是让子组件改变父组件状态的代码更容易被区分。
从 2.3.0 起我们重新引入了 .sync
修饰符,但是这次它只是作为一个编译时的语法糖存在。它会被扩展为一个自动更新父组件属性的 v-on
监听器。
如下代码
<comp :foo.sync="bar"></comp> |
会被扩展为:
<comp :foo="bar" @update:foo="val => bar = val"></comp> |
当子组件需要更新 foo
的值时,它需要显式地触发一个更新事件:
this.$emit(‘update:foo‘, newValue) |
使用自定义事件的表单输入组件
自定义事件可以用来创建自定义的表单输入组件,使用 v-model
来进行数据双向绑定。要牢记:
<input v-model="something"> |
这不过是以下示例的语法糖:
<input v-bind:value="something" v-on:input="something = $event.target.value"> |
所以在组件中使用时,它相当于下面的简写:
<custom-input v-bind:value="something" v-on:input="something = arguments[0]"> </custom-input> |
所以要让组件的 v-model
生效,它应该 (从 2.2.0 起是可配置的):
- 接受一个
value
prop - 在有新的值时触发
input
事件并将新值作为参数
我们来看一个非常简单的货币输入的自定义控件:
<currency-input v-model="price"></currency-input> |
Vue.component(‘currency-input‘, { template: ‘\ <span>\ $\ <input\ ref="input"\ v-bind:value="value"\ v-on:input="updateValue($event.target.value)"\ >\ </span>\ ‘, props: [‘value‘], methods: { // 不是直接更新值,而是使用此方法来对输入值进行格式化和位数限制 updateValue: function (value) { var formattedValue = value // 删除两侧的空格符 .trim() // 保留 2 位小数 .slice( 0, value.indexOf(‘.‘) === -1 ? value.length : value.indexOf(‘.‘) + 3 ) // 如果值尚不合规,则手动覆盖为合规的值 if (formattedValue !== value) { this.$refs.input.value = formattedValue } // 通过 input 事件带出数值 this.$emit(‘input‘, Number(formattedValue)) } } }) |
$
当然,上面的例子还是比较初级的。比如,用户输入多个小数点或句号也是允许的,好恶心吧!因此我们需要一个复杂一些的例子,下面是一个更加完善的货币过滤器:
自定义组件的 v-model
2.2.0 新增
默认情况下,一个组件的 v-model
会使用 value
prop 和 input
事件。但是诸如单选框、复选框之类的输入类型可能把 value
用作了别的目的。model
选项可以避免这样的冲突:
Vue.component(‘my-checkbox‘, { model: { prop: ‘checked‘, event: ‘change‘ }, props: { checked: Boolean, // 这样就允许拿 `value` 这个 prop 做其它事了 value: String }, // ... }) |
<my-checkbox v-model="foo" value="some value"></my-checkbox> |
上述代码等价于:
<my-checkbox :checked="foo" @change="val => { foo = val }" value="some value"> </my-checkbox> |
注意你仍然需要显式声明 checked
这个 prop。
非父子组件的通信
有时候,非父子关系的两个组件之间也需要通信。在简单的场景下,可以使用一个空的 Vue 实例作为事件总线:
var bus = new Vue() |
// 触发组件 A 中的事件 bus.$emit(‘id-selected‘, 1) |
// 在组件 B 创建的钩子中监听事件 bus.$on(‘id-selected‘, function (id) { // ... }) |
在复杂的情况下,我们应该考虑使用专门的状态管理模式。
使用插槽分发内容
在使用组件时,我们常常要像这样组合它们:
<app> <app-header></app-header> <app-footer></app-footer> </app> |
注意两点:
<app>
组件不知道它会收到什么内容。这是由使用<app>
的父组件决定的。<app>
组件很可能有它自己的模板。
为了让组件可以组合,我们需要一种方式来混合父组件的内容与子组件自己的模板。这个过程被称为内容分发 (即 Angular 用户熟知的“transclusion”)。Vue.js 实现了一个内容分发 API,参照了当前 Web Components 规范草案,使用特殊的 <slot>
元素作为原始内容的插槽。
编译作用域
在深入内容分发 API 之前,我们先明确内容在哪个作用域里编译。假定模板为:
<child-component> {{ message }} </child-component> |
message
应该绑定到父组件的数据,还是绑定到子组件的数据?答案是父组件。组件作用域简单地说是:
父组件模板的内容在父组件作用域内编译;子组件模板的内容在子组件作用域内编译。
一个常见错误是试图在父组件模板内将一个指令绑定到子组件的属性/方法:
<!-- 无效 --> <child-component v-show="someChildProperty"></child-component> |
假定 someChildProperty
是子组件的属性,上例不会如预期那样工作。父组件模板并不感知子组件的状态。
如果要绑定子组件作用域内的指令到一个组件的根节点,你应当在子组件自己的模板里做:
Vue.component(‘child-component‘, { // 有效,因为是在正确的作用域内 template: ‘<div v-show="someChildProperty">Child</div>‘, data: function () { return { someChildProperty: true } } }) |
类似地,被分发的内容会在父作用域内编译。
单个插槽
除非子组件模板包含至少一个 <slot>
插口,否则父组件的内容将会被丢弃。当子组件模板只有一个没有属性的插槽时,父组件传入的整个内容片段将插入到插槽所在的 DOM 位置,并替换掉插槽标签本身。
最初在 <slot>
标签中的任何内容都被视为备用内容。备用内容在子组件的作用域内编译,并且只有在宿主元素为空,且没有要插入的内容时才显示备用内容。
假定 my-component
组件有如下模板:
<div> <h2>我是子组件的标题</h2> <slot> 只有在没有要分发的内容时才会显示。 </slot> </div> |
父组件模板:
<div> <h1>我是父组件的标题</h1> <my-component> <p>这是一些初始内容</p> <p>这是更多的初始内容</p> </my-component> </div> |
渲染结果:
<div> <h1>我是父组件的标题</h1> <div> <h2>我是子组件的标题</h2> <p>这是一些初始内容</p> <p>这是更多的初始内容</p> </div> </div> |
具名插槽
<slot>
元素可以用一个特殊的特性 name
来进一步配置如何分发内容。多个插槽可以有不同的名字。具名插槽将匹配内容片段中有对应 slot
特性的元素。
仍然可以有一个匿名插槽,它是默认插槽,作为找不到匹配的内容片段的备用插槽。如果没有默认插槽,这些找不到匹配的内容片段将被抛弃。
例如,假定我们有一个 app-layout
组件,它的模板为:
<div class="container"> <header> <slot name="header"></slot> </header> <main> <slot></slot> </main> <footer> <slot name="footer"></slot> </footer> </div> |
父组件模板:
<app-layout> <h1 slot="header">这里可能是一个页面标题</h1> <p>主要内容的一个段落。</p> <p>另一个主要段落。</p> <p slot="footer">这里有一些联系信息</p> </app-layout> |
渲染结果为:
<div class="container"> <header> <h1>这里可能是一个页面标题</h1> </header> <main> <p>主要内容的一个段落。</p> <p>另一个主要段落。</p> </main> <footer> <p>这里有一些联系信息</p> </footer> </div> |
在设计组合使用的组件时,内容分发 API 是非常有用的机制。
作用域插槽
2.1.0 新增
作用域插槽是一种特殊类型的插槽,用作一个 (能被传递数据的) 可重用模板,来代替已经渲染好的元素。
在子组件中,只需将数据传递到插槽,就像你将 prop 传递给组件一样:
<div class="child"> <slot text="hello from child"></slot> </div> |
在父级中,具有特殊特性 slot-scope
的 <template>
元素必须存在,表示它是作用域插槽的模板。slot-scope
的值将被用作一个临时变量名,此变量接收从子组件传递过来的 prop 对象:
<div class="parent"> <child> <template slot-scope="props"> <span>hello from parent</span> <span>{{ props.text }}</span> </template> </child> </div> |
如果我们渲染上述模板,得到的输出会是:
<div class="parent"> <div class="child"> <span>hello from parent</span> <span>hello from child</span> </div> </div> |
在 2.5.0+,
slot-scope
能被用在任意元素或组件中而不再局限于<template>
。
作用域插槽更典型的用例是在列表组件中,允许使用者自定义如何渲染列表的每一项:
<my-awesome-list :items="items"> <!-- 作用域插槽也可以是具名的 --> <li slot="item" slot-scope="props" class="my-fancy-item"> {{ props.text }} </li> </my-awesome-list> |
列表组件的模板:
<ul> <slot name="item" v-for="item in items" :text="item.text"> <!-- 这里写入备用内容 --> </slot> </ul> |
解构
slot-scope
的值实际上是一个可以出现在函数签名参数位置的合法的 JavaScript 表达式。这意味着在受支持的环境 (单文件组件或现代浏览器) 中,您还可以在表达式中使用 ES2015 解构:
<child> <span slot-scope="{ text }">{{ text }}</span> </child> |
动态组件
通过使用保留的 <component>
元素,动态地绑定到它的 is
特性,我们让多个组件可以使用同一个挂载点,并动态切换:
var vm = new Vue({ el: ‘#example‘, data: { currentView: ‘home‘ }, components: { home: { /* ... */ }, posts: { /* ... */ }, archive: { /* ... */ } } }) |
<component v-bind:is="currentView"> <!-- 组件在 vm.currentview 变化时改变! --> </component> |
也可以直接绑定到组件对象上:
var Home = { template: ‘<p>Welcome home!</p>‘ } var vm = new Vue({ el: ‘#example‘, data: { currentView: Home } }) |
keep-alive
如果把切换出去的组件保留在内存中,可以保留它的状态或避免重新渲染。为此可以添加一个 keep-alive
指令参数:
<keep-alive> <component :is="currentView"> <!-- 非活动组件将被缓存! --> </component> </keep-alive> |
在 API 参考中查看更多 <keep-alive>
的细节。
杂项
编写可复用组件
在编写组件时,最好考虑好以后是否要进行复用。一次性组件间有紧密的耦合没关系,但是可复用组件应当定义一个清晰的公开接口,同时也不要对其使用的外层数据作出任何假设。
Vue 组件的 API 来自三部分——prop、事件和插槽:
- Prop 允许外部环境传递数据给组件;
- 事件允许从组件内触发外部环境的副作用;
- 插槽允许外部环境将额外的内容组合在组件中。
使用 v-bind
和 v-on
的简写语法,模板的意图会更清楚且简洁:
<my-component :foo="baz" :bar="qux" @event-a="doThis" @event-b="doThat" > <img slot="icon" src="..."> <p slot="main-text">Hello!</p> </my-component> |
子组件引用
尽管有 prop 和事件,但是有时仍然需要在 JavaScript 中直接访问子组件。为此可以使用ref
为子组件指定一个引用 ID。例如:
<div id="parent"> <user-profile ref="profile"></user-profile> </div> |
var parent = new Vue({ el: ‘#parent‘ }) // 访问子组件实例 var child = parent.$refs.profile |
当 ref
和 v-for
一起使用时,获取到的引用会是一个数组,包含和循环数据源对应的子组件。
$refs
只在组件渲染完成后才填充,并且它是非响应式的。它仅仅是一个直接操作子组件的应急方案——应当避免在模板或计算属性中使用 $refs
。
异步组件
在大型应用中,我们可能需要将应用拆分为多个小模块,按需从服务器下载。为了进一步简化,Vue.js 允许将组件定义为一个工厂函数,异步地解析组件的定义。Vue.js 只在组件需要渲染时触发工厂函数,并且把结果缓存起来,用于后面的再次渲染。例如:
Vue.component(‘async-example‘, function (resolve, reject) { setTimeout(function () { // 将组件定义传入 resolve 回调函数 resolve({ template: ‘<div>I am async!</div>‘ }) }, 1000) }) |
工厂函数接收一个 resolve
回调,在收到从服务器下载的组件定义时调用。也可以调用reject(reason)
指示加载失败。这里使用 setTimeout
只是为了演示,实际上如何获取组件完全由你决定。推荐配合 webpack 的代码分割功能 来使用:
Vue.component(‘async-webpack-example‘, function (resolve) { // 这个特殊的 require 语法告诉 webpack // 自动将编译后的代码分割成不同的块, // 这些块将通过 Ajax 请求自动下载。 require([‘./my-async-component‘], resolve) }) |
你可以在工厂函数中返回一个 Promise
,所以当使用 webpack 2 + ES2015 的语法时可以这样:
Vue.component( ‘async-webpack-example‘, // 该 `import` 函数返回一个 `Promise` 对象。 () => import(‘./my-async-component‘) ) |
当使用局部注册时,也可以直接提供一个返回 Promise
的函数:
new Vue({ // ... components: { ‘my-component‘: () => import(‘./my-async-component‘) } }) |
如果你是 Browserify 用户,可能就无法使用异步组件了,它的作者已经表明Browserify 将“永远不会支持异步加载”。Browserify 社区发现了一些解决方法,可能会有助于已存在的复杂应用。对于其他场景,我们推荐使用 webpack,因为它对异步加载进行了内置、全面的支持。
高级异步组件
2.3.0 新增
自 2.3.0 起,异步组件的工厂函数也可以返回一个如下的对象:
const AsyncComp = () => ({ // 需要加载的组件。应当是一个 Promise component: import(‘./MyComp.vue‘), // 加载中应当渲染的组件 loading: LoadingComp, // 出错时渲染的组件 error: ErrorComp, // 渲染加载中组件前的等待时间。默认:200ms。 delay: 200, // 最长等待时间。超出此时间则渲染错误组件。默认:Infinity timeout: 3000 }) |
注意,当一个异步组件被作为 vue-router
的路由组件使用时,这些高级选项都是无效的,因为在路由切换前就会提前加载所需要的异步组件。另外,如果你要在路由组件中使用上述写法,需要使用 vue-router
2.4.0 以上的版本。
组件命名约定
当注册组件 (或者 prop) 时,可以使用 kebab-case (短横线分隔命名)、camelCase (驼峰式命名) 或 PascalCase (单词首字母大写命名)。
// 在组件定义中 components: { // 使用 kebab-case 注册 ‘kebab-cased-component‘: { /* ... */ }, // 使用 camelCase 注册 ‘camelCasedComponent‘: { /* ... */ }, // 使用 PascalCase 注册 ‘PascalCasedComponent‘: { /* ... */ } } |
在 HTML 模板中,请使用 kebab-case:
<!-- 在 HTML 模板中始终使用 kebab-case --> <kebab-cased-component></kebab-cased-component> <camel-cased-component></camel-cased-component> <pascal-cased-component></pascal-cased-component> |
当使用字符串模式时,可以不受 HTML 大小写不敏感的限制。这意味实际上在模板中,你可以使用下面的方式来引用你的组件:
- kebab-case
- camelCase 或 kebab-case (如果组件已经被定义为 camelCase)
- kebab-case、camelCase 或 PascalCase (如果组件已经被定义为 PascalCase)
components: { ‘kebab-cased-component‘: { /* ... */ }, camelCasedComponent: { /* ... */ }, PascalCasedComponent: { /* ... */ } } |
<kebab-cased-component></kebab-cased-component> <camel-cased-component></camel-cased-component> <camelCasedComponent></camelCasedComponent> <pascal-cased-component></pascal-cased-component> <pascalCasedComponent></pascalCasedComponent> <PascalCasedComponent></PascalCasedComponent> |
这意味着 PascalCase 是最通用的声明约定而 kebab-case 是最通用的使用约定。
如果组件未经 slot
元素传入内容,你甚至可以在组件名后使用 /
使其自闭合:
<my-component/> |
当然,这只在字符串模板中有效。因为自闭的自定义元素是无效的 HTML,浏览器原生的解析器也无法识别它。
递归组件
组件在它的模板内可以递归地调用自己。不过,只有当它有 name
选项时才可以这么做:
name: ‘unique-name-of-my-component‘ |
当你利用 Vue.component
全局注册了一个组件,全局的 ID 会被自动设置为组件的name
。
Vue.component(‘unique-name-of-my-component‘, { // ... }) |
如果稍有不慎,递归组件可能导致死循环:
name: ‘stack-overflow‘, template: ‘<div><stack-overflow></stack-overflow></div>‘ |
上面组件会导致一个“max stack size exceeded”错误,所以要确保递归调用有终止条件 (比如递归调用时使用 v-if
并最终解析为 false
)。
组件间的循环引用
假设你正在构建一个文件目录树,像在 Finder 或资源管理器中。你可能有一个tree-folder
组件:
<p> <span>{{ folder.name }}</span> <tree-folder-contents :children="folder.children"/> </p> |
以及一个 tree-folder-contents
组件:
<ul> <li v-for="child in children"> <tree-folder v-if="child.children" :folder="child"/> <span v-else>{{ child.name }}</span> </li> </ul> |
当你仔细看时,会发现在渲染树上这两个组件同时为对方的父节点和子节点——这是矛盾的!当使用 Vue.component
将这两个组件注册为全局组件的时候,框架会自动为你解决这个矛盾。如果你已经是这样做的,就跳过下面这段吧。
然而,如果你使用诸如 webpack 或者 Browserify 之类的模块化管理工具来 require/import 组件的话,就会报错了:
Failed to mount component: template or render function not defined. |
为了解释为什么会报错,简单的将上面两个组件称为 A 和 B。模块系统看到它需要 A,但是首先 A 需要 B,但是 B 需要 A,而 A 需要 B,循环往复。因为不知道到底应该先解析哪个,所以将会陷入无限循环。要解决这个问题,我们需要在其中一个组件中告诉模块化管理系统:“A 虽然最后会用到 B,但是不需要优先导入 B”。
在我们的例子中,可以选择让 tree-folder
组件中来做这件事。我们知道引起矛盾的子组件是 tree-folder-contents
,所以我们要等到 beforeCreate
生命周期钩子中才去注册它:
beforeCreate: function () { this.$options.components.TreeFolderContents = require(‘./tree-folder-contents.vue‘) } |
问题解决了!
内联模板
如果子组件有 inline-template
特性,组件将把它的内容当作它的模板,而不是把它当作分发内容。这让模板编写起来更灵活。
<my-component inline-template> <div> <p>这些将作为组件自身的模板。</p> <p>而非父组件透传进来的内容。</p> </div> </my-component> |
但是 inline-template
让模板的作用域难以理解。使用 template
选项在组件内定义模板或者在 .vue
文件中使用 template
元素才是最佳实践。
X-Template
另一种定义模板的方式是在 JavaScript 标签里使用 text/x-template
类型,并且指定一个 id。例如:
<script type="text/x-template" id="hello-world-template"> <p>Hello hello hello</p> </script> |
Vue.component(‘hello-world‘, { template: ‘#hello-world-template‘ }) |
这在有很多大模板的演示应用或者特别小的应用中可能有用,其它场合应该避免使用,因为这将模板和组件的其它定义分离了。
对低开销的静态组件使用 v-once
尽管在 Vue 中渲染 HTML 很快,不过当组件中包含大量静态内容时,可以考虑使用 v-once
将渲染结果缓存起来,就像这样:
Vue.component(‘terms-of-service‘, { template: ‘\ <div v-once>\ <h1>Terms of Service</h1>\ ...很多静态内容...\ </div>\ ‘ }) |