Vuejs——(8)Vuejs组件的定义

Posted 岁月如歌(丁盛)

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Vuejs——(8)Vuejs组件的定义相关的知识,希望对你有一定的参考价值。

 

目录(?)[+]

 

 

本篇资料来于官方文档:

http://cn.vuejs.org/guide/components.html

本文是在官方文档的基础上,更加细致的说明,代码更多更全。

简单来说,更适合新手阅读


 

(二十五)组件的定义

①组件的作用:

【1】扩展HTML元素,封装可重用的代码;

【2】组件是自定义元素,Vuejs的编译器可以为其添加特殊的功能;

【3】某些情况下,组件可以是原生HTML元素的形式,以is的方式扩展。

 

②写一个标准的组件:

分为以下几步:

【1】挂载组件的地方,需要是Vue实例所渲染的html元素,具体来说,比如上面的<div id=”app”></div>这样的html元素及他的子节点;

 

【2】定义一个组件,用

var 变量名 = Vue.extend({template:”这里是html的模板内容”})

这样的形式创建,例如:

[javascript] view plain copy
 
 技术分享技术分享
  1. //定义一个组件  
  2. var btn = Vue.extend({  
  3.     template: "<button>这是一个按钮</button>"  
  4. })  

 

【3】将定义的组件注册到Vue实例上,这会让指定标签,被组件的内容所替代。

如代码:

[javascript] view plain copy
 
 技术分享技术分享
  1. //注册他到Vue实例上  
  2. Vue.component("add-button", btn);  

具体而言,每一个以下这样的标签(在Vue的根实例范围内的)

[html] view plain copy
 
 技术分享技术分享
  1. <add-button></add-button>  

会被

[html] view plain copy
 
 技术分享技术分享
  1. <button>这是一个按钮</button>  

所替代。

 

【4】以上方法是全局注册(每个Vue实例的add-button标签都会被我们定义的所替代);

解决办法是局部注册。

 

