mp:增加上传临时素材的接口

pull/2/head
YunaiV 2023-01-11 20:13:20 +08:00
parent 541ee81300
commit ec872c702c
14 changed files with 426 additions and 66 deletions

View File

@ -4,9 +4,6 @@ import lombok.*;
import io.swagger.annotations.*; import io.swagger.annotations.*;
import javax.validation.constraints.*; import javax.validation.constraints.*;
/**
* @author fengdan
*/
@ApiModel("管理后台 - 公众号账号更新 Request VO") @ApiModel("管理后台 - 公众号账号更新 Request VO")
@Data @Data
@EqualsAndHashCode(callSuper = true) @EqualsAndHashCode(callSuper = true)

View File

@ -0,0 +1,39 @@
package cn.iocoder.yudao.module.mp.controller.admin.material;
import cn.iocoder.yudao.framework.common.pojo.CommonResult;
import cn.iocoder.yudao.module.mp.controller.admin.material.vo.MpMaterialUploadRespVO;
import cn.iocoder.yudao.module.mp.controller.admin.material.vo.MpMaterialUploadTemporaryReqVO;
import cn.iocoder.yudao.module.mp.convert.material.MpMaterialConvert;
import cn.iocoder.yudao.module.mp.dal.dataobject.material.MpMaterialDO;
import cn.iocoder.yudao.module.mp.service.material.MpMaterialService;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource;
import javax.validation.Valid;
import java.io.IOException;
import static cn.iocoder.yudao.framework.common.pojo.CommonResult.success;
@Api(tags = "管理后台 - 公众号素材")
@RestController
@RequestMapping("/mp/material")
@Validated
public class MpMaterialController {
@Resource
private MpMaterialService mpMaterialService;
@ApiOperation("上传临时素材")
@PostMapping("/upload-temporary")
public CommonResult<MpMaterialUploadRespVO> uploadTemporaryMaterial(
@Valid MpMaterialUploadTemporaryReqVO reqVO) throws IOException {
MpMaterialDO material = mpMaterialService.uploadTemporaryMaterial(reqVO);
return success(MpMaterialConvert.INSTANCE.convert(material));
}
}

View File

@ -0,0 +1,17 @@
package cn.iocoder.yudao.module.mp.controller.admin.material.vo;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
@ApiModel("管理后台 - 公众号素材上传结果 Response VO")
@Data
public class MpMaterialUploadRespVO {
@ApiModelProperty(value = "素材的 media_id", required = true, example = "123")
private String mediaId;
@ApiModelProperty(value = "素材的 URL", required = true, example = "https://www.iocoder.cn/1.png")
private String url;
}

View File

@ -0,0 +1,29 @@
package cn.iocoder.yudao.module.mp.controller.admin.material.vo;
import com.fasterxml.jackson.annotation.JsonIgnore;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import org.springframework.web.multipart.MultipartFile;
import javax.validation.constraints.NotEmpty;
import javax.validation.constraints.NotNull;
@ApiModel("管理后台 - 公众号素材上传临时 Request VO")
@Data
public class MpMaterialUploadTemporaryReqVO {
@ApiModelProperty(value = "公众号账号的编号", required = true, example = "2048")
@NotNull(message = "公众号账号的编号不能为空")
private Long accountId;
@ApiModelProperty(value = "文件类型", required = true, example = "image", notes = "参见 WxConsts.MediaFileType 枚举")
@NotEmpty(message = "文件类型不能为空")
private String type;
@ApiModelProperty(value = "文件附件", required = true)
@NotNull(message = "文件不能为空")
@JsonIgnore // 避免被操作日志,进行序列化,导致报错
private MultipartFile file;
}

View File

@ -0,0 +1,25 @@
package cn.iocoder.yudao.module.mp.convert.material;
import cn.iocoder.yudao.module.mp.controller.admin.material.vo.MpMaterialUploadRespVO;
import cn.iocoder.yudao.module.mp.dal.dataobject.account.MpAccountDO;
import cn.iocoder.yudao.module.mp.dal.dataobject.material.MpMaterialDO;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.Mappings;
import org.mapstruct.factory.Mappers;
@Mapper
public interface MpMaterialConvert {
MpMaterialConvert INSTANCE = Mappers.getMapper(MpMaterialConvert.class);
@Mappings({
@Mapping(target = "id", ignore = true),
@Mapping(source = "account.id", target = "accountId"),
@Mapping(source = "account.appId", target = "appId"),
})
MpMaterialDO convert(String mediaId, String type, String url, MpAccountDO account);
MpMaterialUploadRespVO convert(MpMaterialDO bean);
}

