Skip to content

页码列表组件封装思路

作者:Yuan Tang
更新于:5 个月前
字数统计:1.5k 字
阅读时长:6 分钟

前言

项目中封装业务组件两大原则:

  1. 不结合具体业务逻辑

    • 组件只作为数据的容器,数据统一父组件传入。
    • 只编写 UI 逻辑,具体的数据操作这样的业务逻辑,触发父组件的监听交给父组件处理
  2. 尽量的提供简便

    在不结合具体业务逻辑的前提下,让父组件使用组件尽可能方便,原则是观察大部分页面的设计,能方便满足大多数页面的需求。少数页面有差距,也有扩展方案。

组件搭建

本案例以 element-plus 组件库中的表格和分页器组件为基础,做初步搭建。

vue
<script setup>
import { ref } from 'vue'

const { tableData, total } = defineProps({
  tableData: {
    type: Array
  },
  total: {
    type: Number
  },
})
const emit = defineEmits(['changePage', 'changeSize'])

// 页码参数对象
const pageParams = ref({
  page: 1,
  pageSize: 10
})

// 页码切换让父组件去做处理
function changePage(e) {
  pageParams.value.page = val
  emit('changePage', pageParams.value)
}
// 单页码数据数量切换让父组件去做处理
function changeSize(e) {
  pageParams.value.pageSize = val
  emit('changeSize', pageParams.value)
}
</script>

<template>
  <el-table>
    <!-- 子组件无法控制渲染什么内容,流出插槽让父组件传入 -->
    <slot>
      <el-table-column v-for="(item, index) in tableColumn" :key="index" :prop="item.prop" :label="item.label" />
    </slot>
  </el-table>

  <el-pagination :total="total" :layout="prev, pager, next, sizes" @current-change="changePage" @size-change="changeSize" />
</template>

现在已经尽可能的简便来搭建组件了,但是父组件用起来还是很麻烦,和不封装直接用没什么区别。

因此现在要观察大部分的页面和接口,加入部分业务逻辑,使其满足大部分页面。但也能让部分极端情况满足扩展自定义。

翻页逻辑

子组件修改页码组件让父组件更方便:

  1. 单页码数据数量切换给定一个默认值,但是也要支持父组件扩展自定义传入使用
  2. 后端接口一般要保持统一性,页码一般都是一样的单词,不会突然改变。因此父组件无需监听两个事件,使用一个事件即可
vue
<script setup>
import { ref } from 'vue'

const { tableData, total, pageSizes } = defineProps({
  tableData: {
    type: Array
  },
  total: {
    type: Number
  },
  pageSizes: {
    type: Array,
    default: () => [10, 20, 30]
  }
})
const emit = defineEmits(['getPage'])

// 页码参数对象
const pageParams = ref({
  page: 1,
  pageSize: 10
})

// 分页器触发切换让父组件去做处理
function changePage(e) {
  pageParams.value.page = val
  emit('getPage', pageParams.value)
}
</script>

<template>
  <el-table>
    <!-- 子组件无法控制渲染什么内容,流出插槽让父组件传入 -->
    <slot />
  </el-table>

  <el-pagination :total="total" :layout="prev, pager, next, sizes" :page-size="pageSize[0]" :page-sizes="pageSizes" @current-change="getPage" @size-change="getPage" />
</template>

父组件传入对应的参数:

vue
<script setup>
import { onMounted } from 'vue'
import axios from 'axios'
import PageList from './components/pageList.vue'

async function initList(e) {
  axios.get('http://xxxxx', {
    params: { ...e }
  }).then((res) => {
    console.log(res)
  })
}
function getPage(e) {
  initList(e)
}

onMounted(() => initList({
  page: 1,
  pageSize: 10
}))
</script>

<template>
  <PageList :table-data="tableData" :total="400" :page-sizes="[50, 100, 150]" @get-page="getPage" />
</template>

表格子行

父组件要使用这个封装的组件时,需要通过插槽设置表格行,代码如下所示:

vue
<template>
  <page-list :table-data="tableData" @get-page="getPage">
    <el-table-column prop="name" label="名称" />
    <el-table-column prop="price" label="价格" />
    <el-table-column label="操作">
      <el-button>删除</el-button>
      <el-button>编辑</el-button>
    </el-table-column>
  </page-list>
</template>

但是这么写并不是很方便,需要为插槽提供大量的代码。

如果需要简化,就先观察大部分页面表格的使用。大多数都是表格渲染数组数据,然后有需要的话加上操作。

因此可以不用再传插槽了,直接通过传递一个数组变量,里面为多个对象,每个对象提供 propname 属性。子组件接收遍历渲染即可。代码如下:

父组件:

vue
<script setup>
const tableColumn = [
  {
    prop: 'name',
    name: '名称'
  },
  {
    prop: 'price',
    name: '价格'
  },
]
</script>

<template>
  <page-list :table-data="tableData" :table-column="tableColumn" @get-page="getPage" />
</template>

子组件在默认插槽中获取并循环遍历。

注意

这里用默认插槽是因为还需要考虑部分极端情况,要提供其他具名插槽给父组件使用

vue
<script setup>
import { ref } from 'vue'

const { tableColumn } = defineProps({
  // ...
  tableColumn: {
    type: Array,
    default: () => [10, 20, 30]
  }
})

// ...
</script>

<template>
  <el-table>
    <!-- 子组件无法控制渲染什么内容,流出插槽让父组件传入 -->
    <slot>
      <el-table-column v-for="(item, index) in tableColumn" :key="index" :prop="item.prop" :label="item.label" />
      <!-- 假设每个页面都要有删除和编辑按钮(更精细点应该提供变量让父组件设置需不需要 -->
      <el-table-column label="操作">
        <el-button>删除</el-button>
        <el-button>编辑</el-button>
      </el-table-column>
    </slot>
  </el-table>

  <!-- ... -->
</template>

在极端情况下,父组件想要也使用自己的按钮,此时需要支持父组件传入按钮数组,子组件循环遍历。

父组件代码如下所示:

vue
<script setup>
const buttonList = [
  {
    emit: 'stop',
    name: '禁用',
    type: 'danger'
  }
]

// 子组件点击禁用按钮
function stopFn(row) {
  console.log(row)
}

// ...
</script>

<template>
  <page-list :button-list="buttonList" :table-data="tableData" :table-column="tableColumn" @get-page="getPage" @stop="stopFn" />
</template>

子组件做如下操作:

  1. 获取数组循环遍历渲染
  2. 为按钮绑定点击事件,传入该行的数据以及按钮的自定义事件

代码如下:

vue
<script setup>
import { ref } from 'vue'

const { tableColumn, buttonList } = defineProps({
  // ...
  tableColumn: {
    type: Array,
    default: () => [10, 20, 30]
  },
  buttonList: {
    type: Array,
    default: () => []
  }
})

// 点击操作模块的按钮
function handleClickFn(emitName, row) {
  emit(emitName, row)
}

// ...
</script>

<template>
  <el-table>
    <!-- 子组件无法控制渲染什么内容,流出插槽让父组件传入 -->
    <slot>
      <el-table-column v-for="(item, index) in tableColumn" :key="index" :prop="item.prop" :label="item.label" />
      <!-- 假设每个页面都要有删除和编辑按钮(更精细点应该提供变量让父组件设置需不需要 -->
      <el-table-column label="操作">
        <template #default="scope">
          <el-button>删除</el-button>
          <el-button>编辑</el-button>
          <el-button v-for="item in buttonList" :key="item.emit" @click="handleClickFn(item.emit, scope)">
            {{ item.name }}
          </el-button>
        </template>
      </el-table-column>
    </slot>
  </el-table>

  <!-- ... -->
</template>

总结

  1. 组件只作为数据容器,数据父组件传入
  2. 只做最简单的逻辑,深层次逻辑由父组件处理
  3. 尽可能让父组件使用简便,但也要考虑特殊情况

Contributors

Yuan Tang