使用element ui的el-upload组件上传图片,记录一下

news/2024/4/20 6:18:12/文章来源:https://www.cnblogs.com/code-R/p/16776489.html

使用element ui的el-upload组件上传图片

效果预览

下面是实现效果,接口方面是把有两个接口,一个接口上传图片,传参是图片和路径,返回值是路径。另一个接口是上传表单内容(用户,地址,照片),照片是传一个路径。
image

具体实现

html

<el-form-item label="上传照片" prop="imagePath"><el-uploadref="upload"class="avatar-uploader"list-type="picture-card"action:file-list="fileList":show-file-list="true":limit="uploadLimit":before-upload="beforeUpload":http-request="uploadPicture":class="{ hide: hideUploadEdit }"><i slot="default" class="el-icon-plus"></i><div slot="file" slot-scope="{ file }"><img class="el-upload-list__item-thumbnail" :src="file.url" alt="" /><span class="el-upload-list__item-actions"><spanclass="el-upload-list__item-preview"@click="handlePictureCardPreview(file)"><i class="el-icon-zoom-in"></i></span><spanv-if="!disabled"class="el-upload-list__item-delete"@click="handleRemove(file)"><i class="el-icon-delete"></i></span></span></div></el-upload>
</el-form-item>

下面是预览

<el-dialog :visible.sync="dialogVisible" :width="dialogWidth"><img :src="dialogImageUrl" @load="onLoadImg" :width="imgWidth" alt="" />
</el-dialog>

js
文件上传前的过滤,如文件格式,文件大小

beforeUpload(file) {// console.log('beforeUpload:');// console.log(file);// 1MB=1024*1024(1MB=1024KB 1KB=1024B)const isAllowSize = file.size / 1024 / 1024 < 10;const allowType = ['jpeg', 'jpg', 'png'];const isAllowType = allowType.some(e => 'image/' + e === file.type);if (!isAllowType) {this.$notify({message: `上传图片只能是 ${allowType} 格式!`,type: 'warning'});// 不能直接return 空值,不然就算格式不符合也会生成缩略图return Promise.reject();}if (!isAllowSize) {this.$notify({message: `上传图片大小不能超过 2MB!`,type: 'warning'});return Promise.reject();}this.fileChange(file);return isAllowType && isAllowSize;
},

文件上传后的显示,这里为了省事不做回显了,后端现成的接口也不支持,文件上传失败再删除(或者显示在那个图片上显示一个error的样式)吧。照片墙需要的参数,url参数是必选的,做显示用;uid是为了删除等操作,文件的uid每次上传都不一样,不是按照文件哈希值来的,这一点确保了能精确地进行删除操作,比如照片墙允许4张图,第一张图和第三种图上传的是同一个图片,如果使用文件名或者哈希值,就可能出现删除了错误位置的图片的情况。

fileChange(file) {console.log('fileChange:');// 单个图片直接赋值,如果需要多个图片,改为对象数组this.fileInfo = file;const url = URL.createObjectURL(file);// console.log(url);this.fileList.push({ uid: file.uid, url });// 文件到达上限后隐藏上传按钮this.hideUploadEdit = this.fileList.length >= this.uploadLimit;
},

删除和预览操作,这个是网上找了一些博客后参考实现的,不过原本找到的还挺多坑的,经过一番找Bug后实现了下面的方法。坑点:删除要根据uid来而不是文件名,不然会照片墙出现同名图片时,想删后一张实际删前一张的情况。onLoadImg图片的原始宽度要用img.naturalWidth获取,如果使用width获取,是获取到当前宽度,会固定显示第一次预览的宽度。

handleRemove(file) {console.log(file);for (let i = 0; i < this.fileList.length; i++) {if (this.fileList[i]['uid'] === file.uid) {this.fileList.splice(i, 1);break;}}if (this.fileList.length <= 0) {this.formData.imagePath = '';}// 文件未到达上限后显示上传按钮this.hideUploadEdit = this.fileList.length > this.uploadLimit - 1;console.log('handleRemove:', this.hideUploadEdit);
},
handlePictureCardPreview(file) {this.dialogImageUrl = file.url;this.dialogVisible = true;
},
onLoadImg(e) {const img = e.target;let width = 0;if (img.fileSize > 0 || (img.naturalWidth > 1 && img.naturalHeight > 1)) {width = img.naturalWidth;}console.log('图片大小:', img.naturalWidth, img.naturalHeight);if (img.naturalWidth < 200) {width = 200;} else if (img.naturalHeight > img.naturalWidth && width > 370) {width = 370;} else if (width > 1000) {width = 1000;}this.imgWidth = width + 'px';this.dialogWidth = width + 40 + 'px';console.log('imgWidth:', this.imgWidth);console.log('dialogWidth:', this.dialogWidth);
},

