2021如何让你的Table组件无限可能

在管理后台中咱们会使用大量的表格表单组件, 导入导出各类报表, 有些场景还须要对报表数据进行可视化分析, 动态生成可视化图表, 笔者将基于以上场景, 总结一些实用的 Table 组件开发技巧, 让前端开发再也不吃力.javascript

往期经典

效果演示

技术点

  • 实现 Table 动态渲染
  • Table 排序, 多列排序, 自定义搜索
  • 批量导入 Excel 数据渲染 Table
  • Table 数据导出为 Excel 文件
  • 基于 Table 数据自动生成多维度可视化报表
  • 实现简单的 Table 编辑器

以上是几个经常使用的业务分场景, 接下来笔者带你们一一实现.css

技术实现

1. 实现 Table 动态渲染

通常咱们渲染表格, 大多数是预先将表格结构写好, 比先定义好columns再向后端请求数据填充表格, 以下:前端

const columns = [
  {
    title: '姓名',
    dataIndex: 'name',
    key: 'name',
  },
  {
    title: '年龄',
    dataIndex: 'age',
    key: 'age',
  },
  {
    title: '住址',
    dataIndex: 'address',
    key: 'address',
  },
];
const dataSource = [
  {
    key: '1',
    name: '徐小夕',
    age: 18,
    address: '杭州夕湖区',
  }
];

<Table dataSource={dataSource} columns={columns} />
复制代码

这种业务场景虽然能够知足大部分后台管理系统的Table需求, 也可使用 antd 或者 element 构建, 可是对于 lowcode 系统而言, 不少模块都是不肯定的, 咱们须要根据协议数据来驱动 Table 的渲染.vue

好比咱们在 H5-Dooring 中配置了一个表单, 咱们要统计分析表单的数据, 因为表单项是不肯定的, 因此咱们没法提早定义好一个 table schema. java

那如何来动态渲染这个 Table 呢? 这里给你们提供一个思路, 基于数据驱动 + 协议层约束. 相似于国外 SAP 的 低代码平台, 彻底基于 odata 协议, 咱们能够约束表单的提交数据格式, 而后结合用户提交的数据, 动态提取出 Table 所需的 columns, 最后再渲染 Table 组件.node

协议层主要约束不一样字段的展现类型, 好比字符串, 按钮, 连接, 标签等, 用户在提交表单以后会携带协议层对应的 flag 和用户输入的值, 这有利于咱们解析器渲染Table时能够对不一样的列展现不一样的类型. 以下: react

笔者这里简单实现一个demo, 以下:webpack

// table数据源
let tableData = res.map((item:any,i:number) => ({ ID: nanoid(8), ...item }));
let baseRow = tableData[0],
keys = Object.keys(baseRow);
setColumns(() => {
  const baseColumn = keys.map(item => {
    return {
      title: item,
      dataIndex: item,
      key: item,
      width: item === 'ID' ? 0 : null,
      render: (v:any) => {
        if(typeof v === 'object') {
          return <> { v.map(item => <Tag color="#2F54EB">{ item.label || item }</Tag>) } </>
        }
        return item === 'ID' ? '' : v
      }
    }
  })
  baseColumn.push({
    title: '操做',
    key: 'operation',
    fixed: 'right',
    width: 100,
    render: (row) => <a onClick={() => handleDel(row)}>删除</a>,
  })
  return baseColumn
})
复制代码

以上咱们就实现了一个动态 Table 渲染方案, 案例中使用了 react, 你们也可使用熟悉的 vue3.0.css3

2. Table 排序, 多列排序, 自定义搜索

Table 排序, 多列排序实现方式也很简单, 咱们只须要自定义 Table 头部, 对排序字段提高为 Table 的公共 State, 最后经过排序标识和排序方法进行排序便可. 目前 antd4.0已经支持多列排序, 你们能够直接参考学习便可, 以下: 对于自定义搜索, 也就是文章开头的 demo 展现的列搜索, 咱们能够采用以下方案实现:git

