vue3上传excel并在线预览的实现

 更新时间:2025年12月17日 09:39:08   作者:Summer不秃  
文章介绍了通过将Excel文档的数据处理成HTML表格字符串,实现在线预览的方法,使用了XLSX库中的`sheet_to_html`和`sheet_to_json`函数将Excel工作表转换为HTML表格和JSON格式

前言

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

安装 xlsx 依赖

npm i xlsx --save

XLSX.utils.sheet_to_html

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

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,
      });
      // 拿第-个sheet
      const worksheet = workbook.Sheets[workbook.SheetNames[0]];
      const html = XLSX.utils.sheet_to_html(worksheet);
      // 返回html
      resolve(html);
    };
    reader.readAsBinaryString(file);
  });
};

调用实现 

<template>
    <n-upload
    v-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格式的函数,使用组件显示

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);
  });
};

调用实现

<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-upload
    v-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-table
    id="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>

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Vue异步更新机制及$nextTick原理的深入讲解

    Vue异步更新机制及$nextTick原理的深入讲解

    最近在学习一些底层方面的知识,所以想做个系列尝试去聊聊这些比较复杂又很重要的知识点,下面这篇文章主要给大家介绍了关于Vue异步更新机制及$nextTick原理的相关资料,需要的朋友可以参考下
    2022-04-04
  • vue 实现列表跳转至详情且能添加至购物车功能

    vue 实现列表跳转至详情且能添加至购物车功能

    列表页面显示数据,点击跳转到对应的详情页,详情页可以添加并跳转到购物车,购物车具有常见功能,这篇文章主要介绍了vue 实现列表跳转至详情且能添加至购物车,需要的朋友可以参考下
    2022-10-10
  • vue3 elementplus table合并写法

    vue3 elementplus table合并写法

    这篇文章主要介绍了vue3 elementplus table合并写法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-06-06
  • 如何在vue3.0+中使用tinymce及实现多图上传文件上传公式编辑功能

    如何在vue3.0+中使用tinymce及实现多图上传文件上传公式编辑功能

    本文给大家分享tinymce编辑器如何在vue3.0+中使用tinymce及实现多图上传文件上传公式编辑功能,tinymce安装方法文中也给大家详细介绍了,对vue tinymce实现上传公式编辑相关知识感兴趣的朋友跟随小编一起学习下吧
    2021-05-05
  • vue中使用Tinymce的示例代码

    vue中使用Tinymce的示例代码

    这篇文章主要介绍了vue中使用Tinymce的示例,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-08-08
  • Vue中使用Swiper简单封装组件示例

    Vue中使用Swiper简单封装组件示例

    这篇文章主要为大家介绍了Vue中使用Swiper简单封装组件示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-08-08
  • vue3数据更新而视图未更新问题解决

    vue3数据更新而视图未更新问题解决

    本文主要介绍了vue3 解决数据更新而视图未更新问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-10-10
  • Vue使用pdf-lib实现为文件流添加水印并预览

    Vue使用pdf-lib实现为文件流添加水印并预览

    这篇文章主要为大家详细介绍了Vue如何使用pdf-lib实现为文件流添加水印并预览的功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起了解一下
    2023-03-03
  • vue filter的几种用法示例小结

    vue filter的几种用法示例小结

    Vue.filter函数用于注册一个全局的过滤器,该过滤器可以在模板和组件中使用,这篇文章主要介绍了vue filter的几种用法示例小结,需要的朋友可以参考下
    2024-08-08
  • Vue3的7种种组件通信详情

    Vue3的7种种组件通信详情

    Vue3兼容大部分Vue2的特性,用Vue2代码开发Vue3都可以,性能上面打包大小减少 41%,初次渲染快 55%,更新快 133%,内存使用减少 54%,本篇文章主要介绍Vue3的7种种组件通信,需要的朋友可以参考下面文章的具体内容
    2021-09-09

最新评论