了解VUE的render函数

Vue 推荐在绝大多数情况下使用 template 来创建你的 HTML。然而在一些场景中,你真的需要 JavaScript 的完全编程的能力,这就是 render 函数,它比 template 更接近编译器。 在 HTML 层, 我们决定这样定义组件接口:通过传入不同的level 1-6 生成h1-h6标签,和使用slot生成内容

<div id="div1">
    <child :level="1">Hello world!</child>
</div>
<script type="text/x-template" id="child-template">
  <h1 v-if="level === 1">
    <slot></slot>
  </h1>
  <h2 v-if="level === 2">
    <slot></slot>
  </h2>
  <h3 v-if="level === 3">
    <slot></slot>
  </h3>
  <h4 v-if="level === 4">
    <slot></slot>
  </h4>
  <h5 v-if="level === 5">
    <slot></slot>
  </h5>
  <h6 v-if="level === 6">
    <slot></slot>
  </h6>
 </script>

<script type="text/javascript">
    /**
     * 全局注册child组件,注意template如果值以 # 开始,则它用作选项符,将使用匹配元素的 innerHTML 作为模板。常用的技巧是用 <script type="x-template"> 包含模板,这样的好处是html不会渲染里面的内容
     *  这里使用template不是最好的选择,
     * 一、代码冗长 
     * 二、在不同的标题插入内容需要重复使用slot 
     * 三、由于组件必须有根元素,所以标题和内容被包裹在一个无用的div中,比如<div><h1>hello world</h1></div>
     */

    Vue.component('child', {
      template: '#child-template',
      props: {
        level: {
          type: Number,
          required: true
        }
      },
      data: function() {
        return {
          a: 1
        }
      }
    })

    new Vue({
        el:"#div1"
    })
  </script>

我们尝试使用render函数实现上面的例子,注意使用render函数,template 选项将被忽略。 createElement接收3个参数:
第一个参数可以是HTML标签名,组件或者函数都可以;此参数是必须的;
第二个为数据对象{Object}(可选);
第三个为子节点{String | Array}(可选),多个子节点[createElement(tag1),createElement(tag2)]。

<div id="div1">
   <child :level="1">
     Hello world!
   </child>
   <child :level="2">
     <!-- 将不会被显示 -->
     <span slot="footer">span</span>
     <p slot="header">header slot<span>span</span></p>
   </child>
 </div>

Vue.component('child', {
      render: function(createElement) {
        console.log(this.$slots);
        return createElement(
          'h'+ this.level, // tagName标签名称
          {
            // 为每个h标签设置class
            'class': {
              foo: true,
              bar: false
            },
            // 最终被渲染为内联样式
            style: {
              color: 'red',
              fontSize: '14px'
            },
            // 其他的html属性
            attrs: {
              id: 'foo',
              'data-id': 'bar'
            },
            // DOM属性
            domProps: {
              // innerHTML: 'from domProps',
            },
            // 事件监听器基于 "on"
            // 所以不再支持如 v-on:keyup.enter 修饰器
            on: {
              click: this.clickHandler
            },
            // ...
          },
          // 你可以从this.$slots获取VNodes列表中的静态内容
          // $slots.default用来访问组件的不具名slot
          // 当你可能需要具名slot的时候需要指定slot的name, this.$slots.header
          [this.$slots.default]
        )
      },
      template: '<div v-if="level===1"><slot></slot></div>', // 将被忽略
      props: {
        level: {
          type: Number,
          required: true
        }
      },
      methods: {
        clickHandler: function() {
          console.log('clickHandler')
        }
      }
    })

    new Vue({
        el:"#div1"
    })

我们现在可以完成这样的组件

<h1>
     <a name="hello-world" href="#hello-world">
        Hello world!
     </a>
</h1>