const getColumnSearchProps = dataIndex => ({
    filterDropdown: ({ setSelectedKeys, selectedKeys, confirm, clearFilters }) => (
      <div style={{ padding: 8 }}> <Input ref={node => { searchInput = node; }} placeholder={`Search ${dataIndex}`} value={selectedKeys[0]} onChange={e => setSelectedKeys(e.target.value ? [e.target.value] : [])} onPressEnter={() => handleSearch(selectedKeys, confirm, dataIndex)} style={{ width: 188, marginBottom: 8, display: 'block' }} /> <Space> <Button type="primary" onClick={() => handleSearch(selectedKeys, confirm, dataIndex)} icon={<SearchOutlined />} size="small" style={{ width: 90 }} > 搜索 </Button> <Button onClick={() => handleReset(clearFilters)} size="small" style={{ width: 90 }}> 重置 </Button> </Space> </div>
    ),
    filterIcon: filtered => <SearchOutlined style={{ color: filtered ? '#1890ff' : undefined }} />,
    onFilter: (value, record) =>
      record[dataIndex]
        ? record[dataIndex].toString().toLowerCase().includes(value.toLowerCase())
        : '',
    onFilterDropdownVisibleChange: visible => {
      if (visible) {
        setTimeout(() => searchInput.select(), 100);
      }
    },
    render: text =>
      searchedColumn === dataIndex ? (
        <Highlighter highlightStyle={{ backgroundColor: '#ffc069', padding: 0 }} searchWords={[searchText]} autoEscape textToHighlight={text ? text.toString() : ''} />
      ) : (
        text
      ),
  });
复制代码

此时咱们只须要对动态生成的columns每一列添加自定义头部便可:

{
  title: item,
  dataIndex: item,
  key: item,
  ...getColumnSearchProps(item)
}
复制代码

antd4.0 中也有详细的使用方式, 这里笔者就不一一介绍了, 效果以下: 你们也能够在 H5-Dooring 的管理后台中查看具体效果.

3. 批量导入 Excel 数据渲染 Table

在不少数据分析后台中咱们须要处理不少电子表格, 用传统的excel手动录入的方式将慢慢被淘汰. 好比不一样渠道方收集到了不少业务数据, 整理到 excel 中, 那如何快速保存到本身的后台系统中呢? 通常的思路以下:

  • 经过表单的方式一条条录入
  • 后台解析文件处理成规范的可入库数据结构
  • 前端实现一件导入 excel, 自动同步数据

第一种方案因为效率过低, 适合C端用户手动录入, 咱们暂时忽略, 笔者将实现一下第三种方案. 实现思路以下:

这里咱们用到了 XLSX 这个库, 结合 FileReader API. 咱们经过FileReader 拿到excel文件的二进制数据, 而后传给 XLSX 解析成 js object, 最后经过笔者写的 table 渲染器生成符合规范的table数据结构. 源码以下:

// 解析并提取excel数据
let reader = new FileReader();
reader.onload = function(e) {
  let data = e.target.result;
  let workbook = XLSX.read(data, {type: 'binary'});
  let sheetNames = workbook.SheetNames; // 工做表名称集合
  let draftArr = {}
  sheetNames.forEach(name => {
    let worksheet = workbook.Sheets[name]; // 只能经过工做表名称来获取指定工做表
    for(let key in worksheet) {
      // v是读取单元格的原始值
      if(key[0] !== '!') {
        if(draftArr[key[0]]) {
          draftArr[key[0]].push(worksheet[key].v)
        }else {
          draftArr[key[0]] = [worksheet[key].v]
        }
      }
    }
  });
  // 获得table合法的数据产物
  const sourceData = Object.values(draftArr);
}
reader.readAsBinaryString(file);
复制代码

拿到合法的table 数据源以后咱们就能够进行第一节说的动态渲染 Table 的逻辑了.

经过以上的方式, 咱们能够实现任何结构的excel表格的导入. 在导入后咱们能够自动发送请求存储到咱们的业务后台中.

4. 将 Table 数据导出为 Excel

相似的, 上面咱们介绍了将 excel 导入 table, 一样咱们也能够将Table 导出为 excel, 进行数据的分发, 本地化, 好比咱们最近流行的在线文档等应用. 笔者这里简单讲一下实现思路:

也就是咱们第3节说的反解析. excel 文件生成笔者采用 js-export-excel 这个库, 基于它笔者实现了一个开箱即用的方法, 避免你们烧脑造轮子. 以下:

import ExportJsonExcel from 'js-export-excel';

const generateExcel = () => {
    let option:any = {};  //option表明的就是excel文件
    let dataTable = [];  //excel文件中的数据内容
    let len = list.length;
    if (len) {
        for(let i=0; i<len; i++) {
            let row = list[i];
            let obj:any = {};
            for(let key in row) {
                if(typeof row[key] === 'object') {
                    let arr:any = row[key];
                    obj[key] = arr.map((item:any) => (typeof item === 'object' ? item.label : item)).join(',')
                }else {
                    obj[key] = row[key]
                }
            }
            dataTable.push(obj);  //设置excel中每列所获取的数据源
        }
    }
    let tableKeys = Object.keys(dataTable[0]);
    option.fileName = tableName;  //excel文件名称
    option.datas = [
          {
            sheetData: dataTable,  //excel文件中的数据源
            sheetName: tableName,  //excel文件中sheet页名称
            sheetFilter: tableKeys,  //excel文件中需显示的列数据
            sheetHeader: tableKeys,  //excel文件中每列的表头名称
          }
    ]
    let toExcel = new ExportJsonExcel(option);  //生成excel文件
    toExcel.saveExcel();  //下载excel文件
}
复制代码

5. 基于 Table 数据自动生成多维度可视化报表

在后台管理系统和 BI 平台中咱们会遇到不少数据分析和报表展现的需求, 接下来笔者未来介绍一下如何基于 Table 数据动态生成多维度可视化分析报表.

笔者在以前的文章中介绍过 度量行这个概念, 对于数据分析而言, 咱们也要考虑可分析维度的概念, 好比什么是可分析的, 什么是不可分析的. 好比咱们又一个表格, 里面有以下结构: 对于联系方式而言, 它是不可度量的, 即分析该项指没有任何价值, 因此在自动生成多维度分析中咱们理论上不因该分析它, 基于这个原理, 咱们来设计一个简单的自动生成多维度可视化报表的方案.

5.1 基于数据源获取维度数据

咱们针对具备范围属性的维度进行度量, 生成度量数据, 代码以下:

const generateDistData = (key:string, list:any) => {
    let distDataMap:any = {},
        distData = []
    list.forEach((item:any) => {
        // 当前纬度的类别
        let curKey = typeof item[key] === 'object' ? item[key][0].label : item[key];
        if(distDataMap[curKey]) {
            distDataMap[curKey]++;
        }else {
            distDataMap[curKey] = 1;
        }
    })

    // 生成目标数组
    for(let key in distDataMap) {
        distData.push({name: key, value: distDataMap[key]})
    }
    return distData
  }
复制代码

此时咱们只须要根据维度的字段, 便可获取某一维度的数据值, 后经过可视化组件渲染便可.

5.2 基于某一维度生成可视化报表

咱们用@ant-design/charts, 代码以下:

<div className={styles.anazlyHeader}>
    <div className={styles.anazlyItem}> <span>分析纬度: </span> <Select style={{ width: 120 }} onChange={(v) => handleAnazlyChange(0, v)} defaultValue={keys[0]}> { keys.map((item,i) => { return <Option value={item} key={i}>{ item }</Option> }) } </Select> </div>
</div>
<div className={styles.anazlyContent}> { !!config && <Pie {...config} /> } </div>
复制代码

实现效果以下:

6. 实现简单的 Table 编辑器

实现 Table 编辑器其实笔者在 前端如何一键生成多维度数据可视化分析报表 已经详细分析过了,也集成在了H5-Dooring 的可视化组件编辑器中, 具体 demo 以下:

你们感兴趣能够研究一下.

最后

目前笔者也在持续更新H5编辑器 H5-Dooring, 最近来同步一下功能:

  • 修复图片库选择bug
  • 添加省市级联组件
  • 添加批量导入 excel 数据的能力
  • 添加表单自定义校验
  • 音频组件添加自动播放控制, 循环播放等配置项
  • 添加横向滑动组件

以为有用 ?喜欢就收藏,顺便点个吧,你的支持是我最大的鼓励!微信搜 “趣谈前端”,发现更多有趣的H5游戏, webpack,node,gulp,css3,javascript,nodeJS,canvas数据可视化等前端知识和实战.

相关文章
相关标签/搜索