View File

@ -0,0 +1,95 @@
package cn.iocoder.yudao.module.mp.dal.dataobject.material;
import cn.iocoder.yudao.framework.mybatis.core.dataobject.BaseDO;
import cn.iocoder.yudao.module.mp.dal.dataobject.account.MpAccountDO;
import com.baomidou.mybatisplus.annotation.KeySequence;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.*;
import me.chanjar.weixin.common.api.WxConsts;
/**
* DO
*
* 1. <a href="https://developers.weixin.qq.com/doc/offiaccount/Asset_Management/New_temporary_materials.html"></a>
* 2. <a href="https://developers.weixin.qq.com/doc/offiaccount/Asset_Management/Adding_Permanent_Assets.html"></a>
*
* @author
*/
@TableName("mp_material")
@KeySequence("mp_material_seq") // 用于 Oracle、PostgreSQL、Kingbase、DB2、H2 数据库的主键自增。如果是 MySQL 等数据库,可不写。
@Data
@EqualsAndHashCode(callSuper = true)
@ToString(callSuper = true)
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class MpMaterialDO extends BaseDO {
/**
*
*/
@TableId
private Long id;
/**
* ID
*
* {@link MpAccountDO#getId()}
*/
private Long accountId;
/**
* appid
*
* {@link MpAccountDO#getAppId()}
*/
private String appId;
/**
* id
*/
private String mediaId;
/**
*
*
* {@link WxConsts.MediaFileType}
*/
private String type;
/**
*
*
* true -
* false -
*/
private Boolean permanent;
/**
* URL
*/
private String url;
/**
*
*
* 使
*/
private String name;
/**
* URL
*
* 使
*/
private String mpUrl;
/**
*
*
* 使
*/
private String title;
/**
*
*
* 使
*/
private String introduction;
}

View File

@ -0,0 +1,15 @@
package cn.iocoder.yudao.module.mp.dal.mysql.material;
import cn.iocoder.yudao.framework.mybatis.core.mapper.BaseMapperX;
import cn.iocoder.yudao.module.mp.dal.dataobject.material.MpMaterialDO;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface MpMaterialMapper extends BaseMapperX<MpMaterialDO> {
default MpMaterialDO selectByAccountIdAndMediaId(Long accountId, String mediaId) {
return selectOne(MpMaterialDO::getAccountId, accountId,
MpMaterialDO::getMediaId, mediaId);
}
}

View File

@ -80,4 +80,25 @@ public class MpUtils {
ValidationUtils.validate(validator, message, group); ValidationUtils.validate(validator, message, group);
} }
/**
*
*
* WxConsts.MediaFileType.THUMB
*
* @param messageType {@link WxConsts.XmlMsgType}
* @return {@link WxConsts.MediaFileType}
*/
public static String getMediaFileType(String messageType) {
switch (messageType) {
case WxConsts.XmlMsgType.IMAGE:
return WxConsts.MediaFileType.IMAGE;
case WxConsts.XmlMsgType.VOICE:
return WxConsts.MediaFileType.VOICE;
case WxConsts.XmlMsgType.VIDEO:
return WxConsts.MediaFileType.VIDEO;
default:
return WxConsts.MediaFileType.FILE;
}
}
} }

View File

@ -0,0 +1,30 @@
package cn.iocoder.yudao.module.mp.service.material;
import cn.iocoder.yudao.module.mp.controller.admin.material.vo.MpMaterialUploadTemporaryReqVO;
import cn.iocoder.yudao.module.mp.dal.dataobject.material.MpMaterialDO;
import me.chanjar.weixin.common.api.WxConsts;
import javax.validation.Valid;
import java.io.IOException;
/**
* Service
*
* @author
*/
public interface MpMaterialService {
/**
* URL
*
* URL URL URL
*
* @param accountId
* @param mediaId id
* @param type {@link WxConsts.MediaFileType}
* @return URL
*/
String downloadMaterialUrl(Long accountId, String mediaId, String type);
MpMaterialDO uploadTemporaryMaterial(@Valid MpMaterialUploadTemporaryReqVO reqVO) throws IOException;
}