如代码:(这是是设置了template属性,也可以在没有这个属性的时候,在<div id=”app”></div>标签内放置<add-button></add-button>标签

[html] view plain copy
 
 技术分享技术分享
  1. <div id="app">  
  2. </div>  
  3. <script>  
  4.     //定义一个组件  
  5.     var btn = Vue.extend({  
  6.         template: "<button>这是一个按钮</button>"  
  7.     })  
  8.   
  9.     Vue.component("add-button", btn);  
  10.   
  11.     //创建根实例,也就是说让Vue对这个根生效  
  12.     var vm = new Vue({  
  13.         el: ‘#app‘,  
  14.         template: "<add-button></add-button>"  
  15.     });  
  16. </script>  

 

 

③局部注册组件:

简单来说,只对这一个Vue实例生效,具体做法是,在注册那一步,跳过;

然后在声明Vue实例的时候,将添加到components这个属性中(他是一个对象,以KV形式放置)(注意,这个单词多一个s)

如代码:

[html] view plain copy
 
 技术分享技术分享
  1. <div id="app">  
  2. </div>  
  3. <script>  
  4.     //定义一个组件  
  5.     var btn = Vue.extend({  
  6.         template: "<button>这是一个按钮</button>"  
  7.     })  
  8.   
  9.     //创建根实例,也就是说让Vue对这个根生效  
  10.     var vm = new Vue({  
  11.         el: ‘#app‘,  
  12.         template: "<add-button></add-button>",  
  13.         components: {  
  14.             "add-button": btn  
  15.         }  
  16.     });  
  17. </script>  

 

注:

根据官方教程,这种方法(指局部注册),也适用于其他资源,比如指令过滤器过渡

 

 

④步骤简化:

【1】定义组件注册组件结合起来一步完成:

[javascript] view plain copy
 
 技术分享技术分享
  1. //定义一个组件  
  2. Vue.component("add-button", {  
  3.     template: "<button>这是一个按钮</button>"  
  4. });  

 

【2】局部注册时,定义和注册一步完成:

[javascript] view plain copy
 
 技术分享技术分享
  1. //创建根实例,也就是说让Vue对这个根生效  
  2. var vm = new Vue({  
  3.     el: ‘#app‘,  
  4.     template: "<add-button></add-button>",  
  5.     components: {  
  6.         "add-button": {  
  7.             template: "<button>这是一个按钮</button>"  
  8.         }  
  9.     }  
  10. });  

 

⑤data属性

直接给组件添加data属性是不可以的(无效);

 

原因在于,假如这么干,那么组件的data属性有可能是一个对象,而这个对象也有可能是外部传入的(例如先声明一个对象,然后这个对象作为data的值),可能导致这个组件的所有副本,都共享一个对象(那个外部传入的),这显然是不对的。

 

因此,data属性应该是一个函数,然后有一个返回值,这个返回值作为data属性的值。

 

且这个返回值应该是一个全新的对象(即深度复制的,避免多个组件共享一个对象);

 

如代码:

[javascript] view plain copy
 
 技术分享技术分享
  1. var vm = new Vue({  
  2.     el: ‘#app‘,  
  3.     template: "<add-button></add-button>",  
  4.     components: {  
  5.         "add-button": {  
  6.             template: "<button>这是一个按钮{{btn}}</button>",  
  7.             data: function () {  
  8.                 return {btn: "123"};  
  9.             }  
  10.         }  
  11.     }  
  12. });  

 

另外,假如这样的话,btn的值是一样的(因为他们实际上还是共享了一个对象)

[html] view plain copy
 
 技术分享技术分享
  1. <div id="app">  
  2. </div>  
  3. <div id="app2">  
  4. </div>  
  5. <script>  
  6.     var obj = {btn: "123"};  
  7.     var vm = new Vue({  
  8.         el: ‘#app‘,  
  9.         template: "<add-button></add-button>",  
  10.         components: {  
  11.             "add-button": {  
  12.                 template: "<button>这是一个按钮{{btn}}</button>",  
  13.                 data: function () {  
  14.                     return obj;  
  15.                 }  
  16.             }  
  17.         }  
  18.     });  
  19.   
  20.     obj.btn = "456";  
  21.     var vm2 = new Vue({  
  22.         el: ‘#app2‘,  
  23.         template: "<add-button></add-button>",  
  24.         components: {  
  25.             "add-button": {  
  26.                 template: "<button>这是一个按钮{{btn}}</button>",  
  27.                 data: function () {  
  28.                     return obj;  
  29.                 }  
  30.             }  
  31.         }  
  32.     });  
  33. </script>  

 

注:

el属性用在Vue.extend()中时,也须是一个函数。

 

 

⑥is特性:

【1】按照官方教程,一些HTML元素对什么元素可以放在它之中是有限制的;

简单来说,如果我要在table标签内复用某个组件,这个组件展开后是tr标签,但是展开前不是,那么就无法正常运行(被放置在table标签内);

 

如代码(错误写法,会渲染错误):

 

[html] view plain copy
 
 技术分享技术分享
  1. <div id="app">  
  2.     <table>  
  3.         <tr>  
  4.             <td>索引</td>  
  5.             <td>ID</td>  
  6.             <td>说明</td>  
  7.         </tr>  
  8.         <thetr v-for="i in items" v-bind:id="i" :index="$index"></thetr>  
  9.     </table>  
  10. </div>  
  11. <script>  
  12.     var vm = new Vue({  
  13.         el: ‘#app‘,  
  14.         data: {  
  15.             items: [1, 2, 3, 4]  
  16.         },  
  17.         methods: {  
  18.             toknowchildren: function () {   //切换组件显示  
  19.                 console.log(this.$children);  
  20.             }  
  21.         },  
  22.         components: {  
  23.             thetr: { //第一个子组件  
  24.                 template: "<tr>" +  
  25.                 "<td>{{index}}</td>" +  
  26.                 "<td>{{id}}</td>" +  
  27.                 "<td>这里是子组件</td>" +  
  28.                 "</tr>",  
  29.                 props: [‘id‘, ‘index‘]  
  30.             }  
  31.         }  
  32.     });  
  33. </script>  

 

渲染结果如下:

[html] view plain copy
 
 技术分享技术分享
  1. <div id="app">  
  2.     <tr><td>0</td><td>1</td><td>这里是子组件</td></tr>  
  3.     <tr><td>1</td><td>2</td><td>这里是子组件</td></tr>  
  4.     <tr><td>2</td><td>3</td><td>这里是子组件</td></tr>  
  5.     <tr><td>3</td><td>4</td><td>这里是子组件</td></tr>  
  6.     <table>  
  7.         <tbody>  
  8.         <tr>  
  9.             <td>索引</td>  
  10.             <td>ID</td>  
  11.             <td>说明</td>  
  12.         </tr>  
  13.         </tbody>  
  14.     </table>  
  15. </div>  

可以明显发现,内容没有被放在table之中。

 

正确写法如下:

[html] view plain copy
 
 技术分享技术分享
  1. <div id="app">  
  2.     <button @click="toknowchildren">点击让子组件显示</button>  
  3.     <table>  
  4.         <tr>  
  5.             <td>索引</td>  
  6.             <td>ID</td>  
  7.             <td>说明</td>  
  8.         </tr>  
  9.         <tr is="thetr" v-for="i in items" v-bind:id="i" :index="$index"></tr>  
  10.     </table>  
  11. </div>  










以上是关于Vuejs——(8)Vuejs组件的定义的主要内容,如果未能解决你的问题,请参考以下文章

VueJS 自定义组件导入有时有效

VueJS注册组件错误未知自定义元素

VueJS 自定义组件

VueJS 道具在组件中未定义

VueJS 组件 ref 在所有阶段都未定义

以编程方式为 VueJS 中的动态组件绑定自定义事件