文件上传到服务器,这里是用了http-request钩子,改成手动触发也行的,如果按照本案例的后端接口的话,大概就是点发送按钮后,先发送图片,如果发送异常就直接弹错误信息,正常就把返回的图片路径作为发送表单所需要的图片路径。
这个路径值得说一下,如果每次上传图片,都进行单独的存储的话,那么使用uuid是最好的,但是这样容易存储冗余文件。在经过一些考虑后,包括用户量,项目应用场景等,决定当文件名并且文件修改时间一致时,认为这是同一张图片。
ps: 听说云盘一般是按照哈希值来存储的,即哈希值一致,即认为是同一份文件,不进行额外的存储,不过我查了一下,如果哈希值计算是前端处理的话,性能不好,也不安全。至于后端,现成的接口又没有那么大的需求,感觉也没有必要。折衷一下就按照文件修改时间/文件名来存储吧

async uploadPicture(param) {// http-request钩子参数名只能用param// console.log('param:', param);const fileInfo = this.fileInfo;// 缺少一个id,使用文件修改时间来减少被其他同名文件覆盖的可能const path = `xxxr/${fileInfo.lastModified}`;console.log('path:', path);const formData = new FormData();formData.append(fileInfo.name, fileInfo);const res = await commonUploadFiles(path, formData);if (res) {this.formData.imagePath = res[0];}
},

css

