指南
基础概要
- 安装
- 介绍
- Vue 实例
- 模板语法
- computed 属性和 watcher
- class 和 style 绑定
- 根据条件进行渲染
- 列表渲染
- 事件处理
- 表单 input 绑定
- 组件基础
过渡 & 动画
- 进入、离开和列表的过渡
深入组件
- 组件注册
- props
- 自定义事件
- slot
- 动态组件和异步组件
- Handling Edge Cases
- 状态间的过渡
可重用 & 合成
- mixin
- 自定义指令
- render 函数 & jsx
- 插件
- 过滤器
工具
- 生产环境部署
- 单文件组件
- 单元测试
- TypeScript 支持
扩展升级
- 路由
- 状态管理
- 服务端渲染
内部原理
- 深入响应式原理
升级迁移
- 从 Vue 1.x 迁移
- 从 Vue Router 0.7.x 迁移
- 从 Vuex 0.6.x 迁移到 1.0
其他更多
- 对比其他框架
- 加入 Vue.js 社区
- 认识团队
render 函数 & jsx
基础
在绝大多数情况下,Vue 推荐你使用模板来组织构建 HTML。然而在一些场景下,你确实需要完整的 JavaScript 编程能力。作为模板的替代增强,你可以使用render 函数,它更接近编译器。
让我们深入一个例子,其中,使用 render
函数是比较推荐的方式。假设你要生成固定的标题:
<h1> |
对于以上 HTML,你决定要使用如下组件接口:
<anchored-heading :level="1">Hello world!</anchored-heading> |
当你开始创建一个只根据 level
prop 生成标题的组件时,你会很快想到这样实现:
<script type="text/x-template" id="anchored-heading-template"> |
Vue.component('anchored-heading', { |
在这种场景中,并不适合使用模板。不仅会很繁琐,而且要在每个不同级的 h 标签中重复 <slot></slot>
,并且在其中添加固定元素时也必须做同样的事。
虽然大多数组件中都非常适合使用模板,但是在这里明显是一种特殊场景。因此,让我们尝试使用 render
函数来重写上面的示例:
Vue.component('anchored-heading', { |
现在显得简单清晰!一定程度上。代码变得简短很多,但是需要更加熟悉 Vue 实例属性。在这种情况下,你必须知道在不使用 slot
属性向组件中传递内容时(比如 anchored-heading
中的 Hello world!
),这些子节点会被存储到组件实例的 $slots.default
中。如果你对此还不够了解,在深入 render 函数之前,建议先阅读实例属性 API。
nodes, trees 和 virtual DOM
在我们深入 render 函数之前,略微了解浏览器的工作原理是比较重要的事情。以如下 HTML 为例:
<div> |
在浏览器读取这段代码时,会构建一个 “DOM 节点”树(tree of “DOM nodes”),以帮助浏览器跟踪所有情况,就像通过创建一个家谱树,来跟踪一个大家族。
对于以上 HTML,其 DOM 节点树如下:
每个元素都是一个节点。每一段文字都是一个节点。甚至注释也都是节点!节点只是页面的一部分。正如在一棵家谱树中一样,每个节点都可以有子节点(也就是说,每个节点都可以包含多个子节点)。
有效地更新所有这些节点可能是很困难的,但幸运的是,你无需手动执行。相反,只需在 Vue 模板中,在页面中添加你需要用到的 HTML:
<h1>{{ blogTitle }}</h1> |
或者通过一个 render 函数:
render: function (createElement) { |
在这两种场景中,Vue 会自动保持页面更新,更确切地说,在 blogTitle
修改时,也同样能够及时更新。
virtual DOM
Vue 通过实现一个 virtual DOM,来跟踪那些「真正需要对 DOM 所做的修改」。仔细看看这一行:
return createElement('h1', this.blogTitle) |
createElement
实际返回的是什么呢?准确地说,返回的并非一个真正的 DOM 元素。可能更确切地说,应该将其命名为 createNodeDescription
(译注:createNodeDescription,创建节点描述),包含「哪些节点应该由 Vue 渲染在页面上」的相关描述信息,也包括所有子节点的相关描述。我们将以上这个节点描述称为 “virtual node”(译注:virtual node,虚拟节点),通常缩写为 VNode。”virtual DOM” 就是由 Vue 组件树构建出来的,被称为整个 VNodes 树。
createElement
参数
接下来你必须熟悉的事情是,如何在 createElement
函数中使用模板功能。以下是 createElement
接受的参数:
// @returns {VNode} |
深入了解数据对象(The Data Object In-Depth)
需要注意的一点:类似于 v-bind:class
和 v-bind:style
在模板中有着特殊的处理,下面这些字段(field)都处于 VNode 数据对象的顶层(top-level)。此对象还允许绑定普通的 HTML 属性和 DOM 属性,如 innerHTML
(这将替换为 v-html
指令):
{ |
完整示例
有了这些知识,我们现在就可以完成开始想实现的组件:
var getChildrenTextContent = function (children) { |
限制(Constraints)
VNodes 必须唯一
组件树中的所有 VNode 必须是唯一的。也就是说,以下 render 函数无效:
render: function (createElement) { |
如果你真的需要多次重复相同的元素/组件,你可以使用工厂函数来实现。例如,下面的 render 函数,完美有效地渲染了 20 个相同的段落。
render: function (createElement) { |
使用原生 JavaScript 的替换模板功能(Replacing Template Features with Plain JavaScript)
v-if
and v-for
无论什么功能,都可以在原生 JavaScript 中轻松实现,所以 Vue 的 render 函数无需提供专用的替代方案。例如,在模板中使用 v-if
和 v-for
:
<ul v-if="items.length"> |
这可以在 render 函数中,通过重写为 JavaScript 的 if
/else
和 map
来实现:
props: ['items'], |
v-model
在 render 函数中没有直接和 v-model
对标的功能 - 你必须自己实现逻辑:
props: ['value'], |
这就是深入底层实现需要付出的代价,但是和 v-model
相比较,这也可以更好地控制交互细节。
事件 & 按键修饰符(Event & Key Modifiers)
对于 .passive
, .capture
和 .once
这样的事件修饰符, Vue 提供了用于 on
的前缀:
修饰符 | 前缀 |
---|---|
.passive |
& |
.capture |
! |
.once |
~ |
.capture.once 或.once.capture |
~! |
示例:
on: { |
对于其他的事件和关键字修饰符, 你可以在处理程序中使用事件方法实现:
Modifier(s) | Equivalent in Handler |
---|---|
.stop |
event.stopPropagation() |
.prevent |
event.preventDefault() |
.self |
if (event.target !== event.currentTarget) return |
Keys:.enter , .13 |
if (event.keyCode !== 13) return (change 13 to another key code for other key modifiers) |
Modifiers Keys:.ctrl , .alt , .shift , .meta |
if (!event.ctrlKey) return (change ctrlKey to altKey , shiftKey , or metaKey , respectively) |
这里是所有修饰符一起使用的例子:
on: { |
slots
使用this.$slots
访问静态插槽内容作为 VNode 数组:
render: function (createElement) { |
使用 this.$scopedSlots
访问作用域插槽作为返回 VNodes 的函数:
props: ['message'], |
使用 render 函数传递作用域插槽到子组件,使用 VNode 数据中的 scopedSlots
关键字:
render: function (createElement) { |
JSX
如果你写了很多 render
函数,可能会觉得痛苦:
createElement( |
特别是模板如此简单的情况下:
<anchored-heading :level="1"> |
这就是会有一个 Babel plugin 插件,用于在 Vue 中使用 JSX 语法的原因,它可以让我们回到于更接近模板的语法上。
import AnchoredHeading from './AnchoredHeading.vue' |
将 h
作为 createElement
的别名是 Vue 生态系统中的一个通用惯例,实际上也是 JSX 所要求的,如果在作用域中 h
失去作用, 在应用中会触发报错。
更多关于 JSX 映射到 JavaScript,阅读 使用文档。
函数式组件
前面我们创建的锚点标题组件比较简单,没有管理维护状态,或者 watch 任何传递给它的状态,也没有生命周期方法。它只是一个接收 props 的函数。
在这个例子中,我们将组件记为 functional
,这意味它无状态(没有 响应式 data)),无实例(没有 this
上下文)。一个函数式组件就像这样:
Vue.component('my-component', { |
注意:在 2.3.0 之前的版本中,如果一个函数式组件想要接收 props,则
props
选项是必选项。而在 2.3.0+ 版本中,你可以省略props
选项,所有组件节点上的发现的属性,都会被隐式地提取为 props。
在 2.5.0+ 版本,如果你使用的是单文件组件,则可以通过如下方式,声明基于模板的函数式组件:
<template functional> |
组件需要的一切都是通过 context
传递,包括:
props
:一个提供 props 的对象children
:一个由 VNode 子节点构成的数组slots
:一个返回 slots 对象的函数data
:传递给组件的整个 data 对象,作为createElement
函数的第二个参数parent
:一个父组件的引用listeners
:(2.3.0+)一个包含父组件上注册的事件监听器的对象。这是一个指向data.on
的别名。injections
:(2.3.0+)如果使用了inject
选项,则 injections 对象中包含了应当被注入的属性。
在添加 functional: true
之后,修改下锚点标题组件的 render 函数,需要添加 context
参数,将 this.$slots.default
修改为 context.children
,然后将 this.level
修改为 context.props.level
。
由于函数式组件只是函数,因此渲染性能开销也低很多。然而,由于缺少持久性实例,意味着它们不会出现在 Vue devtools 的组件树中。
在作为容器组件(wrapper component)时它们也同样非常有用。例如,当你需要实现如下需求:
- 程序化地在选择多个组件中的一个组件进行委派
- 在将 children, props, data 传递给子组件之前操作它们。
以下是一个 smart-list
组件示例,我们能将它的行为委派给更加具体的组件中,这取决于向其传入的 prop:
var EmptyList = { /* ... */ } |
向子元素或子组件传递属性和事件
在普通组件中,非 Prop 属性将自动添加到组件的根元素中,并会替换或智能合并具有相同名称的所有已有属性。
然而,在函数式组件中,则要求你显式定义该行为:
Vue.component('my-functional-button', { |
向 createElement
传递 context.data
作为第二个参数,我们就把 my-functional-button
组件上所有的属性和事件监听器都向下进行传递。这是非常明确的行为,事实上,这些事件甚至不需要添加 .native
修饰符。
如果你在使用基于模板的函数式组件,那么你还必须手动添加属性和监听器。由于我们可以访问到其独立的上下文内容,所以我们可以使用 data.attrs
来延续传递所有 HTML 属性,以及使用 listeners
(也就是 data.on
的别名)来延续传递所有事件监听器。
<template functional> |
slots()
和 children
对比
你可能想知道为什么同时需要 slots()
和 children
。slots().default
不是和 children
类似的吗?在一些场景中,是这样的 - 但是,如果是具有如下 children 的函数式组件呢?
<my-functional-component> |
对于这个组件,children
会给你两个 p 标签,而 slots().default
只会传递第二个匿名 p 标签,slots().foo
会传递第一个具名 p 标签。同时具有 children
和 slots()
,可以帮助你进行明确选择,使组件确定是通过一个 slot 系统处理,还是直接延续传递 children
,委派给其他组件去负责处理。
模板编译
你或许很有兴趣了解 Vue 模板实际编译为 render 函数的过程。这是一个实现细节,通常不需要关心,但如果你想看看特定模板的功能是怎样被编译的,你会发现会非常有趣。下面是一个使用 Vue.compile
实时编译模板字符串的简单示例:
{{ result.render }}
_m({{ index }}): {{ fn }}
{{ result.staticRenderFns }}
{{ result }}