vue3上传excel并在线预览

embedded/2024/10/18 10:25:45/

目录

前言

安装 xlsx 依赖

XLSX.utils.sheet_to_html

XLSX.utils.sheet_to_json

前言

关于实现excel文档在线预览的做法,一种方式是通过讲文档里的数据处理成html,一种是将文档处理成图片进行预览,这里使用的是第一种。

安装 xlsx 依赖

npm i xlsx --save

 XLSX.utils.sheet_to_html

使用XLSX.utils.sheet_to_html方法将工作表(workSheet)转化成html表格的字符串表示,然后显示在前端页面,按照表格的行列结构进行排序,从而实现在线预览的效果

javascript">import * as XLSX from 'xlsx'export const file2Preview = (file: File) => {return new Promise(function (resolve) {const reader = new FileReader();// 文件加载事件reader.onload = function (e) {const data = e.target?.result;const workbook = XLSX.read(data, {type: "binary", cellDates: true,});// 拿第-个sheetconst worksheet = workbook.Sheets[workbook.SheetNames[0]];const html = XLSX.utils.sheet_to_html(worksheet);// 返回htmlresolve(html);};reader.readAsBinaryString(file);});
};

调用实现 

javascript"><template><n-uploadv-model:file-list="fileList"action="https://www.mocky.io/v2/5e4bafc63100007100d8b70f"@change="handleUploadChange"@remove="handleRemove"@update:file-list="handleFileListChange"><n-button>上传文件</n-button></n-upload><div class="flex-center preview-table" id="previewTable"></div></template>
<script lang="ts" setup>
import { defineComponent, ref } from 'vue'
import { useMessage } from 'naive-ui'
import type { UploadFileInfo } from 'naive-ui'
import { NButton } from "naive-ui";
import {file2Preview} from '@main/utils/index';const message = useMessage()const fileListRef = ref<UploadFileInfo[]>([])const fileList = ref(fileListRef)const handleUploadChange = async (data: { fileList: UploadFileInfo[] }) => {const html= await file2Preview(data.fileList[0].file as File);const previewTable = document.getElementById('previewTable');}const handleRemove = (data: { file: UploadFileInfo, fileList: UploadFileInfo[] }) => {if (data.file.id === 'text-message') {message.info('居然没传上去,算了,删了吧')}else if (data.file.id === 'notification') {message.error('不行,这个有用,不许删')return false}else if (data.file.id === 'contact') {message.loading('不知道这个有没有用,等我问问服务器能不能删', {duration: 4000})return new Promise((resolve) => {setTimeout(() => {message.error('不行,他们也不许删这个')resolve(false)}, 4000)})}}const handleFileListChange = () => {message.info('是的,file-list 的值变了')}</script><style lang="less" scoped>::v-deep table {border: 1px solid #000 !important;margin-top: 10px;border-collapse: collapse;}::v-deep th {border: 1px solid #000 !important;}::v-deep tr {border: 1px solid #000 !important;}::v-deep td {border: 1px solid #000 !important;text-align: center;min-width:20px;height: 20px;line-height: 20px;padding: 4px;color:#3e3e3e;
}
</style>

 

XLSX.utils.sheet_to_json

XLSX.utils.sheet_to_json用于将excel表格中的工作表(sheet)转换成JSON格式的函数,使用组件显示

javascript">export const file2Data = (file: File)  => {return new Promise((resolve, reject) => {const reader = new FileReader();reader.onload = function (e) {const data = e.target?.result;if (!data) {reject(new Error("Failed to read file data"));return;}try {const workbook = XLSX.read(data, { type: "binary" });const result: any[] = [];workbook.SheetNames.forEach((sheetName) => {const sheet = workbook.Sheets[sheetName];const jsonData = XLSX.utils.sheet_to_json(sheet, { header: 1, raw: false });if (jsonData.length > 0) {result.push({sheetName,sheet: jsonData});}});console.log(result); // 输出解析后的数据resolve(result);} catch (error) {reject(error);}};reader.onerror = function (e) {reject(new Error("FileReader error: " + e.target?.error?.message));};reader.readAsBinaryString(file);});
};

调用实现

