基于ElementUI二次封装el-table与el-pagination分页组件[实际项目使用]

效果:

二次封装elementUI表格及其分页组件,并在项目中使用

二次封装el-table组件

<template>
  <div>
    <!-- 
      showHeader:是否显示头部
      size:表格的大小
      height:表格的高度
      isStripe:表格是否为斑马纹类型
      tableData:表格数据源
      isBorder:是否表格边框
      handleSelectionChange:行选中,多选内容发生变化回调函数
      fit:列的宽度是否自己撑开
      isRowBgc:如果需要设定行背景,需要指定rowClassName
      rowClassName:{
          bgc:"pink", //背景颜色
          attrName:"xs", //需要根据是否背景的属性
        },
      isMutiSelect:是否需要多选
      isRadio:是否单选
      isCondition:表头是否有赛选条件框
     -->
    <el-table
      :show-header="table.showHeader"
      :size="table.size"
      :height="table.height"
      :stripe="table.isStripe"
      :data="table.tableData"
      :border="table.isBorder"
      :row-key="getRowKeys"
      @sort-change="handleSort"
      @select="handleSelect"
      @select-all="handleSelectAll"
      @selection-change="handleSelectionChange"
      style="width: 100%"
      highlight-current-row
      :row-style="table.isRowBgc?tableRowClassName:{}"
    >
    <el-table-column v-if="table.isRadio" align="center" width="55" label="选择">
        <template slot-scope="scope">
          <!-- 可以手动的修改label的值,从而控制选择哪一项 -->
          <el-radio @input="singleElection(scope.row)" class="radio" v-model="templateSelection" :label="scope.row.id">&nbsp;</el-radio>
        </template>
      </el-table-column>
      <el-table-column
        v-if="table.isMutiSelect"
        type="selection"
        style="width: 60px"
      >
      </el-table-column>
      
      <template v-for="(col, key) in table.columns">
        <el-table-column
          v-if="col.type === 'slot'"
          :key="key"
          :prop="col.prop"
          :label="col.label"
          :width="col.width"
          :align="col.align"
          :header-align="col.headerAlign"
        >
          <template slot-scope="scope">
            <slot :name="col.slot_name" :row="scope.row"></slot>
          </template>
          <el-table-column :align="col.align" v-if="col.isCondition" :label="col.label" :prop="col.prop">
            <template
              slot="header"
              slot-scope="/* eslint-disable vue/no-unused-vars */ scope"
            >
            <slot :name="col.slot_siff_name"></slot>
            </template>
          </el-table-column>
        </el-table-column>
        <el-table-column
          v-else
          :key="key"
          :fixed="col.isFixed"
          v-show="col.hide"
          :prop="col.prop"
          :label="col.label"
          :width="col.width"
          :align="col.align"
          :header-align="col.headerAlign"
        >
          <el-table-column  v-if="col.isCondition" :align="col.align" :label="col.label" :prop="col.prop">
            <template
              slot="header"
              slot-scope="/* eslint-disable vue/no-unused-vars */ scope"
            >
            <slot :name="col.slot_siff_name"></slot>
            </template>
          </el-table-column>
        </el-table-column>
      </template>
    </el-table>
  </div>
</template>
<script>
export default {
  name: "hsk-table",
  props: {
    data: Object,
  },
  data() {
    return {
      templateSelection: "",  //当前选择行的id 
      checkList: [],//   当前选择的行的数据
      table: {
        showHeader:true, //是否显示表头
        fit:"true", //列的宽度是否自动撑开
        size:"small", //表格大小类型 medium / small / mini
        height:"500",  //高度
        isRowBgc:true,  //是否开启根据行某个属性更改背景
        rowClassName: null, //行背景及其根据哪一个属性进行判断是否背景
        columns: [], //列数据
        tableData: [], //表数据
        isMutiSelect: false, //是否行多选
        isRadio:false, //是否单选
        isBorder: true, //是否边框
        isStripe: false, //是否斑马纹
      },
    };
  },
  watch: {
    data: {
      handler(newVal) {
        this.init(newVal);
      },
      immediate: true,
      deep: true,
    },
  },
  methods: {
    tableRowClassName(e) {
      if(e.row[this.table.rowClassName.attrName]){
        return {
          background:this.table.rowClassName.bgc
        }
      }else{
        return {}
      }
    },
    async init(val) {
      for (let key in val) {
        if (Object.keys(this.table).includes(key)) {
          this.table[key] = val[key];
        }
      }
    },
    getRowKeys(row) {
      return row.id;
    },
    handleSort(column, prop, order) {
      this.$emit("tableSort", column, prop, order);
    },
    handleSelectionChange(val) {
      this.$emit("selectChange", val);
    },
    //多行选择
    handleSelectAll(val) {
      this.$emit("selectAll", val);
    },
    //多选
    handleSelect(val, row) {
      this.$emit("select", val, row);
    },
    //单选
    singleElection(row){
      this.$emit("radioSelectChange", row);
    }
  },
};
</script>