View File

@ -0,0 +1,133 @@
package cn.iocoder.yudao.module.mp.service.material;
import cn.hutool.core.io.FileTypeUtil;
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.util.ObjUtil;
import cn.iocoder.yudao.module.infra.api.file.FileApi;
import cn.iocoder.yudao.module.mp.controller.admin.material.vo.MpMaterialUploadTemporaryReqVO;
import cn.iocoder.yudao.module.mp.convert.material.MpMaterialConvert;
import cn.iocoder.yudao.module.mp.dal.dataobject.account.MpAccountDO;
import cn.iocoder.yudao.module.mp.dal.dataobject.material.MpMaterialDO;
import cn.iocoder.yudao.module.mp.dal.mysql.material.MpMaterialMapper;
import cn.iocoder.yudao.module.mp.framework.mp.core.MpServiceFactory;
import cn.iocoder.yudao.module.mp.service.account.MpAccountService;
import lombok.extern.slf4j.Slf4j;
import me.chanjar.weixin.common.bean.result.WxMediaUploadResult;
import me.chanjar.weixin.common.error.WxErrorException;
import me.chanjar.weixin.mp.api.WxMpService;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Service;
import org.springframework.validation.annotation.Validated;
import javax.annotation.Resource;
import java.io.File;
import java.io.IOException;
/**
* Service
*
* @author
*/
@Service
@Validated
@Slf4j
public class MpMaterialServiceImpl implements MpMaterialService {
@Resource
private MpMaterialMapper mpMaterialMapper;
@Resource
private FileApi fileApi;
@Resource
@Lazy // 延迟加载,解决循环依赖的问题
private MpAccountService mpAccountService;
@Resource
@Lazy // 延迟加载,解决循环依赖的问题
private MpServiceFactory mpServiceFactory;
@Override
public String downloadMaterialUrl(Long accountId, String mediaId, String type) {
// 第一步,直接从数据库查询。如果已经下载,直接返回
MpMaterialDO material = mpMaterialMapper.selectByAccountIdAndMediaId(accountId, mediaId);
if (material != null) {
return material.getUrl();
}
// 第二步,尝试从临时素材中下载
String url = downloadMedia(accountId, mediaId);
if (url == null) {
return null;
}
MpAccountDO account = mpAccountService.getRequiredAccount(accountId);
material = MpMaterialConvert.INSTANCE.convert(mediaId, type, url, account)
.setPermanent(false);
mpMaterialMapper.insert(material);
// 不考虑下载永久素材,因为上传的时候已经保存
return url;
}
@Override
public MpMaterialDO uploadTemporaryMaterial(MpMaterialUploadTemporaryReqVO reqVO) throws IOException {
WxMpService mpService = mpServiceFactory.getRequiredMpService(reqVO.getAccountId());
// 第一步,上传到公众号
File file = null;
WxMediaUploadResult result;
String mediaId;
String url;
try {
// 写入到临时文件
file = FileUtil.newFile(FileUtil.getTmpDirPath() + reqVO.getFile().getOriginalFilename());
reqVO.getFile().transferTo(file);
// 上传到公众号
result = mpService.getMaterialService().mediaUpload(reqVO.getType(), file);
// 上传到文件服务
mediaId = ObjUtil.defaultIfNull(result.getMediaId(), result.getThumbMediaId());
url = uploadFile(mediaId, file);
} catch (WxErrorException e) {
// TODO yunai待完善
throw new RuntimeException(e);
} finally {
FileUtil.del(file);
}
// 第二步,存储到数据库
MpAccountDO account = mpAccountService.getRequiredAccount(reqVO.getAccountId());
MpMaterialDO material = MpMaterialConvert.INSTANCE.convert(mediaId, reqVO.getType(), url, account)
.setPermanent(false);
mpMaterialMapper.insert(material);
return material;
}
/**
*
*
* 3 3 media_id
*
* @param accountId
* @param mediaId
* @return URL
*/
public String downloadMedia(Long accountId, String mediaId) {
WxMpService mpService = mpServiceFactory.getMpService(accountId);
for (int i = 0; i < 3; i++) {
try {
// 第一步,从公众号下载媒体文件
File file = mpService.getMaterialService().mediaDownload(mediaId);
// 第二步,上传到文件服务
return uploadFile(mediaId, file);
} catch (WxErrorException e) {
log.error("[mediaDownload][media({}) 第 ({}) 次下载失败]", mediaId, i);
}
}
return null;
}
private String uploadFile(String mediaId, File file) {
String path = mediaId + "." + FileTypeUtil.getType(file);
return fileApi.createFile(path, FileUtil.readBytes(file));
}
}