javascript"><script lang="ts" setup>
import { defineComponent, ref } from 'vue'
import { useMessage } from 'naive-ui'
import type { UploadFileInfo } from 'naive-ui'
import { NButton } from "naive-ui";
import {file2Data} from '@main/utils/index';const message = useMessage()
const fileListRef = ref<UploadFileInfo[]>([])const datas = ref({source:[],columns:[]
})const fileList = ref(fileListRef)
const handleUploadChange = async (data: { fileList: UploadFileInfo[] }) => {fileListRef.value = data.fileList;const excelData = await file2Data(data.fileList[0].file as File)datas.value.source = excelData[0].sheet;datas.value.columns = [];Object.keys(datas.value.source[0]).map((v) => {datas.value.columns = [...datas.value.columns,{title: datas.value.source[0][v],key: v}]})datas.value.source = datas.value.source.slice(1);console.log(datas.value);
}const handleRemove = (data: { file: UploadFileInfo, fileList: UploadFileInfo[] }) => {if (data.file.id === 'text-message') {message.info('居然没传上去,算了,删了吧')}else if (data.file.id === 'notification') {message.error('不行,这个有用,不许删')return false}else if (data.file.id === 'contact') {message.loading('不知道这个有没有用,等我问问服务器能不能删', {duration: 4000})return new Promise((resolve) => {setTimeout(() => {message.error('不行,他们也不许删这个')resolve(false)}, 4000)})}
}
const handleFileListChange = () => {message.info('是的,file-list 的值变了')
}</script><template><n-uploadv-model:file-list="fileList"action="https://www.mocky.io/v2/5e4bafc63100007100d8b70f"@change="handleUploadChange"@remove="handleRemove"@update:file-list="handleFileListChange"><n-button>上传文件</n-button></n-upload><n-data-tableid="previewTable"class="preview-table":columns="datas.columns":data="datas.source":pagination="false":bordered="false"/></template><style lang="less" scoped>::v-deep table {border: 1px solid #000 !important;margin-top: 10px;border-collapse: collapse;}::v-deep th {border: 1px solid #000 !important;}::v-deep tr {border: 1px solid #000 !important;}::v-deep td {border: 1px solid #000 !important;text-align: center;min-width:20px;height: 20px;line-height: 20px;padding: 4px;color:#3e3e3e;
}
</style>


http://www.ppmy.cn/embedded/99371.html

相关文章

合宙LuatOS AIR700 IPV6 TCP 客户端向NodeRed发送数据

为了验证 AIR700 IPV6 &#xff0c;特别新建向NodeRed Tcp发送的工程。 Air700发送TCP数据源码如下&#xff1a; --[[ IPv6客户端演示, 仅EC618系列支持, 例如Air780E/Air600E/Air780UG/Air700E ]]-- LuaTools需要PROJECT和VERSION这两个信息 PROJECT "IPV6_SendDate_N…

bug----jdk17使用JOL无法输出java对象的存储布局

项目场景&#xff1a; 提示&#xff1a;这里简述项目相关背景&#xff1a; 在idea中以jdk17为开发环境&#xff0c;使用 jol-core.016 版本没办法输出java 对象的存储结构信息&#xff0c;会让程序崩溃。 问题描述 提示&#xff1a;这里描述项目中遇到的问题&#xff1a; 这…

在Uni-app中实现计时器效果

<template><div class"timer"><p>{{ formatTime }}</p><button click"startTimer" v-if"!isTiming">开始计时</button><button click"stopTimer" v-else>停止计时</button></div&…

appium下载及安装

下载地址&#xff1a;https://github.com/appium/appium-desktop/releases 双击安装就可以

基于矢量控制器的PMSM永磁同步电机速度控制系统simulink建模与仿真

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 5.完整工程文件 1.课题概述 基于矢量控制器的PMSM永磁同步电机速度控制系统simulink建模与仿真&#xff0c;仿真输出电机转速跟踪曲线&#xff0c;PID控制器输出曲线以及Te输出曲线。 2.系统仿真结果 &…

【C++ 面试 - 面向对象】每日 3 题(六)

✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏&…

Prometheus Alertmanager告警之邮件、钉钉群、企业微信群机器人报警

文章目录 一、部署alertmanager相关组件1.alertmanager-config2.alertmanager-message-tmpl3.alertmanager 二、调试邮件告警三、钉钉群/企业微信群 报警3.1添加钉钉群机器人3.2添加企业微信群机器人3.3部署alertmanager-webhook-adaptermessage-tmplalertmanager-webhook-adap…

Prometheus 服务发现

目录 1.基于文件的服务发现 2.基于 Consul 的服务发现 3.基于 Kubernetes API 的服务发现 1.基于文件的服务发现 基于文件的服务发现是仅仅略优于静态配置的服务发现方式&#xff0c;它不依赖于任何平台或第三方服务&#xff0c;因而也是最为简单和通用的实现方式。 Prometh…