组件系统是Vue.js其中一个重要的概念,它提供了一种抽象,让我们可以使用独立可复用的小组件来构建大型应用,任意类型的应用界面都可以抽象为一个组件树。
那么什么是组件呢?组件可以扩展HTML元素,封装可重用的HTML代码,我们可以将组件看作自定义的HTML元素。
一、组件的创建和注册基本步骤
Vue.js的组件的使用有3个步骤:创建组件构造器、注册组件和使用组件。
"app">
上面代码演示了这3个步骤:可以看到,使用组件和使用普通的HTML元素没什么区别。
二、理解组件的创建和注册
我们用以下几个步骤来理解组件的创建和注册:
Vue.extend()
是Vue构造器的扩展,调用Vue.extend()
创建的是一个组件构造器,而不是一个具体的组件实例。
2.
Vue.extend()
构造器有一个选项对象,选项对象的template
属性用于定义组件要渲染的HTML。
3. 使用
Vue.component()
注册组件时,需要提供2个参数,第1个参数是组件的标签,第2个参数是组件构造器。
4.
Vue.component()
方法内部会调用组件构造器,创建一个组件实例。
5. 组件应该挂载到某个Vue实例下,否则它不会生效。
请注意第5点,只有挂载到vue实例下组件才起到作用。
三、全局注册和局部注册
调用Vue.component()
注册组件时,组件的注册是全局的,这意味着该组件可以在任意Vue实例下使用。
如果不需要全局注册,或者是让组件使用在其它组件内,可以用选项对象的components属性实现局部注册。
局部组件注册方式:
// 1.创建一个组件构造器
var myComponent = Vue.extend({
template: 'This is my first component!'
})
new Vue({
el: '#app',
components: {
// 2. 将myComponent组件注册到Vue实例下
'my-component' : myComponent
}
});
注意:该局部组件只能在app下才能起作用
四、父组件和子组件
我们可以在组件中定义并使用其他组件,这就构成了父子组件的关系。
"app">
注意:组件的模板只能有一个根元素。下面的情况是不允许的。
template: 'This is parent component!
',
我们分几个步骤来理解这段代码:
(1)var Child = Vue.extend(...)
定义一了个Child组件构造器
(2)var Parent = Vue.extend(...)
定义一个Parent组件构造器
(3)components: { 'child-component': Child }
,将Child组件注册到Parent组件,并将Child组件的标签设置为child-component
。
This is a Parent component (4)template :'
,在Parent组件内以标签的形式使用Child组件。
(5)Vue.component('parent-component', Parent)
全局注册Parent组件
(6)在页面中使用
Child组件是在Parent组件中局部注册的,它只能在Parent组件中使用,确切地说:子组件只能在父组件的template中使用。
请注意下面两种子组件的使用方式是错误的:
1、以子标签的形式在父组件中使用
2、在父组件标签外使用子组件
//1. 以子标签的形式在父组件中使用
"app">
//为什么这种方式无效呢?因为当子组件注册到父组件时,Vue.js会编译好父组件的模板,模板的内容已经决定了父组件将要渲染的HTML。
… 相当于运行时,它的一些子标签只会被当作普通的HTML来执行, 不是标准的HTML标签,会被浏览器直接忽视掉。
//2. 在父组件标签外使用子组件
"app">
//运行这段代码,浏览器会提示以下错误:Unknown custom element: - did you register the component correctly?
五、组件注册语法糖
以上组件注册的方式有些繁琐,Vue.js为了简化这个过程,提供了注册语法糖:相当于把Vue.extend()直接放入第二个参数
1、使用Vue.component()直接创建和注册组件:
// 全局注册,my-component是标签名称,后面跟模板template
Vue.component('my-component',{
template: 'This is the first component!'
})
Vue.component()
的第1个参数是标签名称,第2个参数是一个选项对象,使用选项对象的template属性定义组件模板。使用这种方式,Vue在背后会自动地调用Vue.extend()
。
2、在选项对象的components属性中实现局部注册:
var vm = new Vue({
el: '#app',
components: {
// 局部注册,my-component2是标签名称
'my-component2': {
template: 'This is the second component!'
},
// 局部注册,my-component3是标签名称
'my-component3': {
template: 'This is the third component!'
}
}
})
六、使用script或template标签
尽管语法糖简化了组件注册,但在template选项中拼接HTML元素比较麻烦,这也导致了HTML和JavaScript的高耦合性。庆幸的是,Vue.js提供了两种方式将定义在JavaScript中的HTML模板分离出来。
1、使用