// 递归函数获得helloworld文本
    function getChildrenTextContent(child) {
        return child.map(function(node) {
            return node.children? getChildrenTextContent(node.children) : node.text
        }).join('')
    }
    Vue.component('child',{
        render: function(createElement) {
            var hello_world = getChildrenTextContent(this.$slots.default)
                              .toLowerCase()
                              .replace(/\W+/g,'-')
                              .replace(/^\-|\-$/g,'');
            return createElement(
                'h'+ this.level,
                {},
                [ // 创建一个a标签,设置属性,并设置a标签的子节点
                    createElement('a',{
                        attrs: {
                            name: hello_world,
                            href: '#' + hello_world
                        }
                    },this.$slots.default)
                ]
            )
        },
        props: {
            level: {
                type: Number,
                required: true
            }
        }
    })
    new Vue({
        el:"#div1"
    })

注意VNode的唯一性,这里两个VNode指向同一引用是错误的,如果要重复创建多个相同元素/组件,可以使用工厂函数实现

<div id="div1">
    <child :level="1">
      Hello world!
    </child>
</div>

Vue.component('child',{
    // render: function(createElement) {
    //  var myParagraphVNode  = createElement('p','hello')
    //  return createElement('div',
    //      [myParagraphVNode, myParagraphVNode]
    //  )
    // },
    render: function(createElement) {
        return createElement('div',
            Array.apply(null, {length:20}).map(function() {
                return createElement('p','hello')
            })
        )
    },
    props: {
        level: {
            type: Number,
            required: true
        }
    }
})
new Vue({
    el:"#div1"
})

使用javascript代替模板功能,某些api要自己实现
①使用if/else代替v-if
②使用map代替v-for

Vue.component('child',{
    render: function(createElement) {
        if (this.lists.length) {
            return createElement('ul',this.lists.map(function() {
                return createElement('li','hi')
            }))
        } else {
            return createElement('p','no lists')
        }
    },
    props: {
        level: {
            type: Number,
            required: true
        }
    },
    data: function() {
        return {
            lists: [1,2,3]
        }
    }
})

// render函数中没有与v-model相应的api - 你必须自己来实现相应的逻辑:
Vue.component('child-msg',{
    render: function(createElement) {
        var self = this;
        return createElement('div', [
                createElement('input',{
                    'on': {
                        input: function(event) {
                            self.value = event.target.value;
                        }
                    }
                }),createElement('p',self.value)
            ])
    },
    props: {
        level: {
            type: Number,
            required: true
        }
    },
    data: function() {
        return {
            value: ''
        }
    }
})
new Vue({
    el:"#div1"
})
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Vuerender 函数Vue.js 中一个非常重要的概念。它是用来描述组件如何渲染的函数。在 Vue 中,我们可以使用模板语法来编写组件的渲染逻辑,也可以使用 render 函数来编写。 使用 render 函数,你可以直接返回一个虚拟 DOM 对象,描述组件的结构和行为。这个函数接收一个参数,通常命名为 h(代表 createElement),它是一个用于创建 VNode(虚拟节点)的函数。 在 render 函数中,你可以通过调用 h 函数来创建 VNode,然后返回一个或多个 VNode 组成的树形结构。VNode 是对真实 DOM 的一种抽象表示,它包含了节点的标签名、属性、子节点等信息。 以下是一个简单的例子,演示了如何使用 render 函数创建一个简单的 Vue 组件: ```javascript // 定义一个组件 const MyComponent = { render(h) { return h('div', { class: 'container' }, [ h('h1', {}, 'Hello, Vue!'), h('p', {}, 'This is a Vue component.') ]) } } // 使用组件 new Vue({ el: '#app', render: h => h(MyComponent) }) ``` 这个例子中,我们定义了一个名为 MyComponent 的组件,在 render 函数中返回了一个包含 h1 和 p 标签的 div 元素。在 new Vue 的时候,我们将 render 函数指定为 MyComponent,从而将组件渲染到具有 id 为 app 的元素中。 通过使用 render 函数,你可以更灵活地控制组件的渲染逻辑,实现更高级的功能。它提供了一种编程式的方式来构建组件,而不仅仅局限于模板语法。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值