vue3 導出數據為 excel 文件

文章目錄

  • 安裝插件
  • 封裝組件 -- Export2Excel.js
  • 多表封裝
  • 界面使用 -- 數據處理成二維數組
  • 更多

菜鳥最近做了一個需求,就是需要上傳表單并識別,然后識別出來的內容要可以修改,然后想的就是識別內容變成 form 表單,所以并沒有使用 SpreadJS ,這個 SpreadJS 有很多 excel 的功能,強大是強大,就是我這個需求沒有那么復雜!

然后上傳識別交給后端處理了,菜鳥做的就是查看詳情的是時候可把表單展示的數據導出成 excel 文件就行!

安裝插件

npm i xlsx
npm install -S file-saver

封裝組件 – Export2Excel.js

import { saveAs } from 'file-saver'
import * as XLSX from 'xlsx'/**** @param {Object} workbook 工作薄* @param {Object} worksheet 工作表* @param {Object} cell 單元格* 標記,引用單元格時所使用的地址格式(如:A1、C7)*/
function datenum (v, date1904) {if (date1904) v += 1462var epoch = Date.parse(v)return (epoch - new Date(Date.UTC(1899, 11, 30))) / (24 * 60 * 60 * 1000)
}function sheetFromArrayOfArrays (data, opts) {var ws = {}var range = {s: {c: 10000000,r: 10000000},e: {c: 0,r: 0}}for (var R = 0; R !== data.length; ++R) {for (var C = 0; C !== data[R].length; ++C) {if (range.s.r > R) range.s.r = Rif (range.s.c > C) range.s.c = Cif (range.e.r < R) range.e.r = Rif (range.e.c < C) range.e.c = Cvar cell = {v: data[R][C] // v表示單元格原始值, t表示內容類型,s-string類型,n-number類型,b-boolean類型,d-date類型,等等}if (cell.v == null) continue/*** 通過地址對象 { r: R, c: C } 來獲取單元格,R 和 C 分別代表從 0 開始的行和列的索引。* XLSX.utils 中的 encode_cell/decode_cell 方法可以轉換單元格地址*    XLSX.utils.encode_cell({ r: 7, c: 2 })  ===》 C7*/var cellRef = XLSX.utils.encode_cell({ c: C, r: R })if (typeof cell.v === 'number') cell.t = 'n'else if (typeof cell.v === 'boolean') cell.t = 'b'else if (cell.v instanceof Date) {cell.t = 'n'cell.z = XLSX.SSF._table[14]cell.v = datenum(cell.v)} else cell.t = 's'ws[cellRef] = cell}}// ws['!ref']:表示所有單元格的范圍,例如從A1到F8則記錄為A1:F8if (range.s.c < 10000000) ws['!ref'] = XLSX.utils.encode_range(range)return ws
}function Workbook () {if (!(this instanceof Workbook)) return new Workbook()this.SheetNames = []this.Sheets = {}
}// 字符串轉為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) & 0xFFreturn buf
}/**** @param {Array} multiHeader  多行表頭* @param {Array} header  表頭* @param {Array} data  數據* @param {String} filename  文件名* @param {Array} merges  合并單元格* @param {Boolean} autoWidth  是否設置單元格寬度* @param {String} bookType  要生成的文件類型*/
export function exportJsonToExcel ({multiHeader = [],header,data,filename,merges = [],autoWidth = true,bookType = 'xlsx'
} = {}) {filename = filename || 'excel-list'data = [...data]data.unshift(header)for (let i = multiHeader.length - 1; i > -1; i--) {data.unshift(multiHeader[i])}var wsName = 'SheetJS'var wb = new Workbook()var ws = sheetFromArrayOfArrays(data)if (merges.length > 0) {// ws[!merges]:存放一些單元格合并信息,是一個數組,每個數組由包含s和e構成的對象組成,s表示開始,e表示結束,r表示行,c表示列if (!ws['!merges']) ws['!merges'] = []merges.forEach(item => {ws['!merges'].push(XLSX.utils.decode_range(item))})}if (autoWidth) {/* 設置worksheet每列的最大寬度 */const colWidth = data.map(row => row.map(val => {/* 先判斷是否為null/undefined */if (val == null) {return { 'wch': 10 }} else if (val.toString().charCodeAt(0) > 255) { /* 再判斷是否為中文 */return {'wch': val.toString().length * 2}} else {return {'wch': val.toString().length}}}))/* 以第一行為初始值 */let result = colWidth[0]for (let i = 1; i < colWidth.length; i++) {for (let j = 0; j < colWidth[i].length; j++) {if (result[j] && result[j]['wch'] < colWidth[i][j]['wch']) {result[j]['wch'] = colWidth[i][j]['wch']}}}// ws['!cols']設置單元格寬度, [{'wch': 10},{'wch': 10}] ===> 第一列和第二列設置了寬度ws['!cols'] = result}/* add worksheet to workbook */wb.SheetNames.push(wsName)wb.Sheets[wsName] = wsvar wbout = XLSX.write(wb, {bookType: bookType,bookSST: false, // 是否生成Shared String Table,官方解釋是,如果開啟生成速度會下降,但在低版本IOS設備上有更好的兼容性type: 'binary'})saveAs(new Blob([s2ab(wbout)], { type: 'application/octet-stream' }),`${filename}.${bookType}`)
}

此處參考:vue3中將數據導出為excel表格

多表封裝

但是菜鳥的需求是要一個 excel 包含兩個表,所以菜鳥對其進行了優化:

import { saveAs } from "file-saver";
import * as XLSX from "xlsx";/**** @param {Object} workbook 工作薄* @param {Object} worksheet 工作表* @param {Object} cell 單元格* 標記,引用單元格時所使用的地址格式(如:A1、C7)*/
function datenum(v, date1904) {if (date1904) v += 1462;var epoch = Date.parse(v);return (epoch - new Date(Date.UTC(1899, 11, 30))) / (24 * 60 * 60 * 1000);
}// eslint-disable-next-line
function sheetFromArrayOfArrays(data, opts) {var ws = {};var range = {s: {c: 10000000,r: 10000000,},e: {c: 0,r: 0,},};for (var R = 0; R !== data.length; ++R) {for (var C = 0; C !== data[R].length; ++C) {if (range.s.r > R) range.s.r = R;if (range.s.c > C) range.s.c = C;if (range.e.r < R) range.e.r = R;if (range.e.c < C) range.e.c = C;var cell = {v: data[R][C], // v表示單元格原始值, t表示內容類型,s-string類型,n-number類型,b-boolean類型,d-date類型,等等};if (cell.v == null) continue;/*** 通過地址對象 { r: R, c: C } 來獲取單元格,R 和 C 分別代表從 0 開始的行和列的索引。* XLSX.utils 中的 encode_cell/decode_cell 方法可以轉換單元格地址*    XLSX.utils.encode_cell({ r: 7, c: 2 })  ===》 C7*/var cellRef = XLSX.utils.encode_cell({ c: C, r: R });if (typeof cell.v === "number") cell.t = "n";else if (typeof cell.v === "boolean") cell.t = "b";else if (cell.v instanceof Date) {cell.t = "n";cell.z = XLSX.SSF._table[14];cell.v = datenum(cell.v);} else cell.t = "s";ws[cellRef] = cell;}}// ws['!ref']:表示所有單元格的范圍,例如從A1到F8則記錄為A1:F8if (range.s.c < 10000000) ws["!ref"] = XLSX.utils.encode_range(range);return ws;
}function Workbook() {if (!(this instanceof Workbook)) return new Workbook();this.SheetNames = [];this.Sheets = {};
}// 字符串轉為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;
}/**** @param {Array} multiHeader  多行表頭* @param {Array} header  表頭* @param {Array} data  數據* @param {String} filename  文件名* @param {Array} merges  合并單元格* @param {Boolean} autoWidth  是否設置單元格寬度* @param {String} bookType  要生成的文件類型*/
export function exportJsonToExcel({multiHeader = [],header,data,filename,sheetname,merges = [],autoWidth = true,bookType = "xlsx",
} = {}) {filename = filename || "excel-list";for (let i in header) {data[i] = [...data[i]];data[i].unshift(header[i]);}for (let j in data) {for (let i = multiHeader.length - 1; i > -1; i--) {data[j].unshift(multiHeader[i]);}}let wsName = [];for (let i in sheetname) {wsName.push(sheetname[i]);}var wb = new Workbook();let ws = [];for (let i in data) {ws.push(sheetFromArrayOfArrays(data[i]));}if (merges.length > 0) {// ws[!merges]:存放一些單元格合并信息,是一個數組,每個數組由包含s和e構成的對象組成,s表示開始,e表示結束,r表示行,c表示列for (let i in ws) {if (!ws[i]["!merges"]) ws["!merges"] = [];merges.forEach((item) => {ws[i]["!merges"].push(XLSX.utils.decode_range(item));});}}if (autoWidth) {/* 設置worksheet每列的最大寬度 */const colWidth = data.map((row) =>row.map((val) => {/* 先判斷是否為null/undefined */if (val == null) {return { wch: 10 };} else if (val.toString().charCodeAt(0) > 255) {/* 再判斷是否為中文 */return {wch: val.toString().length * 2,};} else {return {wch: val.toString().length,};}}));/* 以第一行為初始值 */let result = colWidth[0];for (let i = 1; i < colWidth.length; i++) {for (let j = 0; j < colWidth[i].length; j++) {if (result[j] && result[j]["wch"] < colWidth[i][j]["wch"]) {result[j]["wch"] = colWidth[i][j]["wch"];}}}// ws['!cols']設置單元格寬度, [{'wch': 10},{'wch': 10}] ===> 第一列和第二列設置了寬度for (let i in ws) {ws[i]["!cols"] = result;}}/* add worksheet to workbook */for (let i in wsName) {wb.SheetNames.push(wsName[i]);wb.Sheets[wsName[i]] = ws[i];}var wbout = XLSX.write(wb, {bookType: bookType,bookSST: false, // 是否生成Shared String Table,官方解釋是,如果開啟生成速度會下降,但在低版本IOS設備上有更好的兼容性type: "binary",});saveAs(new Blob([s2ab(wbout)], { type: "application/octet-stream" }), `${filename}.${bookType}`);
}

界面使用 – 數據處理成二維數組

第一個的使用直接參考別人的,我這里使用的是菜鳥封裝的,如果有誤,可以留言,菜鳥進行更改,因為菜鳥這里只測試了一個和兩個表格的情況

// 數據處理  --》 讀者按照自己的邏輯來
for (const key in envFactors.value) {let data1 = []; //設置一個二級列表,導出的表格每一行為一個二級列表for (let i in objkeyArr.value) {let keyname = objkeyArr.value[i];data1.push(envFactors.value[key][keyname]);}data2.value.push(data1); //把二級列表塞入一級列表
}
console.log(data2); // data2 必須是二維數組for (const key in sampleGroups.value) {let data1 = [];data1.push(sampleGroups.value[key].sampleTestName,sampleGroups.value[key].sampleAnalysisName,sampleGroups.value[key].groupFirstWay,sampleGroups.value[key].groupSecWay,sampleGroups.value[key].groupThirdWay,sampleGroups.value[key].groupFourthWay,sampleGroups.value[key].groupFifthWay);data3.value.push(data1);
}
console.log(data3);
// 數據處理結束// 導出成表格
function exportExcel() {exportJsonToExcel({header: [["檢測報告樣品名稱","分析樣品名稱","分組名稱","分組名稱","分組名稱","分組名稱","分組名稱",],objkeyArr.value,], // 表名個數可以不等于數據每一行的列數filename: "詳請表",data: [data3.value, data2.value],sheetname: ["樣品分組表", "環境因子表"],});
}

更多

菜鳥這里只使用了header、data、filename、sheetname 等屬性,其他屬性可以看該篇文章,挺詳細的:純前端js(或者vue)導出excel實現:合并單元格、設置單元格樣式、單元格內換行

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/167287.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/167287.shtml
英文地址,請注明出處:http://en.pswp.cn/news/167287.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

反爬蟲機制與反爬蟲技術(二)

反爬蟲機制與反爬蟲技術二 1、動態頁面處理與驗證碼識別概述2、反爬蟲案例:頁面登錄與滑塊驗證碼處理2.1、用例簡介2.2、庫(模塊)簡介2.3、網頁分析2.4、Selenium準備操作2.5、頁面登錄2.6、模糊移動滑塊測試3、滑塊驗證碼處理:精確移動滑塊3.1、精確移動滑塊的原理3.2、滑…

【模塊補充】importlib

importlib 【一】介紹 importlib 模塊是 Python 中用于動態加載和導入模塊的內置模塊。它提供了一組函數和類&#xff0c;使得我們可以在運行時根據需要加載模塊&#xff0c;并且可以對已導入的模塊進行操作和管理。 【二】詳解及示例&#xff1a; 【1】動態加載模塊&#…

PyQt6簡介

鋒哥原創的PyQt6視頻教程&#xff1a; 2024版 PyQt6 Python桌面開發 視頻教程(無廢話版) 玩命更新中~_嗶哩嗶哩_bilibili2024版 PyQt6 Python桌面開發 視頻教程(無廢話版) 玩命更新中~共計12條視頻&#xff0c;包括&#xff1a;2024版 PyQt6 Python桌面開發 視頻教程(無廢話版…

企業遠程訪問業務系統:對比MPLS專線,貝銳蒲公英為何更優優勢?

如今&#xff0c;企業大多都會采用OA、ERP、CRM等各種數字化業務系統。 私有云、公有云混合架構也變得越來越常見。 比如&#xff1a;研發系統部署在公司本地私有云、確保數據安全&#xff0c;OA采用公有云方案、滿足隨時隨地訪問需求。 如此一來&#xff0c;也產生了遠程訪問…

js前端跨屏效果

效果: 三個球 源碼: <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>三個球</title> </h…

js實現圖片懶加載

方式一&#xff1a;html實現 在img標簽加上 loading"lazy" 方式二&#xff1a;js實現 通過js監聽頁面的滾動&#xff0c;實現的原理主要是判斷當前圖片是否到了可視區域&#xff1a; 拿到所有的圖片 dom 。遍歷每個圖片判斷當前圖片是否到了可視區范圍內。如果到了…

Maven項目下詳細的SSM整合流程

文章目錄 &#x1f389;SSM整合流程一、兩個容器整合? 1、先準備好數據庫config.properties連接、mybatis-config.xml&#x1f38a; 2、容器一&#xff1a;優先配置spring.xml文件&#x1f38a; 3、容器二&#xff1a;配置springMVC.xml文件&#x1f38a; 4、Tomcat整合spring…

解釋PCIe MSI 中斷要求中斷向量連續?PCIe 規范里并沒有明確指出

MSI 向量必須連續&#xff1f; 前言 MSI 物理條件&#xff0c;MSI 中斷產生的邏輯是RC初始化的時候&#xff0c;由軟件將配置寫入到 EP 的 2 個寄存器中&#xff0c;這兩個寄存器一個指示的是地址 Message Address&#xff0c;一個指示的是數據 Message Data。當 EP 試圖觸發…

你再不學Git就來不及了!!!

其他系列文章導航 設計模式合集 多線程合集 分布式合集 ES合集 文章目錄 其他系列文章導航 文章目錄 前言 版本控制 什么是版本控制 為什么要版本控制 一、認識 Git 1.1Git 簡史 1.2Git 與其他版本管理系統的主要區別 1.3Git 的三種狀態 二、Git 使用快速入門 2.1獲…

springboot使用redis緩存亂碼(key或者 value 亂碼)一招解決

如果查看redis中的值是這樣 創建一個配置類就可以解決 package com.deka.config;import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; i…

CPU+GPU多樣化算力,ZStack Cloud助力游戲精釀核心業務上云

游戲精釀通過ZStack Cloud云平臺提供高性能、高可用的云主機、云存儲和云網絡&#xff1b;前期通過超融合架構快速構建云基礎設施&#xff0c;來支持Jira、Redis等關鍵業務&#xff1b;并實現對原有私有云平臺業務的替代&#xff0c;按需將原有私有云業務滾動遷移到ZStack Clou…

移動端瀏覽器 jquery 獲取 pdf blob文件流 預覽pdf

最近遇到一個需求&#xff0c;一個古早的移動端 juery 項目要求做一個頁面&#xff0c;從接口獲取 pdf 文件流&#xff0c;然后預覽出來 這里使用第三方工具&#xff1a;pdf.js 代碼如下&#xff1a; // 引入相關文件<script src"../js/pdf.js" type"text…

N_1 驗證密碼

N_1 驗證密碼 題目 設計一個用戶密碼驗證程序&#xff0c;要求密碼輸入只有3次機會&#xff0c;且密碼中不能包含”*”字符。 分析 需要考慮3個問題&#xff1a;驗證次數、特殊字符和正誤密碼判斷&#xff1b;驗證次數需要使用循環&#xff0c;3個問題需要用到分支結構&…

java 系統屬性和環境屬性

Java系統屬性和環境屬性都是與Java應用程序相關的參數&#xff0c;但它們有以下區別&#xff1a; 系統屬性是由Java虛擬機&#xff08;JVM&#xff09;設置的&#xff0c;而環境屬性是由操作系統設置的。 系統屬性是以“-D”開頭的命令行參數傳遞給JVM的&#xff0c;而環境屬性…

深入理解Spring AOP的工作流程

文章目錄 引言什么是AOP&#xff1f;Spring AOP的工作原理1. JDK動態代理2. CGLIB代理 Spring AOP的注解方式Aspect注解EnableAspectJAutoProxy注解 Spring AOP的工作流程拓展應用1. 自定義注解2. 異常處理3. 切面優先級 結論 &#x1f389;深入理解Spring AOP的工作流程 ☆* o…

關于運行軟件程序出現vcruntime140.dll丟失的修復教程-解決方案

vcruntime140.dll是Microsoft Visual C庫文件的一部分&#xff0c;用于支持Windows操作系統上的應用程序。如果找不到或丟失了這個文件&#xff0c;可能會導致某些應用程序無法正常運行。下面是關于vcruntime140.dll丟失的5個修復方法&#xff0c;以及vcruntime140.dll文件屬性…

Python基礎教程之分支結構詳解

文章目錄 一、分支結構二、單分支結構三、雙分支結構四、多分支結構五、嵌套分支結構六、三元表達式七、條件測試關于Python技術儲備一、Python所有方向的學習路線二、Python基礎學習視頻三、精品Python學習書籍四、Python工具包項目源碼合集①Python工具包②Python實戰案例③P…

Elasticsearch基礎優化

分片策略 分片和副本得設計為ES提供支付分布式和故障轉移得特性&#xff0c;但不意味著分片和副本是可以無限分配&#xff0c; 而且索引得分片完成分配后由于索引得路由機制&#xff0c;不能重新修改分片數&#xff08;副本數可以動態修改&#xff09; 一個分片得底層為一個l…

python之pyqt專欄2-項目文件解析

項目結構 在上一篇文章python之pyqt專欄1-環境搭建&#xff0c;創建新的pyqt項目&#xff0c;下面我們來看一下這個項目下的文件。 從下面的文件結構圖可以看到&#xff0c;該項目下有3個文件&#xff0c;untitled.ui,untitled.py 以及main.py。 QtDesigner可以UI界面的方式&am…

Feign接口請求返回異常 no suitable HttpMessageConvert found for response type

問題場景&#xff1a; 后端調用feign接口請求, 接口返回異常, no suitable HttpMessageConvert found for response type 問題描述 報錯異常如下&#xff1a; //根據圖片特征 去查詢人員信息ResultVo<List> personVos ipbdFaceLibPersonApi.queryFacePersonByFeatur…