vue+elementui表格前端导出excel以及自定义导出样式

vue+elementui表格前端导出excel以及自定义导出样式
项目遇到前端自己导出表格,我自己把后端给我的数据处理了一波,合并重复列啊,以及表头合并啊。可没想到导出竟然也要前端自己来弄,于是乎请教同事以及博客。学到了学到了。
1,装包

npm install --save xlsx file-saver
npm install --save xlsx-style

2,引入

import FileSaver from "file-saver";
import XLSX2 from "xlsx";
import XLSX from "xlsx-style";

3,使用
页面效果:这个页面是根据数据,手动合并了一波的,自动合并重复列,以及表头的效果,所以导出还想要这个样子。
在这里插入图片描述

导出效果:
在这里插入图片描述
4,页面表格需要有id

<el-table
      :id="'mytable'"
      :highlight-current-row="highlightCurrentRow!==false?true:false"
      :data="tableDataWidhtOrder"
      ref="table"
      @select="select"
      @select-all="selectAll"
      @row-click="clickTr"
      @cell-click="clickTd"
      @selection-change="handleSelectionChange"
      @sort-change="sortChange"
      :row-class-name="tableRowClassName"
      :row-style="rowStyleCb"
      :height="height"
      :border="border"
    >
    </el-table>

5,事件

  setExport2Excel() {
    /* generate workbook object from table */
    var xlsxParam = { raw: true }   //这个保证表格只进行解析 不做运算
    var wb = XLSX2.utils.table_to_sheet(document.querySelector("#mytable"),xlsxParam);//mytable为表格的id名
    // if(!wb['!merges']){   //这个东西是当表格有合并的时候才会存在  并不能作为判断有无数据的标准
    //   this.$message.warning('无法导出:报表无数据');
    //   return
    // }
    for(var i = 0;i<11;i++){
      wb["!cols"][i]={wpx:130}
    }
    // 样式的文档地址
    // https://www.npmjs.com/package/xlsx-style
    for (const key in wb) {
       if(key.indexOf('!') === -1&&wb[key].v){
       wb[key].s ={
         font:{//字体设置
           sz:13,
           bold:false,
           color:{
             rgb:'000000'//十六进制,不带#
            }
         },
         alignment:{//文字居中
           horizontal:'center',
           vertical:'center',
           wrap_text:true
         },
         border: { // 设置边框
          top: { style: 'thin' },
          bottom: { style: 'thin' },
          left: { style: 'thin' },
          right: { style: 'thin' }
         }
       }
       }
    }
    var data = this.methods('addRangeBorder',wb['!merges'],wb) //合并项添加边框
    var filedata = this.methods('sheet2blob',data)
    this.methods('openDownloadDialog',filedata,this.todayTimeString + "-xxx报表.xlsx")
  },
  //为合并项添加边框
  addRangeBorder(range,ws){
    let arr = ["A","B","C","D","E","F","G","H","I","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z"];
    if(range){
      range.forEach(item=>{
        let startColNumber = Number(item.s.r), endColNumber = Number(item.e.r);
        let startRowNumber = Number(item.s.c), endRowNumber = Number(item.e.c);
        const test = ws[arr[startRowNumber] + (startColNumber + 1)];
        for(let col = startColNumber ; col <= endColNumber ; col++)
        {
          for(let row = startRowNumber; row <= endRowNumber ; row++)
          {
            ws[arr[row] + (col + 1)] = test;
          }
        }
      })
    }
    return ws;
  },
  //将一个sheet转成最终的excel文件的blob对象,然后利用URL.createObjectURL下载
  sheet2blob(sheet, sheetName) {
    sheetName = sheetName || 'sheet1';
    var workbook = {
      SheetNames: [sheetName],
      Sheets: {}
    };
    workbook.Sheets[sheetName] = sheet; // 生成excel的配置项
  
    var wopts = {
      bookType: 'xlsx', // 要生成的文件类型
      bookSST: false, // 是否生成Shared String Table,官方解释是,如果开启生成速度会下降,但在低版本IOS设备上有更好的兼容性
      type: 'binary'
    };
    var wbout = XLSX.write(workbook, wopts);
    var blob = new Blob([s2ab(wbout)], {
      type: "application/octet-stream"
    }); // 字符串转ArrayBuffer
    function s2ab(s) {
      var buf = new ArrayBuffer(s.length);
      var view = new Uint8Array(buf);
      for (var i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF;
      return buf;
    }
    return blob;
  },
  openDownloadDialog(url, saveName) {
    if (typeof url == 'object' && url instanceof Blob) {
      url = URL.createObjectURL(url); // 创建blob地址
    }
    var aLink = document.createElement('a');
    aLink.href = url;
    aLink.download = saveName || ''; // HTML5新增的属性,指定保存文件名,可以不要后缀,注意,file:///模式下不会生效
    var event;
    if (window.MouseEvent) event = new MouseEvent('click');
    else {
      event = document.createEvent('MouseEvents');
      event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
    }
    aLink.dispatchEvent(event);
  }

下面是一个vue的完整示例,自己的vue安装xlsx以及slsx-style可能会遇到一个报错,可能与安装或者cnpm镜像有关,修改他的源码
因为引入这个报错:

import XLSX from “xlsx-style”

报错如下:

This relative module was not found: ./cptable in ./node_modules/xlsx-style@0.8.13@xlsx-style/dist/cpexcel.js

解决方法:找到在\node_modules\xlsx-style\dist\cpexcel.js 807行 的

var cpt = require('./cpt' + 'able');

改成:

var cpt = cptable;

页面效果:
在这里插入图片描述

导出excel:
在这里插入图片描述

<template>
  <div>
    <button @click="setExport2Excel">导出</button>
    <el-table
      :data="tableData3"
      :id="'mytable'"
      border
      style="width: 100%">
      <el-table-column
        prop="date"
        label="门架"
        width="150">
      </el-table-column>
      <el-table-column
        prop="date"
        label="识别车型"
        width="150">
      </el-table-column>
      <el-table-column label="车型不一致">
        <el-table-column
          prop="province"
          label="问题车数量"
          width="120">
        </el-table-column>
        <el-table-column
          prop="city"
          label="差额"
          width="120">
        </el-table-column>
      </el-table-column>
      <el-table-column label="交易缺失">
        <el-table-column
          prop="province"
          label="问题车数量"
          width="120">
        </el-table-column>
        <el-table-column
          prop="city"
          label="差额"
          width="120">
        </el-table-column>
      </el-table-column>
      <el-table-column label="清分缺失">
        <el-table-column
          prop="province"
          label="问题车数量"
          width="120">
        </el-table-column>
        <el-table-column
          prop="city"
          label="差额"
          width="120">
        </el-table-column>
      </el-table-column>
      <el-table-column
        prop="date"
        label="差额统计"
        width="150">
      </el-table-column>
      <el-table-column label="配送信息">
        <el-table-column
          prop="name"
          label="姓名"
          width="120">
        </el-table-column>
        <el-table-column label="地址">
          <el-table-column
            prop="province"
            label="省份"
            width="120">
          </el-table-column>
          <el-table-column
            prop="city"
            label="市区"
            width="120">
          </el-table-column>
        </el-table-column>
      </el-table-column>
    </el-table>
  </div>
 
</template>

<script>
  // import FileSaver from 'file-saver'
  import XLSX2 from "xlsx";
  import XLSX from "xlsx-style";
  export default {
    data() {
      return {
        tableData3: [{
          date: '2016-05-03',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }, {
          date: '2016-05-02',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }, {
          date: '2016-05-04',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }, {
          date: '2016-05-01',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }, {
          date: '2016-05-08',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }, {
          date: '2016-05-06',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }, {
          date: '2016-05-07',
          name: '王小虎',
          province: '上海',
          city: '普陀区',
          address: '上海市普陀区金沙江路 1518 弄',
          zip: 200333
        }]
      }
    },
    methods:{
      setExport2Excel() {
        /* generate workbook object from table */
        var wb = XLSX2.utils.table_to_sheet(document.querySelector("#mytable"));//mytable为表格的id名
        if(!wb['!merges']){
          this.$message.warning('无法导出:报表无数据');
          return
        }
        for(var i = 0;i<11;i++){
          wb["!cols"][i]={wpx:130}
        }
        // 样式的文档地址
        // https://www.npmjs.com/package/xlsx-style
        for (const key in wb) {
          if(key.indexOf('!') === -1&&wb[key].v){
          wb[key].s ={
            font:{//字体设置
              sz:13,
              bold:false,
              color:{
                rgb:'000000'//十六进制,不带#
                }
            },
            alignment:{//文字居中
              horizontal:'center',
              vertical:'center',
              wrap_text:true
            },
            border: { // 设置边框
              top: { style: 'thin' },
              bottom: { style: 'thin' },
              left: { style: 'thin' },
              right: { style: 'thin' }
            }
          }
          }
        }
        var data = this.addRangeBorder(wb['!merges'],wb) //合并项添加边框
        var filedata = this.sheet2blob(data)
        this.openDownloadDialog(filedata,this.todayTimeString + "-xxx报表.xlsx")
      },
      //为合并项添加边框
      addRangeBorder(range,ws){
        let arr = ["A","B","C","D","E","F","G","H","I","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z"];
        
        range.forEach(item=>{
          let startColNumber = Number(item.s.r), endColNumber = Number(item.e.r);
          let startRowNumber = Number(item.s.c), endRowNumber = Number(item.e.c);
          const test = ws[arr[startRowNumber] + (startColNumber + 1)];
          for(let col = startColNumber ; col <= endColNumber ; col++)
          {
            for(let row = startRowNumber; row <= endRowNumber ; row++)
            {
              ws[arr[row] + (col + 1)] = test;
            }
          }
        })
        return ws;
      },
      //将一个sheet转成最终的excel文件的blob对象,然后利用URL.createObjectURL下载
      sheet2blob(sheet, sheetName) {
        sheetName = sheetName || 'sheet1';
        var workbook = {
          SheetNames: [sheetName],
          Sheets: {}
        };
        workbook.Sheets[sheetName] = sheet; // 生成excel的配置项
      
        var wopts = {
          bookType: 'xlsx', // 要生成的文件类型
          bookSST: false, // 是否生成Shared String Table,官方解释是,如果开启生成速度会下降,但在低版本IOS设备上有更好的兼容性
          type: 'binary'
        };
        var wbout = XLSX.write(workbook, wopts);
        var blob = new Blob([s2ab(wbout)], {
          type: "application/octet-stream"
        }); // 字符串转ArrayBuffer
        function s2ab(s) {
          var buf = new ArrayBuffer(s.length);
          var view = new Uint8Array(buf);
          for (var i = 0; i != s.length; ++i) view[i] = s.charCodeAt(i) & 0xFF;
          return buf;
        }
        return blob;
      },
      openDownloadDialog(url, saveName) {
        if (typeof url == 'object' && url instanceof Blob) {
          url = URL.createObjectURL(url); // 创建blob地址
        }
        var aLink = document.createElement('a');
        aLink.href = url;
        aLink.download = saveName || ''; // HTML5新增的属性,指定保存文件名,可以不要后缀,注意,file:///模式下不会生效
        var event;
        if (window.MouseEvent) event = new MouseEvent('click');
        else {
          event = document.createEvent('MouseEvents');
          event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
        }
        aLink.dispatchEvent(event);
      }
    }
  }
</script>
  • 18
    点赞
  • 78
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 27
    评论
您可以使用 `xlsx` 库和 `FileSaver.js` 库来实现Vue和Element UI导出Excel文件的功能。以下是一个简单的示例: 1. 首先,安装 `xlsx` 和 `file-saver` 包: ```bash npm install xlsx file-saver ``` 2. 在 Vue 组件中引入所需的库: ```javascript import XLSX from 'xlsx'; import { saveAs } from 'file-saver'; ``` 3. 创建一个方法来导出Excel文件: ```javascript methods: { exportToExcel() { // 创建工作簿和工作表 const workbook = XLSX.utils.book_new(); const worksheet = XLSX.utils.json_to_sheet(this.data); // 将工作表添加到工作簿 XLSX.utils.book_append_sheet(workbook, worksheet, 'Sheet1'); // 生成Excel文件的二进制数据 const excelBuffer = XLSX.write(workbook, { bookType: 'xlsx', type: 'array' }); // 将二进制数据转换为Blob对象 const blob = new Blob([excelBuffer], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' }); // 使用FileSaver.js保存文件 saveAs(blob, 'data.xlsx'); } } ``` 在上面的代码中,`this.data` 是要导出的数据,可以是一个数组或对象。 4. 在模板中添加一个按钮来触发导出操作: ```html <template> <div> <button @click="exportToExcel">导出Excel</button> </div> </template> ``` 这样,当用户点击“导出Excel”按钮时,将会触发 `exportToExcel` 方法,并导出Excel文件。 请注意,为了使上述代码正常工作,您需要在项目中正确引入 `xlsx` 和 `file-saver` 库,并将相关的样式和依赖项加载到您的项目中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 27
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

_小郑有点困了

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值