当前位置: 首页 > news >正文

el-upload上传图片和视频,支持预览和删除

话不多说, 直接上代码:

视图层:

        <div class="contentDetail"><div class="contentItem"><div style="margin-top:5px;" class="label csAttachment">客服上传图片:</div><el-upload:auto-upload="false":limit="10":on-change="fileChange":on-remove="handleRemoveImg":file-list="csImages"action="#"accept=".jpg,.jpeg,.png"list-type="picture-card"><i slot="default" class="el-icon-plus"></i><div slot="file" slot-scope="{ file }"><img :src="file.url" class="el-upload-list__item-thumbnail" alt="" /><div class="el-upload-list__item-actions"><span class="el-upload-list__item-preview" @click="handlePictureCardPreview(file)"><i class="el-icon-zoom-in"></i></span><span class="el-upload-list__item-delete" @click="handleRemoveImg(file)"><i class="el-icon-delete"></i></span></div></div></el-upload></div></div><div class="contentDetail"><div class="contentItem"><div class="label csAttachment">客服上传视频:</div><el-upload:auto-upload="false":limit="3":on-change="changeUploadVideo":file-list="csVideos"class="avatar-uploader"action="#"list-type="picture-card"><i slot="default" class="el-icon-plus"></i><div slot="file" slot-scope="{ file }"><video:src="file.url":style="{width: csVideos.length > 0 ? '200px' : 0,height: csVideos.length > 0 ? '120px' : 0}"class="video-avatar"controls="controls"><span>您的浏览器不支持视频播放</span></video><div class="el-upload-list__item-actions" style="z-index:101;height:50px;"><span class="el-upload-list__item-delete" @click="handleRemoveVideo(file)"><i class="el-icon-delete"></i></span></div></div></el-upload></div></div>

逻辑层:

// 监听附件相关数据 
watch: {// 新增图片fileList: {async handler(newList) {this.fileData.imgFiles = []if (newList.length) {let fileObj = {}await newList.map(file => {// 上传的文件流转为base64格式if (file.raw) {getBase64File(file.raw).then(res => {fileObj = {fileName: file.name,fileBase64: res}this.fileData.imgFiles.push(fileObj)})} else {fileObj = {fileBase64: file.fileBase64,fileName: file.name,type: file.type}this.fileData.imgFiles.push(fileObj)}})}}},// 删除已上传图片时newImgList: {handler: function(list) {let obj = {fileBase64: '',fileName: '',type: ''}list.map(file => {obj = {fileBase64: file.fileBase64,fileName: file.name,type: file.type}})this.fileData.imgFiles.push(obj)}},//添加视频videoList: {async handler(newList) {this.fileData.videoFiles = []if (newList.length) {let fileObj = {}await newList.map(file => {// 上传的文件流转为base64格式if (file.raw) {getBase64File(file.raw).then(res => {fileObj = {fileName: file.name,fileBase64: res}this.fileData.videoFiles.push(fileObj)})} else {fileObj = {fileBase64: file.fileBase64,fileName: file.name,type: file.type}this.fileData.videoFiles.push(fileObj)}})}}},// 删除已上传视频时newVideoList: {handler: function(list) {let obj = {fileBase64: '',fileName: '',type: ''}list.map(file => {obj = {fileBase64: file.fileBase64,fileName: file.name,type: file.type}})this.fileData.videoFiles.push(obj)}}},  // 添加图片文件fileChange(file, fileList) {this.fileList = fileListthis.fileList.map((item, index) => {const fileSize = item.size / 1024 / 1024if (fileSize > 20) {this.$message.error('单个附件大小不能超过20M')this.fileList.splice(index, 1)}})setTimeout(() => {this.editFile('image')}, 1000)},// 添加视频文件changeUploadVideo(file, fileList) {const fileSize = file.size / 1024 / 1024 <= 50if (['video/mp4', 'video/ogg', 'video/flv', 'video/avi', 'video/wmv', 'video/rmvb', 'video/mov'].indexOf(file.raw.type) == -1 // 控制格式) {this.$message.error('请上传正确的视频格式')return false}if (!fileSize) {this.$message.error('视频大小不能超过50MB')return false}this.videoList = fileListsetTimeout(() => {this.editFile('video')}, 1000)},// 移除图片文件handleRemoveImg(file) {this.fileList.map((item, index) => {if (item.name === file.name) {// 回显图片时if (item.type === 2) {item.type = 1 // 2 保留 1 删除this.newImgList = this.fileList.splice(index, 1)setTimeout(() => {this.editFile('image')}, 500)} else {// 新增图片时this.fileList.splice(index, 1)}}})},// 移除视频文件handleRemoveVideo(file) {this.videoList.map((item, index) => {if (item.name === file.name) {// 回显视频时if (item.type === 2) {item.type = 1 // 2 保留 1 删除this.newVideoList = this.videoList.splice(index, 1)setTimeout(() => {this.editFile('video')}, 500)} else {// 新增视频时this.videoList.splice(index, 1)}}})},// 预览图片handlePictureCardPreview(file) {this.dialogImageUrl = file.urlthis.$alert(`<img src="${this.dialogImageUrl}" width="100%">`, {dangerouslyUseHTMLString: true,callback: () => {}})},// 编辑附件editFile(type) {const params = {imgFiles: this.fileData.imgFiles,videoFiles: this.fileData.videoFiles,csClass: this.summaryDetail.csClassIds[this.summaryDetail.csClassIds.length - 1],csFeedbackDescribe: this.summaryDetail.csFeedbackDescribe,id: this.summaryDetail.id,role: 1,appPhone: this.summaryDetail.appPhone,sn: this.summaryDetail.sn,qrCode: this.summaryDetail.qrCode,iscallBack: 1 // 是否编辑回电  1否 2是}this.$confirm('确认修改?', '提示', {confirmButtonText: '确定',cancelButtonText: '取消',type: 'warning'}).then(() => {this.loading = trueaddSummary(params) // 后端接口.then(res => {if (res.code === 0) {this.getSummaryList(this.activeName)this.$message.success(res.msg)}}).catch(() => {this.loading = false// 添加修改失败,恢复原有数据if (type === 'image') {this.csImages = handleFileFormat(this.summaryDetail.csImgFiles)} else {this.csVideos = handleFileFormat(this.summaryDetail.csVideoFiles)}})}).catch(() => {// 取消添加修改,恢复原有数据if (type === 'image') {this.csImages = handleFileFormat(this.summaryDetail.csImgFiles)} else {this.csVideos = handleFileFormat(this.summaryDetail.csVideoFiles)}})}

上传附件没有使用单独的上传接口,是调用添加记录接口时,统一传参保存。添加接口请求成功后再回显。

因为我们的需求是在详情页面也能编辑图片和视频,所以加了`type`字段,1代表删除,2代表保留,添加的话就不传。如果你们的需求没有在详情编辑的功能,相关的逻辑可以不用管。

添加失败或取消添加时,恢复原有数据。

视频的时候需要注意:video标签的层级比较高,鼠标hover时上面的删除图标显示不出来,手动修改它们的`z-index`,比如:

 

 删除图标的容器宽度也修改下,否则会覆盖视频播放按钮。

样式设置:

/deep/ .el-upload--picture-card {width: 80px;height: 80px;}/deep/ .el-upload-list__item {width: 80px;height: 80px;}.avatar-uploader {/deep/ .el-upload--picture-card {display: inline-block;width: 200px;height: 120px;}/deep/ .el-upload-list__item {display: inline-block;width: 200px;height: 120px;}}video {display: inline-block;position: relative;z-index: 100;}/deep/ .el-icon-plus {position: relative;top: -35%;}.avatar-uploader {/deep/ .el-icon-plus {position: relative;top: -6%;}}

上传前: 

 上传后:

 

相关文章:

el-upload上传图片和视频,支持预览和删除

话不多说&#xff0c; 直接上代码&#xff1a; 视图层&#xff1a; <div class"contentDetail"><div class"contentItem"><div style"margin-top:5px;" class"label csAttachment">客服上传图片:</div><el…...

clickhouse MPPDB数据库 运维实用SQL总结III

文章目录 CH问题处理使用remote函数报URL "xxxx:9000" is not allowed in configuration fileclickhouse MPPDB数据库 运维实用SQL总结 clickhouse MPPDB数据库 运维实用SQL总结II clickhouse MPPDB数据库 运维实用SQL总结III CH server相关的配置参见 : clickhous…...

ARM和MIPS的区别

ARM和MIPS的区别主要有以下几方面&#xff1a; 指令集&#xff1a;ARM支持32位和64位指令&#xff0c;而MIPS同时支持32位和64位指令。除法器&#xff1a;MIPS有专门的除法器&#xff0c;可以执行除法指令&#xff0c;而ARM没有。寄存器&#xff1a;MIPS的内核寄存器比ARM多一…...

TypeScript -- 类

文章目录 TypeScript -- 类TS -- 类的概念创建一个简单的ts类继承 public / private / protected-- 公共/私有/受保护的public -- 公共private -- 私有的protected -- 受保护的 其他特性readonly -- 只读属性静态属性 -- static修饰ts的getter /setter抽象类abstract TypeScrip…...

【LeetCode】124.二叉树中的最大路径和

题目 二叉树中的 路径 被定义为一条节点序列&#xff0c;序列中每对相邻节点之间都存在一条边。同一个节点在一条路径序列中 至多出现一次 。该路径 至少包含一个 节点&#xff0c;且不一定经过根节点。 路径和 是路径中各节点值的总和。 给你一个二叉树的根节点 root &…...

Linux命令总结

1.目录相关命令 绝对路径&#xff1a; 如/etc/init.d当前目录和上层目录&#xff1a; ./ …/主目录&#xff1a; ~/切换目录&#xff1a; c 2.进程相关命令 查看当前进程&#xff1a; ps ps -ef&#xff08;system v 输出&#xff09;ps -aux bsd 格式输出ps -ef|grep pid 执…...

SpringBoot临时属性设置

在Spring Boot中&#xff0c;可以通过设置临时属性来覆盖应用程序中定义的属性。这在某些情况下很有用&#xff0c;例如在命令行中指定配置参数或在测试环境中覆盖默认值。 你可以使用--&#xff08;双破折号&#xff09;语法来设置临时属性。以下是一些示例&#xff1a; 1. …...

【Python小知识】如何解决代理IP在多线程环境下的并发问题?

前言 在多线程环境下&#xff0c;使用代理IP可能会出现并发问题。具体而言&#xff0c;多个线程可能同时使用同一个代理IP&#xff0c;导致代理IP被封禁或无法访问。为了解决这个问题&#xff0c;我们需要使用一个代理IP池来管理可用的代理IP&#xff0c;并在多线程环境下动态…...

redis常见面试汇总

目录 Redis 适合的场景 Redis 不适合的场景 3、Redis 有哪些常见的功能&#xff1f; 什么是缓存穿透&#xff1f;怎么解决&#xff1f; 什么是缓存雪崩&#xff1f;该如何解决&#xff1f; 参考文献&#xff1a; Redis 适合的场景 缓存&#xff1a;减轻 MySQL 的查询压力…...

子数组的解释与专题

子数组&#xff1a;指在一个数组中&#xff0c;选择一些连续的元素组成的新数组。 例题一&#xff1a;6900. 统计完全子数组的数目 给你一个由 正 整数组成的数组 nums 。 如果数组中的某个子数组满足下述条件&#xff0c;则称之为 完全子数组 &#xff1a; 子数组中 不同 …...

PHP: 开发入门macOS系统下的安装和配置

安装Homebrew 安装 ~~友情提示&#xff1a;这个命令对网络有要求&#xff0c;可能需要翻墙或者用你的手机热点试试&#xff0c;或者把DNS换成&#xff08;114.114.114.114 和 8.8.8.8&#xff09; /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebr…...

在CentOS下安装docker

1&#xff09;在Cent OS安装docker先有一个Cent OS 7.6系统 这个很重要&#xff0c;不同版本按照的时候是不一样的。 2&#xff09;查看CentOS版本 cat /etc/redhat-releas 3&#xff09;用root账户登录进去配置国内yum源 wget -O /etc/yum.repos.d/CentOS-Base.repo http:…...

[JavaWeb]SQL介绍-DQL查询数据

SQL介绍-DQL查询数据 一.基础查询二.条件查询三.排序查询1.聚合函数2.分组查询 四.分页查询 DQL查询基础的语法结构如下&#xff1a; SELECT字段列表 FROM表名列表 WHERE条件列表 GROUP BY分组字段 HAVING分组后条件 ORDER BY排序字段 LIMIT分页限定一.基础查询 说明语法查询…...

[containerd] 在Windows上使用IDEA远程调试containerd, ctr, containerd-shim

文章目录 1. containerd安装2. 源码编译3. 验证编译的二进制文件是否含有调试需要的信息3.1. objdump工具验证3.2. file工具验证3.3. dlv工具验证 4. debug 1. containerd安装 [Ubuntu 22.04] 安装containerd 2. 源码编译 主要步骤如下&#xff1a; 1、从github下载containe…...

Verilog语法学习——LV4_移位运算与乘法

LV4_移位运算与乘法 题目来源于牛客网 [牛客网在线编程_Verilog篇_Verilog快速入门 (nowcoder.com)](https://www.nowcoder.com/exam/oj?page1&tabVerilog篇&topicId301) 题目 题目描述&#xff1a; 已知d为一个8位数&#xff0c;请在每个时钟周期分别输出该数乘1/…...

打卡力扣题目九

#左耳听风 ARST 打卡活动重启# 目录 一、问题 二、解题方法一 三、解题方法二 四、两种方法的区别 关于 ARTS 的释义 —— 每周完成一个 ARTS&#xff1a; ● Algorithm: 每周至少做一个 LeetCode 的算法题 ● Review: 阅读并点评至少一篇英文技术文章 ● Tips: 学习至少一个…...

Python零基础入门(九)——函数,类和对象

系列文章目录 个人简介&#xff1a;机电专业在读研究生&#xff0c;CSDN内容合伙人&#xff0c;博主个人首页 Python入门专栏&#xff1a;《Python入门》欢迎阅读&#xff0c;一起进步&#xff01;&#x1f31f;&#x1f31f;&#x1f31f; 码字不易&#xff0c;如果觉得文章不…...

在linux上面部署activemq

1、下载 网址&#xff1a;ActiveMQ 注意&#xff1a;新版本5.17起 要求jdk11, 5.16兼容jdk8, 所以&#xff0c;确保已经安装 java11 或以上的版本 这里安装较新版&#xff1a;5.18.2&#xff0c;已经安装了java17 如何安装jdk17,请详见我的另一篇文章&#xff1a;linux…...

mysql的sql语句优化方法面试题总结

mysql的sql语句优化方法面试题总结 不要写一些没有意义的查询&#xff0c;如需要生成一个空表结构&#xff1a; select col1,col2 into #t from t where 10 这类代码不会返回任何结果集&#xff0c;但是会消耗系统资源的&#xff0c;应改成这样&#xff1a; create table #t…...

小程序 获取用户头像、昵称、手机号的组件封装(最新版)

在父组件引入该组件 <!-- 授权信息 --><auth-mes showModal"{{showModal}}" idautnMes bind:onConfirm"onConfirm"></auth-mes> 子组件详细代码为: authMes.wxml <!-- components/authMes/authMes.wxml --> <van-popup show…...

nt!MiInitializeSystemCache函数分析之PointerPte->u.List.NextEntry的由来

第一部分&#xff1a; 1: kd> dd 0xc0304200 c0304200 c10c0000 00000000 00000000 00000000 c0304210 00000000 00000000 00000000 00000000 c0304220 00000000 00000000 00000000 00000000 c0304230 00000000 00000000 00000000 00000000 c0304240 00000000 00000000…...

如何理解UDP 和 TCP 区别 应用场景

UDP与TCP的定义、特性、使用场景及对比表格 定义 UDP&#xff08;User Datagram Protocol&#xff09;是一种无连接的传输层协议&#xff0c;提供不可靠的数据报服务。 TCP&#xff08;Transmission Control Protocol&#xff09;是一种面向连接的传输层协议&#xff0c;提供…...

c/c++的opencv图像金字塔缩放

图像金字塔缩放&#xff1a;OpenCV C/C 实践 &#x1f4d0; 图像金字塔是计算机视觉中一种重要且基础的多尺度表示方法。它通过对原始图像进行连续的下采样&#xff08;缩小&#xff09;或上采样&#xff08;放大&#xff09;操作&#xff0c;生成一系列不同分辨率的图像。这些…...

【HarmonyOS Next之旅】DevEco Studio使用指南(二十七) -> 开发云函数

目录 1 -> 开发流程 2 -> 创建并配置函数 2.1 -> 创建函数 2.2 -> 配置函数 3 -> 开发函数 4 -> 调试函数 4.1 -> 前提条件 4.2 -> 通过本地调用方式调试函数 4.3 -> 通过远程调用方式调试函数 5 -> 部署函数 1 -> 开发流程 云函数…...

sqli-labs第二十八关——Trick with ‘union select‘

一&#xff1a;分析 这一关的提示和上一关一样&#xff0c;所以我们查看源码&#xff0c;屏蔽了注释符&#xff0c;空格&#xff0c;union&#xff0c;select等关键词 分析这一条源码的几个新增添符号 \s&#xff1a; 匹配任何的空白字符&#xff08;普通空格&#xff0c;\t&…...

Oracle数据库性能优化的最佳实践

原创&#xff1a;厦门微思网络 以下是 Oracle 数据库性能优化的最佳实践&#xff0c;涵盖设计、SQL 优化、索引管理、系统配置等关键维度&#xff0c;帮助提升数据库响应速度和稳定性&#xff1a; 一、SQL 语句优化 1. 避免全表扫描&#xff08;Full Table Scan&#xff09;…...

element-plus bug整理

1.el-table嵌入el-image标签预览时&#xff0c;显示错乱 解决&#xff1a;添加preview-teleported属性 <el-table-column label"等级图标" align"center" prop"icon" min-width"80"><template #default"scope"&g…...

VScode-使用技巧-持续更新

一、Visual Studio Code - MACOS版本 复制当前行 shiftoption方向键⬇️ 同时复制多行 shiftoption 批量替换换行 在查找和替换面板中&#xff0c;你会看到一个 .∗ 图标&#xff08;表示启用正则表达式&#xff09;。确保这个选项被选中&#xff0c;因为我们需要使用正则…...

在 Ubuntu 服务器上 下载 Clash 文件使用代理

文件Clash.Verge_1.3.8_x64_portable.zip 在 Ubuntu 服务器上不能使用这个Clash 文件**&#xff0c;我们需要的是 Clash.Meta 而不是 Clash Verge GUI 客户端 也就是 Clash Verge GUI 客户端的 Windows 版本&#xff0c;是给 Windows 桌面环境用的图形界面&#xff0c;不适用…...

Linux——数据链路层

1. 认识以太网 认知&#xff1a;以太网是用于局域网数据通信的协议标准&#xff0c;定义了同一局域网内通过电缆/无线怎么在设备之间传输数据帧。 注&#xff1a;整个网络世界可以具象看出由许许多多的局域网组成&#xff0c; • 家庭中的设备A and 家庭中的设备B and 家庭路由…...