版权声明:出处http://blog.csdn.net/qq20004604

 

目录(?)[+]

 

本篇资料来于官方文档:

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的模板内容”})

这样的形式创建,例如:

  1. //定义一个组件
  2. var btn = Vue.extend({
  3. template: "<button>这是一个按钮</button>"
  4. })

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

如代码:

  1. //注册他到Vue实例上
  2. Vue.component("add-button", btn);

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

  1. <add-button></add-button>

会被

  1. <button>这是一个按钮</button>

所替代。

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

解决办法是局部注册。

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

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

③局部注册组件:

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

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

如代码:

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

注:

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

④步骤简化:

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

  1. //定义一个组件
  2. Vue.component("add-button", {
  3. template: "<button>这是一个按钮</button>"
  4. });

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

  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属性的值。

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

如代码:

  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的值是一样的(因为他们实际上还是共享了一个对象)

  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. obj.btn = "456";
  20. var vm2 = new Vue({
  21. el: '#app2',
  22. template: "<add-button></add-button>",
  23. components: {
  24. "add-button": {
  25. template: "<button>这是一个按钮{{btn}}</button>",
  26. data: function () {
  27. return obj;
  28. }
  29. }
  30. }
  31. });
  32. </script>

注:

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

⑥is特性:

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

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

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

  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>

渲染结果如下:

  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之中。

正确写法如下:

  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>

最新文章

  1. WPF 无边框透明按钮
  2. java后台如何获取String 类型 json里的字段值
  3. localstorage sessionstorage cookie 备忘
  4. Spark SQL External Data Sources JDBC简易实现
  5. 【HTML5】特性
  6. cocos2d CCArray
  7. 【转】VIM 快速注释
  8. Oracle将英文字符集数据转换成中文
  9. 【LeetCode】Sum Root to Leaf Numbers
  10. FZU 1686(重复覆盖)
  11. 标准会话对象——StandardSession
  12. 微信小程序入门二
  13. Arm11-mini6410入坑
  14. spring配置问题
  15. vue-router进阶-1-导航守卫
  16. jquery引入
  17. JavaScript编写风格指南 (一)
  18. P4824 [USACO15FEB]Censoring (Silver) 审查(银)&amp;&amp;P3121 [USACO15FEB]审查(黄金)Censoring (Gold)
  19. linux 系统优化+定时任务
  20. 【DP】【P1586】四方定理

热门文章

  1. android 开发 View _5_ Paint详解
  2. html: 仿制soundmanager2右上角面板
  3. ABAP-表中数据的横向累加
  4. tensorflow-serving-gpu 本地编译并使用
  5. 打开控制台F12弹出弹窗
  6. EF连接Mysql 表&#39;TableDetails&#39;中的列&#39;IsPrimaryKey&#39;的值为DBNull
  7. 设置同一个域名同一个源通过cdn用不同的端口访问网站设置
  8. 命令实现linux和客户端文件上传下载
  9. [leetcode]87. Scramble String字符串树形颠倒匹配
  10. [leetcode]29. Divide Two Integers两整数相除