大家在做后台管理系统的时候,写的最多的可能就是表格页面了,一般分三部分:搜索功能区、表格内容区和分页器区。一般这些功能都是使用第三方组件库实现,比如说element-ui,或者vuetify。这两个组件库都各有各的优点,但就table组件来说,我还是比较喜欢vuetify的实现,不用手写一个个column,只要传入headers的配置数组就行,甚至分页器都内置在了table组件里,用起来十分方便。有兴趣可以看看:vuetify data table

上面是一个经典的用element-ui开发的table页面,而且实际工作中如果每个table页面都写一遍,重复代码太多了,所以不妨写一个table模板组件,减少重复代码。我的思路是这样的:

  1. 搜索功能区:

    提供searchBar插槽,可以自定义搜索输入框,搜索、重置按钮必有,新增按钮通过props控制显隐。这里对应的代码如下:

    genSearchBar() {
    if (this.noSearchBar || !this.$scopedSlots.searchBar) return '';
    return (
    <el-form class="seatch-form" inline={true} label-width="100">
    {this.$scopedSlots.searchBar()}
    <el-button
    class="filter-item"
    icon="el-icon-search"
    type="primary"
    onClick={this.handleSearchBtnClick}
    >
    查询
    </el-button>
    <el-button
    class="filter-item"
    icon="el-icon-refresh"
    onClick={this.handleResetBtnClick}
    >
    重置
    </el-button>
    <el-button
    class="filter-item"
    icon="el-icon-plus"
    type="primary"
    v-show={this.showAddBtn}
    onClick={this.handleAddBtnClick}
    >
    新增
    </el-button>
    </el-form>
    );
    }
  2. 表格内容区:

    通过传入headers自动生成columns,参数如下:

    {
    label: '性别',
    prop: 'sex',
    width: '180',
    filter: 'sexFilter'
    }

    可对应如下代码:

    <el-table-column
    prop="sex"
    label="性别"
    width="180">
    <template slot-scope="scope">{{scope.row.sex | sexFilter}}</template>
    </el-table-column>

    注意,只支持全局filter

    如果你想自定义column,也提供tableColumn插槽,支持自定义column,可以如下配置:

    {
    prop: 'action'
    }
    <el-table-column
    prop="action"
    label="操作"
    width="180">
    <template slot-scope="scope">
    <el-button>编辑</el-button>
    <el-button>删除</el-button>
    </template>
    </el-table-column>

    这样,就会按传入的prop匹配对应的column,十分方便。

    实现代码如下:

    genTableSlot(h) {
    let customeColumns = this.$scopedSlots.tableColumn
    ? this.$scopedSlots.tableColumn()
    : [];
    return this.headers.map((item) => {
    // 根据item.prop判断是否使用传入的插槽内容
    let foundItem = customeColumns.find(
    (ele) =>
    ele.componentOptions &&
    ele.componentOptions.propsData.prop === item.prop
    );
    return foundItem
    ? foundItem
    : h('el-table-column', {
    props: {
    ...item,
    },
    scopedSlots: {
    default: (props) => {
    // 根据传入的全局filter处理column数据
    let filter = this.$options.filters[
    item.filter
    ];
    let itemValue = props.row[item.prop];
    return h(
    'span',
    filter ? filter(itemValue) : itemValue
    );
    },
    },
    });
    });
    }
    genTable(h) {
    return h(
    'el-table',
    {
    ref: 'tableRef',
    props: {
    ...this.$attrs,
    data: this.data,
    },
    on: {
    'selection-change': (val) => {
    this.$emit('selection-change', val);
    },
    },
    },
    [...this.genTableSlot(h)]
    );
    }
  3. 分页器区:

    如无特殊需求,分页器功能一致,所以直接内置。

    实现代码如下:

    genPagination() {
    return (
    <div class="pagination-wrap">
    <el-pagination
    layout="total,prev,pager,next,jumper"
    current-page={this.current}
    page-size={this.pageSize}
    total={this.total}
    {...{
    on: { 'current-change': this.handleCurrentChange },
    }}
    ></el-pagination>
    </div>
    );
    }

    最后附完整代码和demo:

    <script>
    export default {
    name: 'TableTemplate',
    props: {
    data: {
    type: Array,
    default: () => [],
    required: true,
    },
    headers: {
    type: Array,
    default: () => [],
    required: true,
    },
    current: {
    type: Number,
    default: 1,
    },
    pageSize: {
    type: Number,
    default: 10,
    },
    total: {
    type: Number,
    default: 0,
    },
    noSearchBar: Boolean,
    showAddBtn: Boolean,
    },
    mounted() {
    this.$nextTick(() => {
    this.$emit('search');
    });
    },
    methods: {
    genSearchBar() {
    if (this.noSearchBar || !this.$scopedSlots.searchBar) return '';
    return (
    <el-form class="seatch-form" inline={true} label-width="100">
    {this.$scopedSlots.searchBar()}
    <el-button
    class="filter-item"
    icon="el-icon-search"
    type="primary"
    onClick={this.handleSearchBtnClick}
    >
    查询
    </el-button>
    <el-button
    class="filter-item"
    icon="el-icon-refresh"
    onClick={this.handleResetBtnClick}
    >
    重置
    </el-button>
    <el-button
    class="filter-item"
    icon="el-icon-plus"
    type="primary"
    v-show={this.showAddBtn}
    onClick={this.handleAddBtnClick}
    >
    新增
    </el-button>
    </el-form>
    );
    },
    genTableSlot(h) {
    let customeColumns = this.$scopedSlots.tableColumn
    ? this.$scopedSlots.tableColumn()
    : [];
    return this.headers.map((item) => {
    // 根据item.prop判断是否使用传入的插槽内容
    let foundItem = customeColumns.find(
    (ele) =>
    ele.componentOptions &&
    ele.componentOptions.propsData.prop === item.prop
    );
    return foundItem
    ? foundItem
    : h('el-table-column', {
    props: {
    ...item,
    },
    scopedSlots: {
    default: (props) => {
    let filter = this.$options.filters[
    item.filter
    ];
    let itemValue = props.row[item.prop];
    return h(
    'span',
    filter ? filter(itemValue) : itemValue
    );
    },
    },
    });
    });
    },
    genTable(h) {
    return h(
    'el-table',
    {
    ref: 'tableRef',
    props: {
    ...this.$attrs,
    data: this.data,
    },
    on: {
    'selection-change': (val) => {
    this.$emit('selection-change', val);
    },
    },
    },
    [...this.genTableSlot(h)]
    );
    },
    genPagination() {
    return (
    <div class="pagination-wrap">
    <el-pagination
    layout="total,prev,pager,next,jumper"
    current-page={this.current}
    page-size={this.pageSize}
    total={this.total}
    {...{
    on: { 'current-change': this.handleCurrentChange },
    }}
    ></el-pagination>
    </div>
    );
    },
    resetPagination() {
    this.$emit('update:current', 1);
    },
    handleCurrentChange(val) {
    this.$emit('update:current', val);
    this.$emit('search');
    },
    handleSearchBtnClick() {
    this.$emit('search');
    },
    handleResetBtnClick() {
    this.resetPagination();
    this.$emit('reset');
    },
    handleAddBtnClick() {
    this.$emit('add');
    },
    getTableRef() {
    return this.$refs.tableRef;
    },
    },
    render(h) {
    return (
    <div>
    {this.genSearchBar()}
    {this.genTable(h)}
    {this.genPagination()}
    </div>
    );
    },
    };
    </script> <style scoped>
    .seatch-form {
    text-align: left;
    }
    .pagination-wrap {
    margin-top: 20px;
    text-align: right;
    }
    </style>

    Demo:

    <template>
    <div>
    <table-template
    border
    :headers="headers"
    :data="tableData"
    :current.sync="current"
    :total="total"
    ref="tableTemplate"
    showAddBtn
    @search="handleSearch"
    @reset="handleReset"
    @add="handleAdd"
    @selection-change="handleSelectionChange"
    >
    <template #searchBar>
    <el-form-item label="姓名:" prop="title">
    <el-input class="filter-item" v-model="searchForm.title" ></el-input>
    </el-form-item>
    </template>
    <template #tableColumn>
    <el-table-column
    prop="selection"
    type="selection"
    width="55"
    ></el-table-column>
    <el-table-column prop="test" label="姓名" width="180">
    <template slot-scope="scope">
    <el-popover trigger="hover" placement="top">
    <p>姓名:{{ scope.row.name }}</p>
    <p>住址:{{ scope.row.address }}</p>
    <div slot="reference" class="name-wrapper">
    <el-tag size="medium">{{scope.row.name}}</el-tag>
    </div>
    </el-popover>
    </template>
    </el-table-column>
    </template>
    </table-template>
    </div>
    </template> <script>
    import TableTemplate from './TableTemplate';
    export default {
    name: 'Demo',
    components: {
    TableTemplate,
    },
    data() {
    return {
    current: 1,
    total: 100,
    headers: [
    {
    prop: 'selection',
    },
    {
    label: '姓名',
    prop: 'name',
    width: '100',
    },
    {
    label: '年龄',
    prop: 'year',
    },
    {
    label: '性别',
    prop: 'sex',
    width: 'sexFilter',
    },
    {
    prop: 'test',
    },
    ],
    tableData: [
    {
    name: 'curry',
    year: 18,
    sex: 'female',
    address: '天安门',
    },
    ],
    searchForm: {
    title: '',
    },
    };
    },
    methods: {
    handleSearch() {
    console.log(this.current);
    },
    handleReset() {
    this.searchForm = {
    title: '',
    };
    },
    handleAdd() {
    console.log('添加');
    },
    handleSelectionChange(val) {
    console.log(val);
    },
    getTableRef() {
    console.log(this.$refs.tableTemplate.getTableRef());
    },
    },
    };
    </script>

