zoukankan      html  css  js  c++  java
  • Vue学习笔记进阶篇——函数化组件

    本文为转载,原文:Vue学习笔记进阶篇——函数化组件

    介绍

    之前创建的锚点标题组件是比较简单,没有管理或者监听任何传递给他的状态,也没有生命周期方法。它只是一个接收参数的函数。
    在这个例子中,我们标记组件为 functional, 这意味它是无状态(没有 data),无实例(没有 this 上下文)。
    一个 函数化组件 就像这样:

    Vue.component('my-component', {
      functional: true,
      // 为了弥补缺少的实例
      // 提供第二个参数作为上下文
      render: function (createElement, context) {
        // ...
      },
      // Props 可选
      props: {
        // ...
      }
    })

    组件需要的一切都是通过上下文传递,包括:

    1. props: 提供props 的对象
    2. children: VNode 子节点的数组
      slots: slots 对象
      data: 传递给组件的 data 对象
      parent: 对父组件的引用
      listeners: (2.3.0+) 一个包含了组件上所注册的 v-on 侦听器的对象。这只是一个指向 data.on 的别名。
      injections: (2.3.0+) 如果使用了 inject 选项, 则该对象包含了应当被注入的属性。

    在添加 functional: true 之后,锚点标题组件的 render 函数之间简单更新增加context参数,this.$slots.default 更新为 context.children,之后this.level 更新为 context.props.level
    因为函数化组件只是一个函数,所以渲染开销也低很多。另外,这也意味着函数化组件不会出现在 VueJS Chrome 开发者工具的组件树里。
    在作为包装组件时它们也同样非常有用,比如,当你需要做这些时:
    程序化地在多个组件中选择一个
    在将 childrenpropsdata 传递给子组件之前操作它们。
    下面是一个依赖传入 props 的值的smart-list组件例子,它能代表更多具体的组件:

    var EmptyList = { /* ... */ }
    var TableList = { /* ... */ }
    var OrderedList = { /* ... */ }
    var UnorderedList = { /* ... */ }
    Vue.component('smart-list', {
      functional: true,
      render: function (createElement, context) {
        function appropriateListComponent () {
          var items = context.props.items
          if (items.length === 0)           return EmptyList
          if (typeof items[0] === 'object') return TableList
          if (context.props.isOrdered)      return OrderedList
          return UnorderedList
        }
        return createElement(
          appropriateListComponent(),
          context.data,
          context.children
        )
      },
      props: {
        items: {
          type: Array,
          required: true
        },
        isOrdered: Boolean
      }
    })

    slots()和children对比

    你可能想知道为什么同时需要 slots()childrenslots().default 不是和 children 类似的吗?在一些场景中,是这样,但是如果是函数式组件和下面这样的 children 呢?

    <my-functional-component>
      <p slot="foo">
        first
      </p>
      <p>second</p>
    </my-functional-component>

    对于这个组件,children 会给你两个段落标签,而 slots().default 只会传递第二个匿名段落标签,slots().foo 会传递第一个具名段落标签。同时拥有 children 和 slots() ,因此你可以选择让组件通过 slot() 系统分发或者简单的通过 children 接收,让其他组件去处理。

    示例

    渐进过渡

    之前的Vue学习笔记进阶篇——列表过渡及其他中可复用的过渡提到用函数组件实现合适,下面就用函数化组件来实现那个渐进过渡

    <div id="app5">
        <input v-model="query">
        <my-transition :query="query" :list="list">
            <li v-for="(item, index) in computedList"
                :key="item.msg"
                :data-index="index">
                {{item.msg}}
            </li>
        </my-transition>
    </div>
        Vue.component('my-transition', {
            functional:true,
            render:function (h, ctx) {
                var data = {
                    props:{
                        tag:'ul',
                        css:false
                    },
                    on:{
                        beforeEnter:function (el) {
                            el.style.opacity = 0
                            el.style.height = 0
                        },
                        enter:function (el, done) {
                            var delay = el.dataset.index * 150
                            setTimeout(function () {
                                Velocity(el, {opacity:1, height:'1.6em'},{complete:done})
                            }, delay)
                        },
                        leave:function (el, done) {
                            var delay = el.dataset.index * 150
                            setTimeout(function () {
                                Velocity(el, {opacity:0, height:0}, {complete:done})
                            }, delay)
                        }
                    }
                }
                return h('transition-group', data, ctx.children)
            },
            props:['query', 'list']
        })
    
        var app5 = new Vue({
            el:'#app5',
            data:{
                query:'',
                list:[
                    {msg:'Bruce Lee'},
                    {msg:'Jackie Chan'},
                    {msg:'Chuck Norris'},
                    {msg:'Jet Li'},
                    {msg:'Kung Furry'},
                    {msg:'Chain Zhang'},
                    {msg:'Iris Zhao'},
                ]
            },
            computed:{
                computedList:function () {
                    var vm = this
                    return this.list.filter(function (item) {
                        return item.msg.toLowerCase().indexOf(vm.query.toLowerCase()) !== -1
                    })
                }
            },
        })

    运行结果:

    本文为原创,转载请注明出处。
    上一节:Vue学习笔记进阶篇——Render函数
    返回目录
    下一节:Vue学习笔记进阶篇——vue-cli安装及介绍

  • 相关阅读:
    Java后台实现方法
    解决横屏下锁屏,再次解锁界面显示异常-一屏下显示反复两个界面
    java平台利用jsoup开发包,抓取优酷视频播放地址与图片地址等信息。
    用C++实现一个Log系统
    LeetCode 3_Longest Substring Without Repeating Characters
    跟着实例学习设计模式(7)-原型模式prototype(创建型)
    XStream 数组(List)输出结构
    【Cocos2d-x 3.0 基础系列一】 各类回调函数写法汇总
    机房收费系统验收小结(一)
    动态隐藏/显示系统状态栏
  • 原文地址:https://www.cnblogs.com/ChainZhang/p/7284822.html
Copyright © 2011-2022 走看看