Upload 常用属性和方法
示例上传接口
# 接口文档
url `https://www.mocky.io/api/main/upload`
头部
x-token: 'xxx'
参数
file: File // 上传的文件
flag: 'xxx' // 上传的标识
// 文件上传 api 函数简单封装
export const uploadApi = ({ file }) => {const formData = new FormData();formData.append('file', file);// ... 略,formData 其他实现逻辑return axios({url: '/api/upload',method: 'POST',data: formData,headers: {'Content-type': 'multipart/form-data',},});
};
组件基本使用
<a-uploadv-model:file-list="fileList"name="file":multiple="true"action="https://www.mocky.io/api/main/upload":headers="{ x-token: 'xxx' }":data="{ flag: 'xxx' }"@change="handleChange"><a-button>点击上传</a-button>
</a-upload>
基本使用方式可以满足大部分上传需求,但是也存在局限性,仅限于配合该组件的属性使用。如果是已经封装好了上传的请求函数(包括头部信息和请求中间件等),可在多个地方使用,还使用该方式拼接 data,headers 就有点冗余了。
customRequest 自定义上传
简单使用
使用组件的 customRequest 属性覆盖默认的上传行为,可以自定义自己的上传实现
<a-uploadv-model:file-list="fileList"list-type="picture-card"name="file":multiple="true":customRequest="customUpload"@change="handleChange"
><upload-outlined></upload-outlined>
</a-upload><script>
...
const fileList = ref([]);// 自定义文件上传公共函数
// e - 上传组件返回的上传实例,里面包括 file,和一些组件方法
// e.file - 上传的文件实例对象
// e.onSuccess(ret, xhr)
// e.onError(err, ret)
const customUpload = (e) => {console.log(e);// uploadApi 你的二次封装上传接口 uploadApi({file: e.file}).then((res) => {console.log('上传成功', res.data);// 调用实例的成功方法通知组件该文件上传成功e.onSuccess(res.data, e); }).catch((err) => {// 调用实例的失败方法通知组件该文件上传失败e.onError(err);});
};</script>
这样的话,实现自定义的上传函数,二次封装上传接口,例如在uploadApi使用 axios 对基本路径、头部信息和请求参数的封装。
在请求成功后调用组件实例成功回调onSuccess(ret,xhr),回传的ret 将会存放在fileList 中的 file.response 中,可运用这点存放所需的数据。
进度条的实现
由于前面 简单使用 的时候我们自定义了上传逻辑,组件未能监听到上传的进度 fileList 中的 file.percent 会一直处于 0 的状态。所以下面我们实现一下请求进度逻辑,补全该进度条。
// 自定义文件上传公共函数
// e.onProgress(event) event的格式为 {percent:xxx}
const customUpload = (e) => {console.log(e);// 上传接口 e.file 就是接口所用的 fileuploadApi({file: e.file,onUploadProgress: (ev) => {// ev - axios 上传进度实例,上传过程触发多次// ev.loaded 当前已上传内容的大小,ev.total - 本次上传请求内容总大小console.log(ev);const percent = (ev.loaded / ev.total) * 100;// 计算出上传进度,调用组件进度条方法e.onProgress({ percent });},}).then((res) => {console.log('上传成功', res.data);e.onSuccess(res.data, e);}).catch((err) => {e.onError(err);});
};
看上去也就多了个onUploadProgress参数。不过这个参数并非只是 uploadApi 方法的。还需要在请求实例(axios)中配置该上传进度回调参数。
// 文件上传 api
export const uploadApi = ({ file, onUploadProgress }) => {const formData = new FormData();formData.append('file', file);// ... 略,formData 其他实现逻辑return axios({url: '/api/upload',method: 'POST',data: formData,headers: {'Content-type': 'multipart/form-data',},onUploadProgress, // 上传进度回调函数 onUploadProgress(ev)});
};
实现了上传进度逻辑,自定义上传的文件列表 fileList 中的 file.percent 才会正确反应进度情况
配合表单组件使用
以 Vue3 的概念逻辑代码为例
<a-form :model="formInfo"><a-form-item label="your name"><a-input v-model:value="formInfo.name" /></a-form-item><a-form-item label="身份证上传"><a-uploadv-model:file-list="fileList"list-type="picture-card"name="file":multiple="true":customRequest="customUpload"@change="handleChange"><upload-outlined></upload-outlined></a-upload></a-form-item><a-form-item :wrapper-col="{ span: 14, offset: 4 }"><a-button type="primary" @click="onSubmit">Submit</a-button><a-button style="margin-left: 10px">Cancel</a-button></a-form-item>
</a-form><script>
...const fileList = ref([]);
const formInfo = reactive({name: '阿姣🐷',
});// 获取上传成功的文件 url 数组
const getSuccessFileUrls = (list) => {let urls = [];list.forEach((item) => {// 获取上传成功的文件数据if (item.status === 'done' && item.response) {urls.push(item.response.url); //改为你想获取的数据格式}});return urls;
};// 表单提交
const onSubmit = () => {const urls = getSuccessFileIds(fileList.value);console.log(urls);console.log(toRaw(formInfo));...submitInfoApi(data).then(res=>{}).cathc(err=>{})
};
</script>
获取详情后回显表单
请求信息详情接口成功后,将图片文件等url重新按照 fileList 的格式赋值文件列表。
<script>
const fileList = ref([]);const fetchDetail = ()=>{getInfoDetail(data).then(res=>{let imageUrls = res.data.images; // 详情数据 imageUrls.forEach(item=>{let newItem = {uid: 'xxxx1',response: {id: 'xxxx1',origin_name: 'image1.png',url: 'http://xxx/image1.png',},name: 'image1.png',status: 'done',url: 'http://xxx/image1.png',};fileList.value.push(newItem); // 赋值文件列表});...});
}
</script>