<style scoped>
.b {
  color: pink;
}
</style>

属性参数

属性说明
showHeader是否显示头部
height表格的高度
size表格大小
isStripe表格是否为斑马纹类型
tableData表格数据源
isBorder是否表格边框
handleSelectionChange行选中,多选内容发生变化回调函数
fit列的宽度是否自己撑开
isRowBgc如果需要设定行背景,需要指定rowClassName
rowClassName例子: { bgc:“pink”, //背景颜色 attrName:“xs”, //需要根据是否背景的属性 },
isMutiSelect是否需要多选
isRadio是否单选
isCondition表头是否有赛选条件框

父组件使用实例

<template>
  <div>
    <HskTable :data="table" @select="tableSlect" @selectChange="selectChange">
      <template v-slot:tag_slot="scope">
        <el-link type="primary">{{ scope.row.status }}</el-link>
      </template>
      <template v-slot:controls_slot="scope">
        <el-button type="text" @click="viewClick(scope.row)" size="small"
          >查看</el-button
        >
        <el-button type="text" size="small">编辑</el-button>
      </template>
      <template v-slot:data_siff_slot>
        <el-input
          v-model="table.roleName"
          size="mini"
          placeholder="请输入"
          clearable
          @clear="getList()"
          @keyup.enter.native="getList()"
        />
      </template>
      <template v-slot:age_siff_slot>
        <el-input
          v-model="table.roleName"
          size="mini"
          placeholder="请输入"
          clearable
          @clear="getList()"
          @keyup.enter.native="getList()"
        />
      </template>
    </HskTable>
    <br />
  </div>
</template>
<script>
import HskTable from "../package/hsk-table/index.vue";
export default {
  name: "hskTable",
  components: {
    HskTable,
  },
  data() {
    return {
      isHidden:false,
      table: {
        showHeader: true, //是否显示表头
        size: "small", //列表的型号
        fit: true, //列的宽度是否自己撑开
        height: "600", //表格高度
        isRowBgc: false, //如果需要设定行背景,需要指定rowClassName
        rowClassName: {
          bgc: "pink",
          attrName: "xs",
        },
        isStripe: false, // 是否边框
        isBorder: true,
        isMutiSelect: false, //是否需要多选
        isRadio: true, //是否单选
        // 列数据
        columns: [
          {
            type: "slot",
            label: "Tag",
            align: "center", //对其方式
            headerAlign: "center", //表头对其方式
            slot_name: "tag_slot",
            prop: "tag",
            width: "",
          },
          {
            label: "日期",
            prop: "date",
            isCondition: true,
            slot_siff_name: "data_siff_slot",
            width: "",
          },
          {
            label: "年龄",
            prop: "age",
            isCondition: true,
            slot_siff_name: "age_siff_slot",
            width: "",
          },
          {
            label: "姓名",
            prop: "name",
            width: "",
          },
          {
            label: "地址",
            prop: "address",
            width: "",
          },
          {
            type: "slot",
            label: "操作",
            slot_name: "controls_slot",
            width: "",
          },
        ],
        // 行数据
        tableData: [
          {
            id: "1",
            date: "2016-05-02",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1518 弄",
            status: true,
            age: 20,
            xs: true,
          },
          {
            id: "2",
            date: "2016-05-04",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1517 弄",
            status: true,
            age: 20,
            xs: false,
          },
          {
            id: "3",
            date: "2016-05-01",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1519 弄",
            status: true,
            age: 20,
            xs: true,
          },
          {
            id: "4",
            date: "2016-05-03",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1516 弄",
            status: true,
            age: 20,
            xs: false,
          },
          {
            id: "5",
            date: "2016-05-02",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1518 弄",
            status: true,
            age: 20,
            xs: true,
          },
          {
            id: "6",
            date: "2016-05-04",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1517 弄",
            status: true,
            age: 20,
            xs: false,
          },
          {
            id: "7",
            date: "2016-05-01",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1519 弄",
            status: true,
            age: 20,
            xs: true,
          },
          {
            id: "8",
            date: "2016-05-03",
            name: "王小虎",
            address: "上海市普陀区金沙江路 1516 弄",
            status: true,
            age: 20,
            xs: false,
          },
        ],
      },
    };
  },
  methods: {
    //展开隐藏
    showHidden(){
    this.isHidden = !this.isHidden
    },
    //行选中
    tableSlect(val, row) {
      console.log("val, row", val, row);
    },
    //选中改变
    selectChange(val) {
      console.log("val", val);
    },
  },
};
</script>

<style>
.code {
  line-height: 20px;
}
.rotate-180 {
  transform: rotate(180deg);
  transition: transform 0.5s;
}
.rotate-0 {
  transform: rotate(0deg);
  transition: transform 0.5s;
}
</style>

效果:

基于ElementUI二次封装el-table

二次封装el-pagination组件

<template>
  <!-- 分页组件 -->
  <!-- size-change(每页条数)    pageSize 改变时会触发 -->
  <!-- current-change(当前页)  currentPage 改变时会触发 -->
  <!-- page-size    每页显示条目个数,支持 .sync 修饰符 -->
  <!-- page-sizes   每页显示个数选择器的选项设置 -->
  <el-pagination
    @size-change="handleSizeChange"
    @current-change="handleCurrentChange"
    :current-page="currentPage"
    :page-sizes="pageSizes"
    :page-size="pageSize"
    :layout="layout"
    :total="total"
  >
  </el-pagination>
</template>

<script>
export default {
  name:"hsk-pagination",
  props: {
    currentPage: {
      type: [String, Number],
      default: 1,
    },
    total: {
      type: [String, Number],
      default: 0,
    },
    pageSizes: {
      type: Array,
      default: () => [10, 15, 30, 50],
    },
    pageSize: {
      type: [String, Number],
      default: 10,
    },
    layout: {
      type: String,
      default: "total, sizes, prev, pager, next, jumper",
    },
  },
  data() {
    return {};
  },
  methods: {
    handleSizeChange(val) {
      this.$emit("sizeChange", val);
    },
    handleCurrentChange(val) {
      this.$emit("currentChange", val);
    },
  },
};
</script>
<style lang="less" scoped>
.page {
  text-align: center;
  color: #409eff;
}
</style>

属性参数

属性说明
size-change(每页条数)pageSize 改变时会触发
current-change(当前页)currentPage 改变时会触发
page-size每页显示条目个数,支持 .sync 修饰符
page-sizes每页显示个数选择器的选项设置

父组件使用实例

<template>
  <div class="tenant" style="margin: 15px">
    <el-row :gutter="24">
      <el-col :span="24" :xs="24">
        <hsk-pagination
            :total="queryParams.total"
            :currentPage.sync="queryParams.current"
            :pageSize="queryParams.pageSize"
            @sizeChange="sizeChange"
            @currentChange="currentChange"
          ></hsk-pagination>
      </el-col>
    </el-row>
  </div>
</template>
  <script>
import { getListAppByPage } from "@/api/application/application";
export default {
  data() {
    return {
      queryParams: {
        current: 1,
        pageSize: 10,
        total: 0,
      }
    }
  },
  created() {
    this.getList();
  },
  mounted(){
  },
  methods: {
    sizeChange(val) {
      this.queryParams.pageSize = val
      this.getList()
    },
    currentChange(val) {
      this.queryParams.current = val
      this.getList()
    },
    /** 新增租户按钮操作 */
    resetQuery() {
      this.$refs.add.add();
    },
    getList() {
    // 接口文档
      getListAppByPage(this.queryParams)
        .then((res) => {
          this.appList = res.data.data;
          this.table.tableData = res.data.data
          this.queryParams.total = parseInt(res.data.total);
        })
        .catch((err) => {
        });
    },
  },
};
</script>
<style lang="scss" scoped>
</style>

效果

基于ElementUI二次封装el-pagination分页组件

  • 21
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,我可以为您介绍一下Vue3企业级项目中如何二次封装el-table,并实现分页功能的步骤。 步骤如下: 1. 安装 Element-UI 首先,需要安装 Element-UI,可以使用 npm 进行安装: ``` npm install element-plus --save ``` 2. 创建封装组件 接下来,我们需要创建一个可以被重复使用封装组件,可以命名为 `MyTable`,并在该组件中引入 Element-UI 中的 `ElTable`、`ElTableColumn`、`ElPagination` 等组件。 示例代码如下: ```html <template> <div> <el-table v-bind="$attrs" :data="tableData" style="width: 100%"> <el-table-column v-for="item in columns" :key="item.prop" :prop="item.prop" :label="item.label"> </el-table-column> </el-table> <el-pagination v-show="showPagination" v-bind="$attrs" :current-page.sync="currentPage" :page-sizes="pageSizes" :page-size="pageSize" :total="total" :layout="layout" @size-change="handleSizeChange" @current-change="handleCurrentChange"></el-pagination> </div> </template> <script> export default { name: 'MyTable', props: { columns: { type: Array, required: true }, data: { type: Array, required: true }, showPagination: { type: Boolean, default: true }, pageSize: { type: Number, default: 10 }, pageSizes: { type: Array, default: () => [10, 20, 30, 40, 50] }, total: { type: Number }, layout: { type: String, default: 'total, sizes, prev, pager, next, jumper' }, }, data() { return { currentPage: 1, tableData: [], }; }, created() { this.tableData = this.data; }, methods: { handleSizeChange(val) { this.pageSize = val; this.$emit('size-change', val); }, handleCurrentChange(val) { this.currentPage = val; this.$emit('current-change', val); }, }, }; </script> ``` 在上述代码中,我们引入了 `ElTable`、`ElTableColumn`、`ElPagination` 等组件,并根据需要设置了一些默认的 props 属性。同时,我们通过 `$attrs` 绑定了父组件传入的属性,方便在父组件使用。 3. 在父组件使用封装组件 在父组件使用我们刚才封装的 `MyTable` 组件,可以传入 `columns`、`data` 等属性,示例代码如下: ```html <template> <div class="container"> <my-table :columns="columns" :data="data" @current-change="handleCurrentPageChange" @size-change="handleSizeChange"></my-table> </div> </template> <script> import MyTable from '@/components/MyTable.vue'; export default { name: 'MyTableDemo', components: { MyTable, }, data() { return { columns: [ { label: '姓名', prop: 'name' }, { label: '年龄', prop: 'age' }, { label: '性别', prop: 'gender' }, ], data: [], currentPage: 1, pageSize: 10, total: 0, }; }, methods: { // 模拟异步请求数据 fetchData() { setTimeout(() => { this.data = [ { name: '张三', age: 18, gender: '男' }, { name: '李四', age: 22, gender: '女' }, { name: '王五', age: 30, gender: '男' }, { name: '赵六', age: 25, gender: '女' }, ]; this.total = 4; }, 1000); }, handleCurrentPageChange(val) { // 处理页码变化 this.currentPage = val; this.fetchData(); }, handleSizeChange(val) { // 处理每页显示数量变化 this.pageSize = val; this.fetchData(); }, }, created() { this.fetchData(); }, }; </script> ``` 在上述代码中,我们引入了刚才封装的 `MyTable` 组件,并传入了 `columns`、`data` 等属性。同时,我们在 `handleCurrentPageChange` 和 `handleSizeChange` 方法中处理了页码变化和每页显示数量变化的情况,并触发了 `fetchData` 方法重新请求数据。 4. 完成 至此,我们就完成了 Vue3 企业级项目二次封装 `el-table` 并实现分页功能的步骤。通过封装组件,我们可以提高代码的复用性和可维护性,同时提升项目的开发效率。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值