View File

@ -1,11 +1,8 @@
package cn.iocoder.yudao.module.mp.service.message; package cn.iocoder.yudao.module.mp.service.message;
import cn.hutool.core.io.FileTypeUtil;
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.lang.Assert; import cn.hutool.core.lang.Assert;
import cn.hutool.core.util.StrUtil; import cn.hutool.core.util.StrUtil;
import cn.iocoder.yudao.framework.common.pojo.PageResult; import cn.iocoder.yudao.framework.common.pojo.PageResult;
import cn.iocoder.yudao.module.infra.api.file.FileApi;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.MpMessagePageReqVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.MpMessagePageReqVO;
import cn.iocoder.yudao.module.mp.controller.admin.message.vo.MpMessageSendReqVO; import cn.iocoder.yudao.module.mp.controller.admin.message.vo.MpMessageSendReqVO;
import cn.iocoder.yudao.module.mp.convert.message.MpMessageConvert; import cn.iocoder.yudao.module.mp.convert.message.MpMessageConvert;
@ -17,9 +14,11 @@ import cn.iocoder.yudao.module.mp.enums.message.MpMessageSendFromEnum;
import cn.iocoder.yudao.module.mp.framework.mp.core.MpServiceFactory; import cn.iocoder.yudao.module.mp.framework.mp.core.MpServiceFactory;
import cn.iocoder.yudao.module.mp.framework.mp.core.util.MpUtils; import cn.iocoder.yudao.module.mp.framework.mp.core.util.MpUtils;
import cn.iocoder.yudao.module.mp.service.account.MpAccountService; import cn.iocoder.yudao.module.mp.service.account.MpAccountService;
import cn.iocoder.yudao.module.mp.service.material.MpMaterialService;
import cn.iocoder.yudao.module.mp.service.message.bo.MpMessageSendOutReqBO; import cn.iocoder.yudao.module.mp.service.message.bo.MpMessageSendOutReqBO;
import cn.iocoder.yudao.module.mp.service.user.MpUserService; import cn.iocoder.yudao.module.mp.service.user.MpUserService;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import me.chanjar.weixin.common.api.WxConsts;
import me.chanjar.weixin.common.error.WxErrorException; import me.chanjar.weixin.common.error.WxErrorException;
import me.chanjar.weixin.mp.api.WxMpService; import me.chanjar.weixin.mp.api.WxMpService;
import me.chanjar.weixin.mp.bean.kefu.WxMpKefuMessage; import me.chanjar.weixin.mp.bean.kefu.WxMpKefuMessage;
@ -31,7 +30,6 @@ import org.springframework.validation.annotation.Validated;
import javax.annotation.Resource; import javax.annotation.Resource;
import javax.validation.Validator; import javax.validation.Validator;
import java.io.File;
/** /**
* Service * Service
@ -48,6 +46,8 @@ public class MpMessageServiceImpl implements MpMessageService {
private MpAccountService mpAccountService; private MpAccountService mpAccountService;
@Resource @Resource
private MpUserService mpUserService; private MpUserService mpUserService;
@Resource
private MpMaterialService mpMaterialService;
@Resource @Resource
private MpMessageMapper mpMessageMapper; private MpMessageMapper mpMessageMapper;
@ -56,9 +56,6 @@ public class MpMessageServiceImpl implements MpMessageService {
@Lazy // 延迟加载,解决循环依赖的问题 @Lazy // 延迟加载,解决循环依赖的问题
private MpServiceFactory mpServiceFactory; private MpServiceFactory mpServiceFactory;
@Resource
private FileApi fileApi;
@Resource @Resource
private Validator validator; private Validator validator;
@ -69,7 +66,6 @@ public class MpMessageServiceImpl implements MpMessageService {
@Override @Override
public void receiveMessage(String appId, WxMpXmlMessage wxMessage) { public void receiveMessage(String appId, WxMpXmlMessage wxMessage) {
WxMpService mpService = mpServiceFactory.getRequiredMpService(appId);
// 获得关联信息 // 获得关联信息
MpAccountDO account = mpAccountService.getAccountFromCache(appId); MpAccountDO account = mpAccountService.getAccountFromCache(appId);
Assert.notNull(account, "公众号账号({}) 不存在", appId); Assert.notNull(account, "公众号账号({}) 不存在", appId);
@ -79,7 +75,7 @@ public class MpMessageServiceImpl implements MpMessageService {
// 记录消息 // 记录消息
MpMessageDO message = MpMessageConvert.INSTANCE.convert(wxMessage, account, user); MpMessageDO message = MpMessageConvert.INSTANCE.convert(wxMessage, account, user);
message.setSendFrom(MpMessageSendFromEnum.USER_TO_MP.getFrom()); message.setSendFrom(MpMessageSendFromEnum.USER_TO_MP.getFrom());
downloadMessageMedia(mpService, message); downloadMessageMedia(message);
mpMessageMapper.insert(message); mpMessageMapper.insert(message);
} }
@ -97,6 +93,7 @@ public class MpMessageServiceImpl implements MpMessageService {
// 记录消息 // 记录消息
MpMessageDO message = MpMessageConvert.INSTANCE.convert(sendReqBO, account, user); MpMessageDO message = MpMessageConvert.INSTANCE.convert(sendReqBO, account, user);
message.setSendFrom(MpMessageSendFromEnum.MP_TO_USER.getFrom()); message.setSendFrom(MpMessageSendFromEnum.MP_TO_USER.getFrom());
// TODO 芋艿downloadMessageMedia
mpMessageMapper.insert(message); mpMessageMapper.insert(message);
// 转换返回 WxMpXmlOutMessage 对象 // 转换返回 WxMpXmlOutMessage 对象
@ -124,7 +121,7 @@ public class MpMessageServiceImpl implements MpMessageService {
// 记录消息 // 记录消息
MpMessageDO message = MpMessageConvert.INSTANCE.convert(wxMessage, account, user); MpMessageDO message = MpMessageConvert.INSTANCE.convert(wxMessage, account, user);
message.setSendFrom(MpMessageSendFromEnum.MP_TO_USER.getFrom()); message.setSendFrom(MpMessageSendFromEnum.MP_TO_USER.getFrom());
downloadMessageMedia(mpService, message); downloadMessageMedia(message);
mpMessageMapper.insert(message); mpMessageMapper.insert(message);
return message; return message;
} }
@ -132,38 +129,17 @@ public class MpMessageServiceImpl implements MpMessageService {
/** /**
* 使 * 使
* *
* @param mpService Service
* @param message * @param message
*/ */
private void downloadMessageMedia(WxMpService mpService, MpMessageDO message) { private void downloadMessageMedia(MpMessageDO message) {
if (StrUtil.isNotEmpty(message.getMediaId())) { if (StrUtil.isNotEmpty(message.getMediaId())) {
message.setMediaUrl(downloadMedia(mpService, message.getMediaId())); message.setMediaUrl(mpMaterialService.downloadMaterialUrl(message.getAccountId(),
message.getMediaId(), MpUtils.getMediaFileType(message.getType())));
} }
if (StrUtil.isNotEmpty(message.getThumbMediaId())) { if (StrUtil.isNotEmpty(message.getThumbMediaId())) {
message.setThumbMediaUrl(downloadMedia(mpService, message.getThumbMediaId())); message.setThumbMediaUrl(mpMaterialService.downloadMaterialUrl(message.getAccountId(),
message.getThumbMediaId(), WxConsts.MediaFileType.THUMB));
} }
} }
/**
*
*
* 3 3 media_id
*
* @param mpService Service
* @param mediaId
* @return URL
*/
private String downloadMedia(WxMpService mpService, String mediaId) {
try {
// 第一步,从公众号下载媒体文件
File file = mpService.getMaterialService().mediaDownload(mediaId);
// 第二步,上传到文件服务
String path = mediaId + "." + FileTypeUtil.getType(file);
return fileApi.createFile(path, FileUtil.readBytes(file));
} catch (WxErrorException e) {
log.error("[mediaDownload][media({}) 下载失败]", mediaId);
}
return null;
}
} }

