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

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

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

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

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

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

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

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

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

html 复制代码
<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

html 复制代码
<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>

优化后的代码

html 复制代码
<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>
相关推荐
浮游本尊2 分钟前
关于考试监听切屏的三种方式
前端
GISer_Jing5 分钟前
2025年前端面试热门题目——HTML|CSS|Javascript|TS知识
前端·javascript·面试·html
m0_748255656 分钟前
Springboot基于Web的景区疫情预警系统设计与实现5170q(程序+源码+数据库+调试部署+开发环境)
前端·数据库·spring boot
m0_748237057 分钟前
web的五个Observer API
java·前端·javascript
小王爱吃月亮糖23 分钟前
QT开发【常用控件1】-Layouts & Spacers
开发语言·前端·c++·qt·visual studio
feifeiyechuan23 分钟前
【Chrome Extension】二、导航栏快速查询
前端·chrome·chromeextension
索然无味io30 分钟前
跨站请求伪造之基本介绍
前端·笔记·学习·web安全·网络安全·php
m0_7482565642 分钟前
Windows 11 Web 项目常见问题解决方案
前端·windows
LOVE️YOU43 分钟前
HTML&CSS&JavaScript&DOM 之间的关系?
前端·javascript·css·html
胡西风_foxww44 分钟前
【es6复习笔记】集合Set(13)
前端·笔记·es6·set·集合