当前位置:首页 » 《关于电脑》 » 正文

使用ExcelJS实现excel的前端导出功能(Vue3+TS)

7 人参与  2024年04月21日 15:48  分类 : 《关于电脑》  评论

点击全文阅读


ExcelJS:读取,操作并写入电子表格数据和样式到 XLSX 和 JSON 文件。一个 Excel 电子表格文件逆向工程项目。
github中文文档:https://github.com/exceljs/exceljs/blob/master/README_zh.md

 封装excel.ts工具文件

import ExcelJS from 'exceljs'// 导入参数数据类型export interface importExcelType {  // 第 i 张工作表  i?: number;  // 表格表头字段数组  header: readonly any[];}// 导出参数数据类型export interface exportExcelType {  // 数据  data: { [key: string]: any }[];  // 文件名  name: string;  // 表头字段  header: string[];  // 表头字段对应中文  customHeader: string[];  // 工作表名  sheetName?: string;  // 标题  title?: string;  // 小标题  subTitle?: string;  // 工作表保护密码  password?: string;  // 对齐方式  alignment?: Partial<ExcelJS.Alignment>;  // 合并单元格  mergeList?: mergeListType[];  // 标题样式  titleStyle?: Partial<ExcelJS.Row>;  // 小标题样式  subTitleStyle?: Partial<ExcelJS.Row>;  // 表头样式  headerStyle?: Partial<ExcelJS.Row>;  // 单元格统一样式  cellStyle?: Partial<ExcelJS.Row & ExcelJS.Column>;}// 合并单元格数据类型export interface mergeListType {  startRow: number;  startColumn: number;  endRow: number;  endColumn: number;}// exceljs相关方法export class Excel {  blob?: Blob; // 导入的blob文件  worksheet?: ExcelJS.Worksheet;  // 当前工作表  header: string[]; // 表头字段数组  constructor(blob?: Blob) {    this.blob = blob;    this.worksheet = undefined;    this.header = [];  }   // 导出excel,参数信息参考exceljs   // data 数据   // name 文件名   // header 表头字段   // customHeader 表头字段对应中文   // sheetName 工作表名   // title 标题   // subTitle 副标题(日期)   // password 冻结表格密码   // mergeList 合并单元格数组   // titleStyle 标题样式(按需补充方法)   // subTitleStyle 小标题样式(按需补充方法)   // headerStyle 表头字段样式(按需补充方法)   // cellStyle 单元格样式(按需补充方法)  public async exportExcel(options: exportExcelType): Promise<void> {    const {         data,         name,         header,         customHeader,         sheetName = 'sheet1',         title = '',         subTitle = '',         password = '',         mergeList = [],         titleStyle,         subTitleStyle,         headerStyle,         cellStyle } = options;    // 创建工作簿    const workbook = new ExcelJS.Workbook();    workbook.creator = '侃侃';    workbook.created = new Date();    // 添加sheet    this.worksheet = workbook.addWorksheet(sheetName, { properties: { tabColor: { argb: 'FF00FF00' } } });    this.header = header;    // 表头行序号    let headerRowId = 1;    if (!!title) headerRowId++;    if (!!subTitle) headerRowId++;    // 插入单元格数据    this.setCells(data, customHeader, cellStyle);    // 插入大标题    this.getTitle(title, titleStyle);    // 插入小标题    this.getSubTitle(subTitle, subTitleStyle);    // 处理表头    this.setHeaderStyle(headerRowId, data, headerStyle);    // 更多处理    this.handleDealExcel(password, mergeList, headerRowId);    // 导出excel(此处也可用file-saver将blob导出到excel文件  fs.saveAs(blob, name+'.xlsx');)    workbook.xlsx.writeBuffer().then((data) => {      let blob = new Blob([data], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' });      const a = document.createElement('a');      a.href = URL.createObjectURL(blob);      a.download = name + '.xlsx';      document.body.appendChild(a);      a.click();      document.body.removeChild(a);      window.URL.revokeObjectURL(a.href);    });  }   // 合并单元格数组   // merge1 需要合并的行/列起止对象数组1   // 例:[{ startRow: 1, endRow: 1},{ startRow: 10, endRow: 10}]   // merge2 需要合并的列/行起止对象数组2   // 例:[{ startColumn: 1, endColumn: 2 }]   // mergeArr 合并后数组   // 例:[{ startRow: 1, endRow: 1, startColumn: 1, endColumn: 2},{ startRow: 10, endRow: 10, startColumn: 1, endColumn: 2}]  public merge(merge1: Partial<mergeListType>[], merge2: Partial<mergeListType>[]): mergeListType[] {    const mergeArr: any[] = [];    merge1.forEach(item1 => {      mergeArr.push(...merge2.map(item2 => {        return { ...item2, ...item1 };      }))    })    return mergeArr;  }   // 单元格数据处理    // data 表格数据   // customHeader  表头中文字段  private setCells(data: exportExcelType['data'], customHeader: string[], style?: Partial<ExcelJS.Row & ExcelJS.Column>): void {    // 设置列,插入中文表头    const column: Partial<ExcelJS.Column>[] = [];    this.header.forEach((item, index) => {      column.push({        header: customHeader[index],        key: item,        width: style?.width || 25,      })    })    this.worksheet!.columns = column;    // 设置行,添加数据    this.worksheet?.addRows(data);    // 设置行高    this.worksheet?.eachRow({ includeEmpty: true }, (row: any, rowNumber) => {      let maxHeight = 1      row.values.forEach((cur: any) => {        if (cur && typeof cur === 'string') {          const tmp = cur.split('\r\n') // 多行数据以此来作为分割设置行高          const tmpLen = tmp.length          maxHeight = Math.max(tmpLen, maxHeight)        }      })      row.height = style?.height || 20;    })    // 获取每一列数据,再依次对齐    this.worksheet!.columns.forEach(column => {        column.alignment = style?.alignment || { vertical: 'middle', horizontal: 'center', wrapText: true }    })  }   // 添加大标题    // title 标题   // style 标题样式  private getTitle(title: string, style?: Partial<ExcelJS.Row>): void {    if (!!title) {      // 插入标题      this.worksheet?.spliceRows(1, 0, [title]);      this.worksheet?.mergeCells(1, 1, 1, this.header.length);      // 调整标题样式      const titleRow = this.worksheet!.getRow(1);      // 高度      titleRow.height = style?.height || 40;      // 字体设置      titleRow.font = style?.font || {        size: 20,        bold: true      }      // 背景色      titleRow.fill = style?.fill || {        bgColor: {          'argb': 'FFFFFF00'        },        type: 'pattern',        pattern: 'none'      }      // 对齐方式      titleRow.alignment = style?.alignment || {        horizontal: "center",        vertical: "middle"      };    }  }   // 添加小标题    // subTitle 标题   // style 小标题样式    private getSubTitle(subTitle: string, style?: Partial<ExcelJS.Row>): void {    if (!!subTitle) {      this.worksheet?.spliceRows(2, 0, [subTitle]);      this.worksheet?.mergeCells(2, 1, 2, this.header.length);      // 调整标题样式      const subtitleRow = this.worksheet!.getRow(2);      // 高度      subtitleRow.height = style?.height || 20;      // 字体设置      subtitleRow.font = style?.font || {        size: 14,      }      // 背景色      subtitleRow.fill = style?.fill || {        bgColor: {          'argb': 'FFFFFF00'        },        type: 'pattern',        pattern: 'none'      }      // 对齐方式      subtitleRow.alignment = style?.alignment || {        horizontal: "right",        vertical: "middle"      };    }  }   // 设置表头样式   // num 表头在第几行   // data 总数据   // style 表头样式  private setHeaderStyle(num: number, data: any, style?: Partial<ExcelJS.Row>): void {    // 自动筛选器    this.worksheet!.autoFilter = {      from: {        row: num,        column: 1      },      to: {        row: data.length,        column: this.header.length      }    }    // 给表头添加背景色    let headerRow = this.worksheet!.getRow(num);    headerRow!.height = style?.height || 30;    // 通过 cell 设置背景色,更精准    headerRow?.eachCell((cell) => {      cell.fill = style?.fill || {        type: 'pattern',        pattern: 'solid',        fgColor: { argb: 'dde0e7' },      }      cell.font = style?.font || {        size: 12      }    })  }     // 其他内容处理   // password 密码   // mergeList 合并行列数组   // headerRowId  表头行序号  private handleDealExcel(password: string, mergeList: mergeListType[], headerRowId: number) {    // 添加工作表保护    if (!!password) {      this.worksheet?.protect(password, {        autoFilter: true,        selectLockedCells: false,      });    }    // 合并单元格    mergeList.forEach(item => {      // 行数为表格数据所在行行数+表头行序号headerRowId      const startRow = item.startRow + headerRowId;      const endRow = item.endRow + headerRowId;      this.worksheet?.mergeCells(startRow, item.startColumn, endRow, item.endColumn);    })    // 冻结前几行    this.worksheet!.views = [      { state: 'frozen', xSplit: 0, ySplit: headerRowId }    ];  }}

表格页面调用excel工具文件 

// export.vue<template>  <div class="export">    <button @click="exportExcel">导出</button>  </div></template><script setup lang="ts">import { Excel } from '@/utils/excel';import { reactive } from 'vue';const tableData = reactive([  { id: 1, name: "张三", age: 18 },  { id: 2, name: "张三", age: 18 }])const header = ['id', 'name', 'age'];const customHeader = ['序号', '姓名', '年龄'];const exportExcel = () => {  const excel = new Excel();  excel.exportExcel({    name: '表格文件名称',    // title: '表格标题',加入这个标题会导致导出的表格再次导入时数据key值获取不正确    data: tableData,    header,    customHeader  })}</script>


点击全文阅读


本文链接:http://zhangshiyu.com/post/98393.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

关于我们 | 我要投稿 | 免责申明

Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1