前端:如何让你的Table组件无限可能

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

目前创新互联已为1000+的企业提供了网站建设、域名、虚拟空间、绵阳服务器托管、企业网站设计、京山网站维护等服务,公司将坚持客户导向、应用为本的策略,正道将秉承"和谐、参与、激情"的文化,与客户和合作伙伴齐心协力一起成长,共同发展。

往期经典

  • 轻松教你搞定组件的拖拽, 缩放, 多控制点伸缩和拖拽数据上报
  • 基于自然流布局的可视化拖拽搭建平台设计方案
  • 基于React+Koa实现一个h5页面可视化编辑器-H5-Dooring

效果演示

技术点

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

以上是几个常用的业务分场景, 接下来笔者带大家一一实现.

技术实现

1. 实现 Table 动态渲染

一般我们渲染表格, 大多数是预先将表格结构写好, 比先定义好columns再向后端请求数据填充表格, 如下:

 
 
 
 
  1. const columns = [
  2.   {
  3.     title: '姓名',
  4.     dataIndex: 'name',
  5.     key: 'name',
  6.   },
  7.   {
  8.     title: '年龄',
  9.     dataIndex: 'age',
  10.     key: 'age',
  11.   },
  12.   {
  13.     title: '住址',
  14.     dataIndex: 'address',
  15.     key: 'address',
  16.   },
  17. ];
  18. const dataSource = [
  19.   {
  20.     key: '1',
  21.     name: '徐小夕',
  22.     age: 18,
  23.     address: '杭州夕湖区',
  24.   }
  25. ];

 这种业务场景虽然可以满足大部分后台管理系统的Table需求, 也可以使用 antd 或者 element构建, 但是对于 lowcode 系统而言, 很多模块都是不确定的, 我们需要根据协议数据来驱动 Table 的渲染.

比如我们在 H5-Dooring中配置了一个表单, 我们要统计分析表单的数据, 由于表单项是不确定的, 所以我们无法提前定义好一个table schema.

那如何来动态渲染这个 Table 呢? 这里给大家提供一个思路, 基于数据驱动 + 协议层约束. 类似于国外 SAP 的 低代码平台, 完全基于 odata 协议, 我们可以约束表单的提交数据格式, 然后结合用户提交的数据, 动态提取出 Table 所需的 columns, 最后再渲染 Table 组件.

协议层主要约束不同字段的展示类型, 比如字符串, 按钮, 链接, 标签等, 用户在提交表单之后会携带协议层对应的 flag 和用户输入的值, 这有利于我们解析器渲染Table时可以对不同的列展示不同的类型. 如下:

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

 
 
 
 
  1. // table数据源
  2. let tableData = res.map((item:any,i:number) => ({ ID: nanoid(8), ...item }));
  3. let baseRow = tableData[0],
  4. keys = Object.keys(baseRow);
  5. setColumns(() => {
  6.   const baseColumn = keys.map(item => {
  7.     return {
  8.       title: item,
  9.       dataIndex: item,
  10.       key: item,
  11.       width: item === 'ID' ? 0 : null,
  12.       render: (v:any) => {
  13.         if(typeof v === 'object') {
  14.           return <>
  15.             {
  16.               v.map(item => { item.label || item })
  17.             }
  18.           
  19.         }
  20.         return item === 'ID' ? '' : v
  21.       }
  22.     }
  23.   })
  24.   baseColumn.push({
  25.     title: '操作',
  26.     key: 'operation',
  27.     fixed: 'right',
  28.     width: 100,
  29.     render: (row) =>  handleDel(row)}>删除,
  30.   })
  31.   return baseColumn
  32. })

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

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

Table排序, 多列排序实现方式也很简单, 我们只需要自定义Table头部, 对排序字段提升为 Table的公共State, 最后通过排序标识和排序方法进行排序即可. 目前antd4.0已经支持多列排序, 大家可以直接参考学习即可, 如下:

对于自定义搜索, 也就是文章开头的 demo 展示的列搜索, 我们可以采用如下方案实现:

 
 
 
 
  1. const getColumnSearchProps = dataIndex => ({
  2.   filterDropdown: ({ setSelectedKeys, selectedKeys, confirm, clearFilters }) => (
  3.     
  4.       
  5.         ref={node => {
  6.           searchInput = node;
  7.         }}
  8.         placeholder={`Search ${dataIndex}`}
  9.         value={selectedKeys[0]}
  10.         onChange={e => setSelectedKeys(e.target.value ? [e.target.value] : [])}
  11.         onPressEnter={() => handleSearch(selectedKeys, confirm, dataIndex)}
  12.         style={{ width: 188, marginBottom: 8, display: 'block' }}
  13.       />
  14.       
  15.         
  16.           type="primary"
  17.           onClick={() => handleSearch(selectedKeys, confirm, dataIndex)}
  18.           icon={}
  19.           size="small"
  20.           style={{ width: 90 }}
  21.         >
  22.           搜索
  23.         
  24.          handleReset(clearFilters)} size="small" style={{ width: 90 }}>
  25.           重置
  26.         
  27.       
  28.     
  •   ),
  •   filterIcon: filtered => ,
  •   onFilter: (value, record) =>
  •     record[dataIndex]
  •       ? record[dataIndex].toString().toLowerCase().includes(value.toLowerCase())
  •       : '',
  •   onFilterDropdownVisibleChange: visible => {
  •     if (visible) {
  •       setTimeout(() => searchInput.select(), 100);
  •     }
  •   },
  •   render: text =>
  •     searchedColumn === dataIndex ? (
  •       
  •         highlightStyle={{ backgroundColor: '#ffc069', padding: 0 }}
  •         searchWords={[searchText]}
  •         autoEscape
  •         textToHighlight={text ? text.toString() : ''}
  •       />
  •     ) : (
  •       text
  •     ),
  • });
  •  此时我们只需要对动态生成的columns每一列添加自定义头部即可:

     
     
     
     
    1. {
    2.   title: item,
    3.   dataIndex: item,
    4.   key: item,
    5.   ...getColumnSearchProps(item)
    6. }

    antd4.0 中也有详细的使用方式, 这里笔者就不一一介绍了, 效果如下:

    大家也可以在 H5-Dooring 的管理后台中查看具体效果.

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

    在很多数据分析后台中我们需要处理很多电子表格, 用传统的excel手动录入的方式将慢慢被淘汰. 比如不同渠道方收集到了很多业务数据, 整理到 excel 中, 那如何快速保存到自己的后台系统中呢? 一般的思路如下:

    第一种方案由于效率太低, 适合C端用户手动录入, 我们暂时忽略, 笔者将实现一下第三种方案. 实现思路如下:

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

     
     
     
     
    1. // 解析并提取excel数据
    2. let reader = new FileReader();
    3. reader.onload = function(e) {
    4.   let data = e.target.result;
    5.   let workbook = XLSX.read(data, {type: 'binary'});
    6.   let sheetNames = workbook.SheetNames; // 工作表名称集合
    7.   let draftArr = {}
    8.   sheetNames.forEach(name => {
    9.     let worksheet = workbook.Sheets[name]; // 只能通过工作表名称来获取指定工作表
    10.     for(let key in worksheet) {
    11.       // v是读取单元格的原始值
    12.       if(key[0] !== '!') {
    13.         if(draftArr[key[0]]) {
    14.           draftArr[key[0]].push(worksheet[key].v)
    15.         }else {
    16.           draftArr[key[0]] = [worksheet[key].v]
    17.         }
    18.       }
    19.     }
    20.   });
    21.   // 得到table合法的数据产物
    22.   const sourceData = Object.values(draftArr);
    23. }
    24. reader.readAsBinaryString(file);

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

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

    4. 将 Table 数据导出为 Excel

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

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

     
     
     
     
    1. import ExportJsonExcel from 'js-export-excel';
    2. const generateExcel = () => {
    3.     let option:any = {};  //option代表的就是excel文件
    4.     let dataTable = [];  //excel文件中的数据内容
    5.     let len = list.length;
    6.     if (len) {
    7.         for(let i=0; i
    8.             let row = list[i];
    9.             let obj:any = {};
    10.             for(let key in row) {
    11.                 if(typeof row[key] === 'object') {
    12.                     let arr:any = row[key];
    13.                     obj[key] = arr.map((item:any) => (typeof item === 'object' ? item.label : item)).join(',')
    14.                 }else {
    15.                     obj[key] = row[key]
    16.                 }
    17.             }
    18.             dataTable.push(obj);  //设置excel中每列所获取的数据源
    19.         }
    20.     }
    21.     let tableKeys = Object.keys(dataTable[0]);
    22.     option.fileName = tableName;  //excel文件名称
    23.     option.datas = [
    24.           {
    25.             sheetData: dataTable,  //excel文件中的数据源
    26.             sheetName: tableName,  //excel文件中sheet页名称
    27.             sheetFilter: tableKeys,  //excel文件中需显示的列数据
    28.             sheetHeader: tableKeys,  //excel文件中每列的表头名称
    29.           }
    30.     ]
    31.     let toExcel = new ExportJsonExcel(option);  //生成excel文件
    32.     toExcel.saveExcel();  //下载excel文件
    33. }

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

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

    笔者在之前的文章中介绍过 度量行这个概念, 对于数据分析而言, 我们也要考虑可分析维度的概念, 比如什么是可分析的, 什么是不可分析的. 比如我们又一个表格, 里面有如下结构:

    对于联系方式而言, 它是不可度量的, 即分析该项指没有任何价值, 所以在自动生成多维度分析中我们理论上不因该分析它, 基于这个原理, 我们来设计一个简单的自动生成多维度可视化报表的方案.

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

    我们针对具有范围属性的维度进行度量, 生成度量数据, 代码如下:

     
     
     
     
    1. const generateDistData = (key:string, list:any) => {
    2.     let distDataMap:any = {},
    3.         distData = []
    4.     list.forEach((item:any) => {
    5.         // 当前纬度的类别
    6.         let curKey = typeof item[key] === 'object' ? item[key][0].label : item[key];
    7.         if(distDataMap[curKey]) {
    8.             distDataMap[curKey]++;
    9.         }else {
    10.             distDataMap[curKey] = 1;
    11.         }
    12.     })
    13.     // 生成目标数组
    14.     for(let key in distDataMap) {
    15.         distData.push({name: key, value: distDataMap[key]})
    16.     }
    17.     return distData
    18.   }

    此时我们只需要根据维度的字段, 即可获取某一维度的数据值, 后通过可视化组件渲染即可.

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

    我们用@ant-design/charts, 代码如下:

       
     
     
     
    1.     
    2.         分析纬度: 
    3.          handleAnazlyChange(0, v)} defaultValue={keys[0]}>
    4.             {
    5.                 keys.map((item,i) => {
    6.                     return { item }
    7.                 })
    8.             }
    9.         
    10.     
  •     {
  •         !!config && 
  •     }
  • 实现效果如下:

    6. 实现简单的 Table 编辑器

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

    大家感兴趣可以研究一下.

    最后

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

    网页标题:前端:如何让你的Table组件无限可能
    链接地址:http://www.shufengxianlan.com/qtweb/news48/41848.html

    网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

    广告

    声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联

    猜你还喜欢下面的内容

    建站公司知识

    行业网站建设