前端组件高级封装技巧--纯干货

本文讨论了前端开发中如何避免使用配置文件生成页面带来的局限性和维护困难,提倡基于组件的开发方式,特别是利用render函数进行灵活布局。作者指出,虽然配置方式起初看起来节省时间,但长期来看可能导致维护难题,而组件化能更好地遵循设计模式和开闭原则。
摘要由CSDN通过智能技术生成

对于前端的小伙伴来说,最常见的工作就是写后台管理系统的页面,而后台管理系统最多的操作就是CRUD了,类似下面的,一个搜索框,一个表格,一个分页,然后点击新增编辑有个弹框

 当你写过一段时间的CRUD,就会发现其中很多代码都是重复的,变动的不过是接口地址,搜索条件,表格列的属性等。这时你肯定想,把这些变动的独立出来,写成一个配置json对象,页面根据这个配置对象生成,不是省事很多。

如果你初出茅庐,我会说小伙子不错,有想法,如果你已经工作过一段时间还有这种想法,我只能无奈的说,图样图森破。

基于配置生成页面有个最大的缺陷是不够灵活,只要需求有点变动就要修改生成方法,比如前提表格都是文字,那你配置文件里面只要配置字段名就行了,但如果表格里面要显示图片,麻烦就来了,你要在配置文件里面加个类型,然后改动生成逻辑,如果如上图所示还要再表格里加个按钮,我估计你头都要大了。按照设计模式来说,这种方式违反了开闭原则。

我见过好几个使用了基于配置的项目,最后无一例外的陷入了维护地狱,只能让原来搭建框架的人来维护,新人很难上手,要改动一点东西都要改好久。这是一个大坑,不要去踩!

基于配置生成组件只适用于局部组件,并且是后期基本不会去改动的组件。

最好的封装方式还是基于组件的方式,特别是render函数,用好了特别方便。

下面分享一个小案例,搜索框一般一行4个,使用el-row, el-col来布局,每4个搜索条件套一层el-row,再在每个搜索条件上面套一层el-col,写起来就很繁琐,类似下面

<el-row>
    <el-col>
        <ui-text v-model="searchParams.q_username_like" label="用户名"/>
    </el-col>
    <el-col>
        <ui-select v-model="searchParams.q_enabled_eq" label="状态" :items="valueCommon.enabledType"/>
     </el-col>
     <el-col>
        <div>
           <ui-button operateType="search" @click="handleSearch"></ui-button>
           <ui-button operateType="reset" @click="handleReset"></ui-button>
         </div>
      </el-col>
</el-row>

有没有办法可以自动套上el-row和el-col呢?这就要使用render函数了

我们封装一个组件叫ui-row

<script lang="ts">
// @ts-nocheck

export default {
  name: "ui-row",
  render() {
    let vRows = [] // 行节点
    let vCols = [] // 列节点
    let slots = this.$slots.default({})
    let countSpan = 0
    let colIndex = 0
    vCols[colIndex] = []
    for (let n = 0; n < slots.length; n++) {
      let ctl = slots[n]
      let span = this.getSpan(ctl)
      vCols[colIndex].push(h(ElCol, {span}, {
        default: () => {
          return ctl
        },
      }))
      let nextSpan = 0;
      if (n + 1 < slots.length) {
        nextSpan = this.getSpan(slots[n + 1])
      }
      countSpan += span
      // 生成行
      if (nextSpan == 0 || countSpan == 24 || countSpan + nextSpan > 24) {
        vRows.push(h(ElRow, null, {
          default: (index => {
            return () => {
              return vCols[index]
            }
          })(colIndex),
        }))
        countSpan = 0
        vCols[++colIndex] = []
      }
    }
    return h('div', null, vRows)
  },
  methods: {
    getSpan(ctl) {
      let span = 6
      if (ctl.props && ctl.props.span) {
        span = ctl.props.span
      }
      return span
    }
  }
}
</script>

<style scoped>

</style>

 优化后的代码

<ui-row>
    <ui-text v-model="searchParams.q_username_like" label="用户名"/>
    <ui-select v-model="searchParams.q_enabled_eq" label="状态" :items="valueCommon.enabledType"/>
    <div>
      <ui-button operateType="search" @click="handleSearch"></ui-button>
      <ui-button operateType="reset" @click="handleReset"></ui-button>
     </div>
</ui-row>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值