【vue、UI】使用 Vue2 和 Element UI 封装 CSV 文件上传组件,实现csv回显
文章目录
- 前言
- 组件功能概述
- 实现效果
- 组件模板结构
- 组件的核心逻辑
- 1.数据属性定义
- 2.方法拆解
- 3.CSV 文件解析方法
- 4. 错误处理方法
- 组件样式
- 完整组件代码
- 总结
- 待优化的地方
前言
在 Vue2 项目中,我们经常需要封装一些可重用的组件来提升开发效率。本文将介绍如何使用 Vue2 和 Element UI 封装一个用于上传 CSV 文件并在对话框中回显其内容的公共组件。此组件共涉及两个接口:一个用于校验 CSV 文件内容是否合规,另一个用于上传经过校验的 CSV 文件。
组件功能概述
该组件主要包括以下功能:
- 选择 CSV 文件并上传。
- 校验文件内容是否符合要求。
- 将文件内容以表格形式展示。
- 支持对不合规内容进行标记和提示。
- 用户可在确认内容无误后手动点击上传。
实现效果

组件模板结构
首先来看组件的模板部分。
<template><el-dialog:visible.sync="visible"title="上传 CSV 文件"width="50%":before-close="handleClose":close-on-click-modal="false"><el-upload:action="!validateStatus ? validate : action":before-upload="beforeUpload":show-file-list="false":headers="headers"ref="upload":on-success="handleAvatarSuccess":auto-upload="!validateStatus":file-list="fileList":on-error="errorFn"><el-button type="primary" @click="selectFile">选择 CSV 文件</el-button></el-upload><el-tablev-if="tableData.length > 0":data="tableData.slice(1)"style="width: 100%; margin-top: 20px"border><el-table-columnv-for="(header, index) in tableData[0]":key="'header-' + index":prop="'col-' + index":label="header"><template slot-scope="scope"><div :class="{ 'error-cell': scope.row[index].value.isError }">{{ scope.row[index].value.value }}<el-tooltipclass="item"effect="dark"placement="top"v-if="scope.row[index].value.isError"><template slot="content">{{ scope.row[index].value.errorMsg }}</template><i class="el-icon-question"></i></el-tooltip></div></template></el-table-column></el-table><span slot="footer" class="dialog-footer"><el-button @click="handleClose">关 闭</el-button><el-buttontype="primary"@click="handleConfirm":disabled="!validateStatus":title="!validateStatus ? '请上传文件并通过校验' : '点击上传文件'">确定上传</el-button></span></el-dialog>
</template>
在这个模板中,使用了 Element UI 的 el-dialog 作为弹出框,el-upload 作为上传组件,el-table 显示上传的 CSV 文件内容。组件的主要逻辑操作通过各种方法(methods)来实现。
组件的核心逻辑
1.数据属性定义
以下是组件的数据属性,用于存储上传文件的状态、表格数据和 HTTP 请求头信息
data() {return {tableData: [], // 存储解析后的表格数据headers: {Authorization: "Bearer " + getToken(),},validateStatus: false,fileList: [], // 存储上传的文件};
}
tableData:存储解析后的 CSV 文件数据。headers:HTTP 请求头,包含授权信息。validateStatus:文件校验状态,决定文件是否可以被上传。fileList:用于存储选择的 CSV 文件。
2.方法拆解
selectFile() 方法
selectFile() {this.validateStatus = false;
}
当用户点击“选择 CSV 文件”按钮时,重置 validateStatus 状态为 false,确保在选择新文件时校验状态被重置
validateData(data) 方法
validateData(data) {const allData = Object.values(data).flat();// 检查是否所有的 isError 都为 falseconst allValid = allData.every((item) => !item.isError);if (allValid) {this.validateStatus = true;this.$message.success("校验通过,可以上传");} else {this.validateStatus = false;this.$message.error("请按要求重新修改上传内容");}
}
该方法用于校验上传的数据,如果数据无误,则设置 validateStatus 为 true,并显示成功提示;否则显示错误提示。
handleAvatarSuccess(res, file) 方法
handleAvatarSuccess(res, file) {if (res.code === 500) {this.$message.error(res.msg);this.validateStatus = false;return;}if (res.data) {const csvHeaders = this.tableData[0];const fields = Object.keys(res.data);const dataLength = res.data[fields[0]].length;const result = Array.from({ length: dataLength }, (_, index) =>fields.map((field) => ({value: res.data[field][index],isError: false,errorMsg: "",})));this.tableData = [csvHeaders, ...result];this.validateData(res.data);}
}
该方法在文件上传成功后调用,处理上传成功后的逻辑,包括数据解析和更新表格数据。
handleClose() 方法
handleClose() {this.tableData = [];this.$emit("update:visible", false);
}
关闭对话框时,清空表格数据并触发 visible 属性更新事件,关闭对话框。
handleConfirm() 方法
handleConfirm() {const formData = new FormData();formData.append("file", this.fileList[0]);const config = {headers: this.headers,};axios.post(this.action, formData, config).then((response) => {if (response.data.code == 500 && response.data.msg == null) {this.$message.error("文件提交失败,未知原因");} else if (response.data.code == 200) {this.$message({dangerouslyUseHTMLString: true,message: response.data.msg,type: "success",duration: 5000,});this.$emit("upload-success");}}).catch((error) => {this.$message.error("文件提交失败");});
}
该方法在用户确认上传时调用,通过 Axios 发起 POST 请求将文件上传至后端接口。
3.CSV 文件解析方法
beforeUpload(file) 方法
beforeUpload(file) {if (!this.validateStatus) {const reader = new FileReader();reader.onload = (e) => {const decoder = new TextDecoder("gbk");const csvText = decoder.decode(e.target.result);this.tableData = this.parseCSV(csvText);if (this.tableData.length > 0) {return true;} else {return false;}};reader.readAsArrayBuffer(file);this.fileList = [file];} else {return true;}
}
该方法在文件上传之前执行,使用 FileReader 对象解析 CSV 文件内容,并将其转换为表格数据。
parseCSV(text) 方法
parseCSV(text) {const lines = text.split("\n").map((line) => line.trim());if (lines.length === 0) return [];const headers = this.parseLine(lines[0]);const data = lines.slice(1).map((line) => {const cells = this.parseLine(line);const row = headers.map((header, index) => ({value: cells[index] || "",isError: false,errorMsg: "",}));return row;});return [headers, ...data];
}
该方法用于解析 CSV 文件内容,按行拆分并解析每一行内容为表格所需的格式。
4. 错误处理方法
errorFn(err, file, fileList) 方法
errorFn(err, file, fileList) {console.log("🚀 ~ errorFn ~ err:", err);
}
该方法处理文件上传过程中的错误,当前仅简单地打印错误信息。
组件样式
<style scoped>
.el-table th,
.el-table td {text-align: center;padding: 10px;
}.error-cell {background-color: #ffb1b1;color: black;padding: 5px;
}
</style>
此部分为组件的样式定义,确保表格居中对齐,并为有错误的单元格添加红色背景。
完整组件代码
完整的组件代码如下所示。这段代码结合了 Vue2 和 Element UI,封装了一个 CSV 文件上传与显示的功能组件。
<template><el-dialog:visible.sync="visible"title="上传 CSV 文件"width="50%":before-close="handleClose":close-on-click-modal="false"><el-upload:action="!validateStatus ? validate : action":before-upload="beforeUpload":show-file-list="false":headers="headers"ref="upload":on-success="handleAvatarSuccess":auto-upload="!validateStatus":file-list="fileList":on-error="errorFn"><el-button type="primary" @click="selectFile">选择 CSV 文件</el-button></el-upload><el-tablev-if="tableData.length > 0":data="tableData.slice(1)"style="width: 100%; margin-top: 20px"border><el-table-columnv-for="(header, index) in tableData[0]":key="'header-' + index":prop="'col-' + index":label="header"><template slot-scope="scope"><div :class="{ 'error-cell': scope.row[index].value.isError }">{{ scope.row[index].value.value }}<el-tooltipclass="item"effect="dark"placement="top"v-if="scope.row[index].value.isError"><template slot="content">{{ scope.row[index].value.errorMsg }}</template><i class="el-icon-question"></i></el-tooltip></div></template></el-table-column></el-table><span slot="footer" class="dialog-footer"><el-button @click="handleClose">关 闭</el-button><el-buttontype="primary"@click="handleConfirm":disabled="!validateStatus":title="!validateStatus ? '请上传文件并通过校验' : '点击上传文件'">确定上传</el-button></span></el-dialog>
</template><script>
import { getToken } from "@/utils/auth";
import axios from "axios";
export default {props: {visible: {type: Boolean,required: true,},action: {type: String,required: true,},validate: {type: String,required: true,},},data() {return {tableData: [], // 存储解析后的表格数据headers: {Authorization: "Bearer " + getToken(),},validateStatus: false,fileList: [], // 存储上传的文件};},methods: {selectFile() {this.validateStatus = false;},validateData(data) {const allData = Object.values(data).flat();const allValid = allData.every((item) => !item.isError);if (allValid) {this.validateStatus = true;this.$message.success("校验通过,可以上传");} else {this.validateStatus = false;this.$message.error("请按要求重新修改上传内容");}},handleAvatarSuccess(res, file) {if (res.code === 500) {this.$message.error(res.msg);this.validateStatus = false;return;}if (res.data) {const csvHeaders = this.tableData[0];const fields = Object.keys(res.data);const dataLength = res.data[fields[0]].length;const result = Array.from({ length: dataLength }, (_, index) =>fields.map((field) => ({value: res.data[field][index],isError: false,errorMsg: "",})));this.tableData = [csvHeaders, ...result];this.validateData(res.data);}},handleClose() {this.tableData = [];this.$emit("update:visible", false);},handleConfirm() {const formData = new FormData();formData.append("file", this.fileList[0]);const config = {headers: this.headers,};axios.post(this.action, formData, config).then((response) => {if (response.data.code == 500 && response.data.msg == null) {this.$message.error("文件提交失败,未知原因");} else if (response.data.code == 200) {this.$message({dangerouslyUseHTMLString: true,message: response.data.msg,type: "success",duration: 5000,});this.$emit("upload-success");}}).catch((error) => {this.$message.error("文件提交失败");});},beforeUpload(file) {if (!this.validateStatus) {const reader = new FileReader();reader.onload = (e) => {const decoder = new TextDecoder("gbk");const csvText = decoder.decode(e.target.result);this.tableData = this.parseCSV(csvText);if (this.tableData.length > 0) {return true;} else {return false;}};reader.readAsArrayBuffer(file);this.fileList = [file];} else {return true;}},errorFn(err, file, fileList) {console.log("🚀 ~ errorFn ~ err:", err);},parseCSV(text) {const lines = text.split("\n").map((line) => line.trim());if (lines.length === 0) return [];const headers = this.parseLine(lines[0]);const data = lines.slice(1).map((line) => {const cells = this.parseLine(line);const row = headers.map((header, index) => ({value: cells[index] || "",isError: false,errorMsg: "",}));return row;});return [headers, ...data];},parseLine(line) {const result = [];let current = "";let inQuotes = false;for (let i = 0; i < line.length; i++) {const char = line[i];if (char === '"') {inQuotes = !inQuotes;} else if (char === "," && !inQuotes) {result.push(current.trim());current = "";} else {current += char;}}result.push(current.trim());return result;},},
};
</script><style scoped>
.el-table th,
.el-table td {text-align: center;padding: 10px;
}.error-cell {background-color: #ffb1b1;color: black;padding: 5px;
}
</style>
总结
通过本文的介绍,了解了如何使用 Vue2 和 Element UI 封装一个 CSV 文件上传和回显的组件。该组件的设计充分考虑了数据校验和用户体验,使得上传和展示过程更加直观和友好。在实际项目中,可以根据业务需求对该组件进行扩展和定制。希望这篇文章对您有所帮助!
待优化的地方
文件handleAvatarSuccess可以优化,生成的数据有点冗余,过于嵌套了。
相关文章:
【vue、UI】使用 Vue2 和 Element UI 封装 CSV 文件上传组件,实现csv回显
文章目录 前言组件功能概述实现效果组件模板结构组件的核心逻辑1.数据属性定义2.方法拆解3.CSV 文件解析方法4. 错误处理方法 组件样式完整组件代码总结待优化的地方 前言 在 Vue2 项目中,我们经常需要封装一些可重用的组件来提升开发效率。本文将介绍如何使用 Vue…...
erlang学习: Mnesia Erlang数据库2
Mnesia数据库增加与查询学习 -module(test_mnesia).-record(shop, {item, quantity, cost}). -record(cost, {name, price}). -record(design, {info, plan}). %% API -export([insert/3,select/1,start/0]). start() ->mnesia:start().insert(Name, Quantity, Cost) ->…...
电脑文件怎么备份?推荐6个高效便捷的文件备份的方法
在日常使用电脑的过程中,数据备份是一项至关重要的任务。无论是个人用户还是企业用户,都需要确保重要文件的安全性和可恢复性。 以下是推荐的六个高效便捷的文件备份方法,帮助你轻松守住你的文档安全。 1. 使用USB存储设备 USB存储设备如U盘…...
Procdump抓ToDesk密码
目录 前言 1.工具教程 2.转储数据 3.密码获取 4.总结 前言 本文是因为在公众号上看到一篇文章随想着实战中利用ToDesk秀操作失败后,实验环境成功复现后写下。ProcDump[1] 是一个命令行实用工具,其主要用途是监视应用程序的 CPU 峰值,并在…...
ESP8266下载固件→连接阿里云
一、工具准备 1、ESP8266Wifi模块 2、ESP8266下载器 ESP8266-01模块 二、固件配置 CH340串口工具-烧录ESP8266-01固件_esp8266 ch340烧录-CSDN博客文章浏览阅读444次,点赞6次,收藏3次。CH340会有供电不足的问题,因此需要外部供电_esp…...
20240911软考架构-------软考156-160答案解析
每日打卡题156-160答案 156、NoSQL整体框架分为4层,由下至上分为数据持久层、数据分布层、数据逻辑模型层和(1)。(2)定义了数据的存储形式。(3)定义了数据是如何分布的。(4…...
工厂模式与策略模式(golang示例)
一、工厂模式简介 工厂模式是一种创建型设计模式,主要用于封装对象的创建过程。通过使用工厂模式,客户端代码无需直接实例化对象,而是通过工厂类来创建对象。这样可以将对象的创建与使用分离,从而提高代码的灵活性。 1.1 工厂模…...
批量视频如何做成一个二维码(分步骤教程)
原创教程,阿酷TONY,2024.9.11,湖南长沙 批量视频如何做成一个二维码(分步骤教程),场景应用: 1. 一批视频需要按组分类,生成一个二维码,实现扫一个二维码,观看…...
OpengGL教程(三)---使用VAO和VBO方式绘制三角形
本章参考官方教程:learnopengl-cn VertexShader.glsl #version 330 core layout(location 0) in vec3 position; layout(location 1) in vec3 color; uniform mat4 projection; // 投影矩阵 out vec4 ourColor; void main() {gl_Position projection * vec4(p…...
【单片机开发】单片机常用开发工具
【前言】 在嵌入式系统领域,单片机(Microcontroller, MCU)作为核心组件,广泛应用于智能家居、工业控制、汽车电子等众多领域。而单片机开发工具,则是开发者们实现创意、解决问题的重要助手。本文主要讲述目前主流的单…...
一、计算机网络的体系结构
1.1 计算机网络的组成 1)从组成部分上分为:硬件、软件、协议。硬件是指主机、通信链路、交换设备和通信处理机组曾。软件包括各种实现资源共享的软件以及各种软件工具(如网络操作系统、邮件收发程序、FTP程序、聊天软件)。 2&…...
C语言补习课——文件篇
来源:黑马程序员 第157讲 C语言操作文件概述 读取文件:输入流 写文件:输出流 读写的方向判断取决与参照,一般我们站在程序的角度判断读写方向。 第158讲 路径 基本概念 路径就是指文件在电脑中的位置,eg…...
【可测试性实践】C++ 单元测试代码覆盖率统计入门
引言 最近在调研C工程怎么做单元测试和代码覆盖率统计,由于我们工程有使用Boost库,尝试使用Boost.Test来实现单元测试并通过Gcov和Lcov来生成代码覆盖率报告。本文记录完整的搭建测试Demo,希望能带来一定参考。 常用C单测框架对比 特性Goo…...
C++笔记---list
1. list的介绍 list其实就是就是我们所熟知的链表(双向循环带头结点),但其是作为STL中的一个类模板而存在。 也就是说,list是可以用来存储任意类型数据的顺序表,既可以是内置类型,也可以是自定义类型&…...
JavaWeb开发中为什么Controller里面的方法是@RequestMapping?
在Java Web开发中,尤其是在使用Spring MVC框架时,RequestMapping注解被广泛应用于Controller层的方法上,这是因为RequestMapping是Spring MVC提供的一个核心注解,用于将HTTP请求映射到相应的处理器类或处理器方法上。通过这种方式…...
若依移动版使用微信小程序打开失败
现象 解决办法:删掉自带的appid...
精准控图工具 Concept Sliders:超好用的 控制 Lora 适配器
Concept Sliders 你有没有遇到这样的情况?你花费大量时间制作提示和寻找种子,以使用文本到图像模型生成所需的图像。但是,你还需要对生成图像中的属性强度(如眼睛大小或照明)进行更细致、更精细的控制。修改提示会破坏…...
【EI会议征稿通知】第四届材料工程与应用力学国际学术会议(ICMEAAE 2025)
第四届材料工程与应用力学国际学术会议(ICMEAAE 2025) 2025 4th International Conference on Materials Engineering and Applied Mechanics 本次会议将重点讨论材料科学、应用力学等领域的最新研究进展与发展趋势。会议旨在为国内外从事这些领域研究…...
Hadoop安全之Knox
Apache Knox 是一个 REST API 网关,为 Hadoop 集群提供安全的访问方式。Knox 提供了一层保护,简化了对 Hadoop 生态系统(如 HDFS、YARN、Hive、HBase 等)中各个组件的访问,并通过单点登录 (SSO)、认证、授权和审计功能…...
SprinBoot+Vue应急信息管理系统的设计与实现
目录 1 项目介绍2 项目截图3 核心代码3.1 Controller3.2 Service3.3 Dao3.4 application.yml3.5 SpringbootApplication3.5 Vue 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍:CSDN认证博客专家,CSDN平台Java领域优质…...
谷歌浏览器插件
项目中有时候会用到插件 sync-cookie-extension1.0.0:开发环境同步测试 cookie 至 localhost,便于本地请求服务携带 cookie 参考地址:https://juejin.cn/post/7139354571712757767 里面有源码下载下来,加在到扩展即可使用FeHelp…...
大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...
【WiFi帧结构】
文章目录 帧结构MAC头部管理帧 帧结构 Wi-Fi的帧分为三部分组成:MAC头部frame bodyFCS,其中MAC是固定格式的,frame body是可变长度。 MAC头部有frame control,duration,address1,address2,addre…...
【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...
如何将联系人从 iPhone 转移到 Android
从 iPhone 换到 Android 手机时,你可能需要保留重要的数据,例如通讯录。好在,将通讯录从 iPhone 转移到 Android 手机非常简单,你可以从本文中学习 6 种可靠的方法,确保随时保持连接,不错过任何信息。 第 1…...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”
2025年#高考 将在近日拉开帷幕,#AI 监考一度冲上热搜。当AI深度融入高考,#时间同步 不再是辅助功能,而是决定AI监考系统成败的“生命线”。 AI亮相2025高考,40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕,江西、…...
算法笔记2
1.字符串拼接最好用StringBuilder,不用String 2.创建List<>类型的数组并创建内存 List arr[] new ArrayList[26]; Arrays.setAll(arr, i -> new ArrayList<>()); 3.去掉首尾空格...
使用LangGraph和LangSmith构建多智能体人工智能系统
现在,通过组合几个较小的子智能体来创建一个强大的人工智能智能体正成为一种趋势。但这也带来了一些挑战,比如减少幻觉、管理对话流程、在测试期间留意智能体的工作方式、允许人工介入以及评估其性能。你需要进行大量的反复试验。 在这篇博客〔原作者&a…...
Python竞赛环境搭建全攻略
Python环境搭建竞赛技术文章大纲 竞赛背景与意义 竞赛的目的与价值Python在竞赛中的应用场景环境搭建对竞赛效率的影响 竞赛环境需求分析 常见竞赛类型(算法、数据分析、机器学习等)不同竞赛对Python版本及库的要求硬件与操作系统的兼容性问题 Pyth…...
