了解VUE的render函数的使用

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

创新互联建站-专业网站定制、快速模板网站建设、高性价比古蔺网站开发、企业建站全套包干低至880元,成熟完善的模板库,直接使用。一站式古蔺网站制作公司更省心,省钱,快速模板网站建设找我们,业务覆盖古蔺地区。费用合理售后完善,十载实体公司更值得信赖。

Hello world!

我们尝试使用render函数实现上面的例子,注意使用render函数,template 选项将被忽略。 createElement接收3个参数:

第一个参数可以是HTML标签名,组件或者函数都可以;此参数是必须的;

第二个为数据对象{Object}(可选);

第三个为子节点{String | Array}(可选),多个子节点[createElement(tag1),createElement(tag2)]。

Hello world! span

header slotspan

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: '
', // 将被忽略 props: { level: { type: Number, required: true } }, methods: { clickHandler: function() { console.log('clickHandler') } } }) new Vue({ el:"#div1" })

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

Hello world!

// 递归函数获得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指向同一引用是错误的,如果要重复创建多个相同元素/组件,可以使用工厂函数实现

Hello world!
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"
})

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持创新互联。


网站栏目:了解VUE的render函数的使用
文章URL:http://pcwzsj.com/article/jecejs.html