基于element-ui封装一个Table模板组件的更多相关文章

  1. 基于element UI 的上传插件

    为了不再重复的上传文件,做了一个统一选择文件和上传文件的 基于 element UI :http://element-cn.eleme.io 前端实现文件下载和拖拽上传 演示 用法 <uploa ...

  2. 封装Vue Element的可编辑table表格组件

    前一段时间,有博友在我那篇封装Vue Element的table表格组件的博文下边留言说有没有那种"表格行内编辑"的封装组件,我当时说我没有封装过这样的组件,因为一直以来在实际开发 ...

  3. vue+element ui 的表格列使用组件

    前言:工作中用到 vue+element ui 的前端框架,有这个场景:很多表格的列有许多一样的,所以考虑将列封装为组件.转载请注明出处:https://www.cnblogs.com/yuxiaol ...

  4. 基于ionic框架封装一个图片轮播指令的几点

    在这里我想在项目中封装一个图片轮播的指令 (本项目使用的是ionic框架) 1)定义指令 define(['app'],function(myapp){ myapp.directive('myslid ...

  5. 基于element ui的图片预览插件

    写插件很简单,满足两个条件即可,一.基本的逻辑思路,二.熟悉插件语法要求.本次Vue插件也比较简单,点击“查看图片”用轮播的方式限制用户上传的图片,如图: 项目采用的是vue-element-admi ...

  6. vue封装一个弹框组件

    这是一个提示框和对话框,例:   这是一个组件 eject.vue <template> <div class='kz-cont' v-show='showstate'> &l ...

  7. 基于element ui的级联选择器组件实现的分类后台接口

    今天在做资产管理系统的时候遇到一个分类的级联选择器,前端是用的element的组件,需要后台提供接口支持.     这个组件需要传入的数据结构大概是这样的,详细的可参考官方案例: [{ value: ...

  8. Element ui 中使用table组件实现分页记忆选中

    我们再用vue和element-ui,或者其他的表格的时候,可能需要能记忆翻页勾选,那么实现以下几个方法就ok了 示例如下 <el-table :data="tableData&quo ...

  9. element ui 里面的table怎么弹出一个框让表中数据点击出现弹框

    <el-table-column label="团队" prop="name" min-width="120px" align=&qu ...

随机推荐

  1. Flink-v1.12官方网站翻译-P001-Local Installation

    本地安装 按照以下几个步骤下载最新的稳定版本并开始使用. 第一步:下载 为了能够运行Flink,唯一的要求是安装了一个有效的Java 8或11.你可以通过以下命令检查Java的正确安装. java - ...

  2. xLua热更新插件

    一.xLua插件下载安装 1.从GitHub上搜索并下载插件 2.将文件复制到unity中 3.检查是否有错误 二.在unity中调用lua 1.简单调用 在c#脚本中使用LuaEnv类可以运行lua ...

  3. Jenkins(3)拉取git仓库代码,执行python自动化脚本

    前言 python自动化的脚本开发完成后需提交到git代码仓库,接下来就是用Jenkins拉取代码去构建自动化代码了 新建项目 打开Jenkins新建一个自由风格的项目 源码管理 Repository ...

  4. Java程序操作Hive

    1.hive的lib+jdbc,还要把mysql的连接驱动加载过来 2.编写程序 开启远程服务:[root@zhiyou ~]# hiveserver2 &[1] 4127[root@zhiy ...

  5. hdu4348 To the moon (主席树 || 离线线段树)

    Problem Description Background To The Moon is a independent game released in November 2011, it is a ...

  6. Bubble Cup 13 - Finals [Online Mirror, unrated, Div. 1] K. Lonely Numbers (数学)

    题意:定义两个数\(a,b\)是朋友,如果:\(gcd(a,b)\),\(\frac{a}{gcd(a,b)}\),\(\frac{b}{gcd(a,b)}\)能构成三角形,现在给你一个正整数\(n\ ...

  7. Codeforces Round #681 (Div. 2, based on VK Cup 2019-2020 - Final) D. Extreme Subtraction (贪心)

    题意:有一个长度为\(n\)的序列,可以任意取\(k(1\le k\le n)\),对序列前\(k\)项或者后\(k\)减\(1\),可以进行任意次操作,问是否可以使所有元素都变成\(0\). 题解: ...

  8. Codeforces Round #678 (Div. 2) C. Binary Search (二分,组合数)

    题意:有长度\(n\)的序列,让你构造序列,使得二分查找能在\(pos\)位置找到值\(x\).问最多能构造出多少种排列? 题解:题目给出的\(pos\)是固定的,所以我们可以根据图中所给的代码来进行 ...

  9. spring再学习之AOP实操

    一.spring导包 2.目标对象 public class UserServiceImpl implements UserService { @Override public void save() ...

  10. Leetcode(877)-石子游戏

    亚历克斯和李用几堆石子在做游戏.偶数堆石子排成一行,每堆都有正整数颗石子 piles[i] . 游戏以谁手中的石子最多来决出胜负.石子的总数是奇数,所以没有平局. 亚历克斯和李轮流进行,亚历克斯先开始 ...