View File

@ -158,7 +158,7 @@ import {getMessagePage, sendMessage} from '@/api/mp/message'
}, { }, {
...this.objData, ...this.objData,
type: this.objData.repType, type: this.objData.repType,
content: this.objData.repContent, // content: this.objData.repContent,
// TODO // TODO
})).then(response => { })).then(response => {
this.sendLoading = false this.sendLoading = false

View File

@ -4,15 +4,12 @@
--> -->
<template> <template>
<el-tabs type="border-card" v-model="objData.repType" @tab-click="handleClick"> <el-tabs type="border-card" v-model="objData.repType" @tab-click="handleClick">
<!-- 类型 1文本 -->
<el-tab-pane name="text"> <el-tab-pane name="text">
<span slot="label"><i class="el-icon-document"></i> 文本</span> <span slot="label"><i class="el-icon-document"></i> 文本</span>
<el-input <el-input type="textarea" :rows="5" placeholder="请输入内容" v-model="objData.content" />
type="textarea"
:rows="5"
placeholder="请输入内容"
v-model="objData.repContent">
</el-input>
</el-tab-pane> </el-tab-pane>
<!-- 类型 2图片 -->
<el-tab-pane name="image"> <el-tab-pane name="image">
<span slot="label"><i class="el-icon-picture"></i> 图片</span> <span slot="label"><i class="el-icon-picture"></i> 图片</span>
<el-row> <el-row>
@ -29,19 +26,10 @@
<el-button type="success" @click="openMaterial"><i class="el-icon-circle-check el-icon--right"></i></el-button> <el-button type="success" @click="openMaterial"><i class="el-icon-circle-check el-icon--right"></i></el-button>
</el-col> </el-col>
<el-col :span="12" class="col-add"> <el-col :span="12" class="col-add">
<el-upload <el-upload :action="actionUrl" :headers="headers" multiple :limit="1" :file-list="fileList" :data="uploadData"
:action="actionUrl" :before-upload="beforeImageUpload" :on-success="handleUploadSuccess">
:headers="headers"
multiple
:limit="1"
:on-success="handleUploadSuccess"
:file-list="fileList"
:before-upload="beforeImageUpload"
:data="uploadData">
<el-button type="primary">上传图片</el-button> <el-button type="primary">上传图片</el-button>
<div slot="tip" class="el-upload__tip"> <div slot="tip" class="el-upload__tip">支持 bmp/png/jpeg/jpg/gif 格式大小不超过 2M</div>
支持bmp/png/jpeg/jpg/gif格式大小不超过2M
</div>
</el-upload> </el-upload>
</el-col> </el-col>
</el-row> </el-row>
@ -69,12 +57,7 @@
<el-button type="success" @click="openMaterial"><i class="el-icon-circle-check el-icon--right"></i></el-button> <el-button type="success" @click="openMaterial"><i class="el-icon-circle-check el-icon--right"></i></el-button>
</el-col> </el-col>
<el-col :span="12" class="col-add"> <el-col :span="12" class="col-add">
<el-upload <el-upload :action="actionUrl" :headers="headers" multiple :limit="1" :on-success="handleUploadSuccess"
:action="actionUrl"
:headers="headers"
multiple
:limit="1"
:on-success="handleUploadSuccess"
:file-list="fileList" :file-list="fileList"
:before-upload="beforeVoiceUpload" :before-upload="beforeVoiceUpload"
:data="uploadData"> :data="uploadData">