开开森森学前端之函数式组件和JSX

前言

我们接着上篇学习完render函数,下面我们来看下Vue中的函数式组件
这也是Vue进阶中一个重要的知识点,下面我们一起来学习下吧(我也不是很熟?,有错误就请指正)

组件封装阶段

我们先封装一个简单的组件,叫做list

代码如下:

<template>
  <ul>
    <li v-for="(item, index) in list" :key="`item_${index}`"></li>
  </ul>
</template>
<script>
export default {
  name: 'List',
  props: {
    list: {
      type: Array,
      default: () => []
    }
  }
}
</script>

简单封装后,里面li里面的内容我们先不管!

我们再新建一个叫做render的组件
我们把刚刚的list组件引入到render组件里

<template>
  <div>
    <list :list="list"></list>
  </div>
</template>
<script>
import List from '../component/list'
export default {
  data () {
    return {
      list: [
        { name: lili},
        { name: Dreams}
      ]
    }
  },
  components: {
    List
  }
}
</script>

接下来,我们回到list组件里完善下li标签里面的内容

<li v-for="(item, index) in list" :key="`item_${index}`">
<span>{{ item.name }}</span>
</li>

这样我们就把刚刚render组件里传递进来的list数据给list组件使用并且渲染出了名字,这里我们用span标签包裹了,那么如果我们想更多的让用户个性化,比如用户想用什么标签包裹就用什么标签包裹,怎么做呢?

那么这里我们就可以使用render函数,然后让用户通过render函数自己去定义,来看下怎么写

这里我们回到render组件,把render函数通过v-bind传递到list里去

render组件的代码:
<template>
  <div>
    <list :list="list" :render="renderFunc"></list>
  </div>
</template>
<script>
import List from '../component/list'
export default {
  data () {
    return {
      list: [
        { name: lili},
        { name: Dreams}
      ]
    }
  },
  components: {
    List
  },
  methods:{
      renderFunc(h){
         return h('i',{
            style:{
                color:'pink'
            }
         },'???') 
      }
  }
}
</script>

上面我们想使用i标签包裹,且字体颜色为粉红色,但这里我们打问号的地方他的值是多少呢?怎么来呢,因为list中他是通过v-for循环的,
那么我们想知道当前循环的这个值是多少,那么这里我们就要用到一个组件叫做:函数式组件

函数式组件开始

那么接着上面,我们先创建一个render-dom.js
代码如下

export default {
  functional: true,//必须这样写才是一个函数式组件
  props: {
    name: String,
    renderFunc: Function
  },
  render: (h, ctx) => {
    //???
  }
}

那么有人会问了,函数式组件有什么作用?

一般而言,我们只给它传入一些数据,它不做任何响应式的操作,不监听传递给他的状态,它也没有生命周期,它只是一个接收参数的函数,当设置functionaltrue时,它是一个没有状态的组件,但是当你把他引入到其他vue组件中去用的时候,vue会把它做相关处理,上面我们写了一个render函数,vue会用render函数里的逻辑,把里面返回的节点做渲染!

接着我们回到list组件,在props里接收父组件(render组件)传递过来的render函数

props处新增代码:

render: {
      type: Function,
      default: () => {}
}

接下来,我们把函数式组件再完善下

把刚刚render方法里的逻辑完善下就是

render: (h, ctx) => {
   return ctx.props.renderFunc(h,ctx.props.name)
}

返回的render其实就是用户传进来的那个render
ctx他其实就指代当前的上下文,那么我们就可以用ctx获取到属性里的renderFunc

接下来继续到list组件里使用renderDom函数式组件

模板中新增:
<li v-for="(item, index) in list" :key="`item_${index}`">
    <span v-if="!render">{{ item.number }}</span>
    <render-dom v-else :render-func="render" :name="item.name"></render-dom> 
</li>
script引入:
import RenderDom from '../component/render-dom'
components: { //注册组件
    RenderDom 
},

这里逻辑就是如果用户传入了render,我们就是用传入的render渲染,否则默认span

接下来我们回到render组件里把刚刚的render函数完善下:

renderFunc(h,name){
     return h('i',{
        style:{
            color:'pink'
        }
     },name) 
}

那么这里的name其实就是函数式组件返回的name,所以这里我们直接使用name

那么这样我们就完成了可以让用户自己定义这个文字该怎么样展现!render配合函数式组件就是这么灵活!

那么大家应该也发现了,render加函数式组件其实非常繁琐!那么有没有什么东西可以简化呢?
其实是有的,那就是JSX!

JSX

JSX是React最先提出的,那么后来Vue也做了相关支持!他的本质其实就是在js中写html标签和一些特定的语法,最终呢他会把jsx转换成render函数去渲染。

接下来我们回到render组件里把刚刚的render函数修改下:

之前template是这样的:
<list :list="list" :render="renderFunc"></list>
之前js是这样的:
renderFunc(h,name){
     return h('i',{
        style:{
            color:'pink'
        }
     },name) 
}
修改之后的template:
<list :list="list" :render="renderFunc" :style="{color:'red'}"></list>
修改之后的js:
renderFunc(h,name){
     return (
        <i style={{color:'red'}}>{name}</i>
     )
}

这里要注意了,在jsx中,变量要用花括号包起来!里面是一个对象,然后又有一个花括号.因为name也是变量,所以也需要使用花括号包起来!

这样就完成了和刚刚render函数一样的功能!

如果我们想给i标签绑定一个事件,我们应该怎么绑定?

methods:{
    renderFunc(h,name){
     return (
        <i on-click={this.handleClick} style={{color:'red'}}>{name}</i>
     )
    },
    handleClick(event){
        console.log(event)
    }
}

这里还要注意,绑定事件必须是on-开头

那么普通的渲染标签我们会了,那么如果渲染一个组件呢?

渲染组件

那么这里假设我们这里有一个组件CountTo(看过上上篇文章的小伙伴应该知道)组件

我们把他引入到render组件(看过上篇文章的小伙伴应该知道这个组件)中.再写一遍!

我们这里需要修改之前的一些代码:

render组件中修改后的:(函数式组件和list里需要做一部分修改,我相信大家可以的,就不罗列相关代码了)

<template>
  <div>
    <list :list="list" :style="{color: 'red'}"></list>
  </div>
</template>
<script>
import List from '_c/list'
import CountTo from '_c/count-to'
export default {
  data () {
    return {
      list: [
        { number: 100 },
        { number: 45 }
      ]
    }
  },
  components: {
    List
  },
  methods: {
    renderFunc (h, number) {
      return (
        <CountTo nativeOn-click={this.handleClick} on-on-animation-end={this.handleEnd} endVal={number} style={{color: 'pink'}}></CountTo>
      )
    },
    handleClick (event) {
      // console.log(event)
    },
    handleEnd () {
      // console.log('end!')
    }
  }
}
</script>

我们如果是在render或者jsx里写的组件是不需要进行注册的!

那么这里我们也使用了JSX做了相同的事情。其实这里还可以使用作用域插槽做同样的事情,这篇文章我们主要讲JSX和函数式组件,所以就不再举插槽相关的代码哦,有兴趣的可以下去自己试试!

总结

本篇文章我们一起学习了render+函数式组件,最终我们完成了之前的需求,但过程还是比较繁琐的,结尾我们也提到了JSX的解决方案, 感兴趣的小伙伴可以关注我,大家一起学习vue中比较难理解的知识点哦!

各位大佬,如果发现文中的错误,请指正,我会及时修改!

感谢大佬们能在百忙中能阅读完这篇文章!

再次总结

相信大家看完这篇文章后再去看vue文档就不会觉得那么吃力了!大家共勉!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值