.avatar-uploader {/deep/ .el-upload {border: 1px dashed #d9d9d9 !important;border-radius: 6px !important;cursor: pointer;position: relative;overflow: hidden;}
}
/deep/ .el-upload--picture-card:hover {border-color: #409eff !important;
}
.avatar-uploader-icon {font-size: 28px;color: #8c939d;width: 178px;height: 178px;line-height: 178px;text-align: center;
}
.avatar {width: 178px;height: 178px;display: block;
}
.hide {/deep/ .el-upload {display: none;}
}

踩坑

  1. 文件上传格式不熟悉,大量log输出后理解了
  2. el-upload组件的action属性必须要有。
  3. 如果想让http-request绑定的函数生效,不能设置:auto-upload="false"
  4. 钩子的执行顺序问题,on-change发送在before-upload之前,这会导致,如果在on-change处理图片显示的话,会发生不通过验证的图片依然会生成缩略图。所以before-upload要放在缩略图。这个问题应该是因为el-upload组件内部有一套数据。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_398894.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

第二十一章 函数递归

一、函数递归调用介绍 函数不仅可以嵌套定义,还可以嵌套调用,即在调用一个函数的过程中,函数内部又调用另一个函数,而函数的递归调用指的是在调用一个函数的过程中又直接或间接地调用该函数本身。例如在调用f1的过程中,又调用f1,这就是直接调用函数f1本身def f1():print(…

springboot(三)

视频链接&#xff1a;https://www.bilibili.com/video/BV1XQ4y1m7ex/?vd_source9545770e4a2968c05878ffac8589ec6c 视频选集&#xff1a;P58— P92 文章目录1.接口架构风格-RESTful1.1 认识REST1.2 RESTful的注解1.2.1 PathVariable1.2.2 PostMapping1.2.3 DeleteMapping1.2.4…

分布式缓存

本文介绍关于缓存的常用设计模式。以及如何保证缓存的一致性进行分类讨论。 还会介绍关于缓存失效的常见问题&#xff0c;以及针对缓存失效的解决方法。 在高并发的环境下&#xff0c;比如春节抢票大战&#xff0c;一到放票的时间节点&#xff0c;分分钟大量用户以及黄牛的各种…

魔改xxl-job,彻底告别手动配置任务!

xxl-job是一款非常优秀的任务调度中间件,轻量级、使用简单,但是苦于手动注册任务久矣,今天就来魔改一下,实现任务的自动注册!原创:微信公众号 码农参上,欢迎分享,转载请保留出处。哈喽大家好啊,我是Hydra。 xxl-job是一款非常优秀的任务调度中间件,轻量级、使用简单、…

12个小细节让普源示波器使用更加高效(上)

俗话说细节决定成败&#xff0c;示波器作为电子测量的第一工具&#xff0c;虽然使用简单&#xff0c;但并不是每个人都能注意到细节。运用好细节&#xff0c;可以使你的示波器使用更加的便捷。以下由安泰测试带来普源示波器测量相关的12个小细节可作为示波器常识快速自检的小文…

Spring Boot(4):@Import注解和@Conditional注解

说明&#xff1a;基于atguigu学习笔记。 在了解spring boot自动配置原理前&#xff0c;再来了解下两个注解Import注解和Conditional注解。 Import Import注解主要用于导入某些特殊的Bean&#xff0c;这些特殊的Bean和Bean Definitaion 有关。 主要用于导入Configuration 类…

Python实现桌面挂件,做一只可爱的桌面宠物~

文章目录嗨嗨&#xff0c;大家好 ~ 我是小圆相关文件开发工具相关模块&#xff1a;环境搭建安装原理简介1.初始化一个窗口组件&#xff1a;效果2.设置一下窗口的属性&#xff1a;随机导入一张图片&#xff0c;看效果随机导入一个宠物的所有图片的函数代码3.宠物随机出现在桌面上…

服务端渲染的探索与实践

服务端渲染(SSR)近两年炒得很火热,相信各位同学对这个名词多少有所耳闻。本节我们将围绕“是什么”(服务端渲染的运行机制)、“为什么”(服务端渲染解决了什么性能问题 )、“怎么做”(服务端渲染的应用实例与使用场景)这三个点,对服务端渲染进行探索。 服务端渲染是一…

GOM引擎登录器的研究,逆向技术在这款GOM20151108引擎上是一个大舞台

最近遇到一个逆向类课题&#xff0c;是关于GOM20151108版本对应登录器研究。刚接触传奇的时候是2002年&#xff0c;那时候网吧玩SF&#xff0c;都是手动用IP直接连接&#xff0c;当时的一款 联创传奇 很好玩&#xff0c;有传送戒子&#xff0c;木域戒指&#xff0c;土域戒指&am…

不会 Vue,但不影响我学 diff 算法

前言 现在社会各行各业大都面临着寒冬&#xff0c;互联网行业最近还出现了裁员潮&#xff0c;导致前端是越来越卷&#xff0c;普通学校的应届生不仅要跟985、211毕业的学生以及研究生进行竞争&#xff0c;甚至还需要和最近刚被裁的、有了几年工作经验的程序员竞争&#xff0c;…

page.json

uni-app需要给page.json文件需要进行配置路由,否则会不报错,也跳转不过去

【数模/启发式算法】蚁群算法

文章目录简介符号说明核心思想流程图文章使用到的测试函数基本步骤蚁群算法代码简介 蚁群算法是一种用来寻找优化路径的概率型算法。它由Marco Dorigo于1992年在他的博士论文中提出&#xff0c;其灵感来源于蚂蚁在寻找食物过程中发现路径的行为。 这种算法具有分布计算、信息正…

Arduino播放声音

玩软件有点虚无&#xff0c;没有实际东西&#xff0c;所以接下来要体验下硬件与软件结合。 1 Arduino Arduino是一种包含硬件&#xff08;各种型号的Arduino板&#xff09;和软件&#xff08;Arduino IDE&#xff09;的开源电子平台。硬件部分是可以用来做电路连接的Arduino电…

小白学习Java第四十三天

Git概述 &#xff08;一&#xff09;什么是Git Git是一个开源的分布式版本控制系统&#xff0c;可以有效、高速地处理从很小到非常大的项目版本管理。版本控制是指对软件开发过程中各种程序代码、配置文件及说明文档等文件变更的管理&#xff0c;是软件配置管理的核心思想之一…

设计模式学习笔记(五) - 观察者模式 Observer

目录 观察者模式 Observer 一、背景描述 Version 1 (面向过程) Version 2 (面向对象) Version 3 (单个观察者) Version 4 (多个观察者) Version 5 (分离观察者与被观察者) 二、不同事件下的观察者模式 三、事件本身也可以形成继承体系 四、观察者常用场景 观察者模式…

Selenium基础 — 鼠标操作

1、鼠标事件介绍 前面例子中我们已经学习到可以用click()来模拟鼠标的单击操作&#xff0c;而我们在实际的web产品测试中发现&#xff0c;有关鼠标的操作&#xff0c;不单单只有单击&#xff0c;有时候还要用到右击&#xff0c;双击&#xff0c;拖动等操作&#xff0c;这些操作…

【Nginx】认识与基本使用 Nginx 实现反向代理、配置负载均衡

文章目录1. Nginx 概述1.1 Nginx 介绍1.2 Nginx 下载和安装1.3 Nginx 目录结构2. Nginx 命令3. Nginx 配置文件结构4. Nginx 具体应用4.1 部署静态资源4.2 反向代理4.2.1 介绍4.2.2 配置反向代理4.3 负载均衡4.3.1 介绍4.3.2 配置负载均衡4.3.3 负载均衡策略1. Nginx 概述 1.1…

Ubuntu开机界面出现“error found when loading /root/.profile”

原因 今天一开始按照一篇文章&#xff0c;想把普通用户的权限提高到最高权限&#xff0c;修改了**/etc/passwd**文件&#xff0c;然后重启&#xff0c;发现之前的用户进不去了&#xff0c;一开机就出现如下信息 解决方法 1、重启虚拟机进入recovery模式&#xff08;长按shi…

计算机网络-第一章 | 王道考研

目录 一、基本介绍 定义 功能 组成 分类 标准化工作 标准的分类 标准化工作相关组织 二、性能指标 ※ 速率 带宽 ※吞吐量 时延 时延带宽积 往返时延RTT 利用率 三、分层结构 ※ 分层基本规则 正式认识分层 7层OSI参考模型 怎么来的 怎么分的 怎么传的…

<特殊类设计与单例模式>——《C++高阶》

目录 1.请设计一个类&#xff0c;不能被拷贝 2. 请设计一个类&#xff0c;只能在堆上创建对象 3. 请设计一个类&#xff0c;只能在栈上创建对象 4. 请设计一个类&#xff0c;不能被继承 5. 请设计一个类&#xff0c;只能创建一个对象(单例模式) 后记&#xff1a;●由于…