10-AntD框架的upload组件上传图片时使用customRequest方法自定义上传行为

本次做后台管理系统,采用的是 AntD 框架。涉及到图片的上传,用的是AntD的 upload 组件。

我在上一篇文章《前端AntD框架的upload组件上传图片时遇到的一些坑》中讲到:AntD 的 upload 组件有很多坑,引起了很多人的关注。折腾过的人,自然明白其中的苦楚。

今天这篇文章,我们继续来研究 AntD 的 upload 组件的另一个坑。

备注:本文写于2020-06-11,使用的 antd 版本是 3.13.6。

使用 AntD 的 upload 组件做图片的上传,效果演示

因为需要上传多张图片,所以采用的是照片墙的形式。上传成功后的界面如下:

(1)上传中:

10-AntD框架的upload组件上传图片时使用customRequest方法自定义上传行为 - 图1

(2)上传成功:

10-AntD框架的upload组件上传图片时使用customRequest方法自定义上传行为 - 图2

(3)图片预览:

10-AntD框架的upload组件上传图片时使用customRequest方法自定义上传行为 - 图3

代码实现

首先,你需要让后台同学提供好图片上传的接口。上一篇文章中,我们是把接口调用直接写在了 <Upload> 标签的 action 属性当中。但如果你在调接口的时候,动作很复杂(比如根据业务要求,需要连续调两个接口才能上传图片,或者在调接口时还要做其他的事情),这个 action 方法就无法满足需求了。那该怎么做呢?

好在 AntD 的 upload 组件给我们提供了 customRequest这个方法:

10-AntD框架的upload组件上传图片时使用customRequest方法自定义上传行为 - 图4

关于customRequest 这个方法, AntD 官方并没有给出示例,他们只是在 GitHub 上给出了这样一个简短的介绍:

10-AntD框架的upload组件上传图片时使用customRequest方法自定义上传行为 - 图5

但这个方法怎么用呢?用的时候,会遇到什么问题呢?AntD 官方没有说。我在网上搜了半天,也没看到比较完整的、切实可行的 Demo。我天朝地大物博,网络资料浩如烟海,AntD 可是口口声声被人们号称是天朝最好用的管理后台的样式框架。可如今,却面临这样的局面。我看着你们,满怀羡慕。

既然如此,那我就自己研究吧。折腾了一天,总算是把 customRequest 的坑踩得差不多了。

啥也不说了,直接上代码。

采用 AntD框架的 upload 组件的 customRequest 方法,自定义上传行为。核心代码如下:

  1. import React, { PureComponent } from 'react';
  2. import { Button, Card, Form, message, Upload, Icon, Modal, Row, Col } from 'antd';
  3. import { connect } from 'dva';
  4. import { queryMyData, submitData } from '../api';
  5. import { uploadImage } from '../../utils/wq.img.upload';
  6. import styles from '../../utils/form.less';
  7. const FormItem = Form.Item;
  8. @Form.create()
  9. export default class PicturesWall extends PureComponent {
  10. constructor(props) {
  11. super(props);
  12. const { id } = this.props.match.params;
  13. this.state = {
  14. id,
  15. img: undefined, // 从接口拿到的图片字段
  16. imgList: [], // 展示在 antd图片组件上的数据
  17. previewVisible: false,
  18. previewImage: '',
  19. };
  20. }
  21. componentDidMount() {
  22. const { id } = this.state;
  23. id && this.queryData();
  24. }
  25. // 调接口,查询已有的数据
  26. queryData() {
  27. const { id } = this.state;
  28. queryMyData({
  29. id,
  30. })
  31. .then(({ ret, data }) => {
  32. if (ret == 0 && data && data.list && data.list.length) {
  33. const item = data.list[0];
  34. const img = data.img;
  35. const imgList = item.img
  36. ? [
  37. {
  38. uid: '1', // 注意,这个uid一定不能少,否则展示失败
  39. name: 'hehe.png',
  40. status: 'done',
  41. url: img,
  42. },
  43. ]
  44. : [];
  45. this.setState({
  46. img,
  47. imgList,
  48. });
  49. } else {
  50. return Promise.reject();
  51. }
  52. })
  53. .catch(() => {
  54. message.error('查询出错,请重试');
  55. });
  56. }
  57. handleCancel = () => this.setState({ previewVisible: false });
  58. // 方法:图片预览
  59. handlePreview = (file) => {
  60. console.log('smyhvae handlePreview:' + JSON.stringify(file));
  61. this.setState({
  62. previewImage: file.url || file.thumbUrl,
  63. previewVisible: true,
  64. });
  65. };
  66. // 参考链接:https://www.jianshu.com/p/f356f050b3c9
  67. handleBeforeUpload = (file) => {
  68. console.log('smyhvae handleBeforeUpload file:' + JSON.stringify(file));
  69. console.log('smyhvae handleBeforeUpload file.file:' + JSON.stringify(file.file));
  70. console.log('smyhvae handleBeforeUpload file type:' + JSON.stringify(file.type));
  71. //限制图片 格式、size、分辨率
  72. const isJPG = file.type === 'image/jpeg';
  73. const isJPEG = file.type === 'image/jpeg';
  74. const isGIF = file.type === 'image/gif';
  75. const isPNG = file.type === 'image/png';
  76. const isLt2M = file.size / 1024 / 1024 < 1;
  77. if (!(isJPG || isJPEG || isPNG)) {
  78. Modal.error({
  79. '只能上传JPG、JPEG、PNG格式的图片~',
  80. });
  81. } else if (!isLt2M) {
  82. Modal.error({
  83. '图片超过1M限制,不允许上传~',
  84. });
  85. }
  86. return (isJPG || isJPEG || isPNG) && isLt2M;
  87. };
  88. // checkImageWH 返回一个promise 检测通过返回resolve 失败返回reject阻止图片上传
  89. checkImageWH(file) {
  90. return new Promise(function (resolve, reject) {
  91. let filereader = new FileReader();
  92. filereader.onload = (e) => {
  93. let src = e.target.result;
  94. const image = new Image();
  95. image.onload = function () {
  96. // 获取图片的宽高
  97. file.width = this.width;
  98. file.height = this.height;
  99. resolve();
  100. };
  101. image.onerror = reject;
  102. image.src = src;
  103. };
  104. filereader.readAsDataURL(file);
  105. });
  106. }
  107. // 图片上传
  108. doImgUpload = (options) => {
  109. const { onSuccess, onError, file, onProgress } = options;
  110. // start:进度条相关
  111. // 伪装成 handleChange里面的图片上传状态
  112. const imgItem = {
  113. uid: '1', // 注意,这个uid一定不能少,否则上传失败
  114. name: 'hehe.png',
  115. status: 'uploading',
  116. url: '',
  117. percent: 99, // 注意不要写100。100表示上传完成
  118. };
  119. this.setState({
  120. imgList: [imgItem],
  121. }); // 更新 imgList
  122. // end:进度条相关
  123. const reader = new FileReader();
  124. reader.readAsDataURL(file); // 读取图片文件
  125. reader.onload = (file) => {
  126. const params = {
  127. myBase64: file.target.result, // 把 本地图片的base64编码传给后台,调接口,生成图片的url
  128. };
  129. // 上传图片的base64编码,调接口后,返回 imageId
  130. uploadImage(params)
  131. .then((res) => {
  132. console.log('smyhvae doImgUpload:' + JSON.stringify(res));
  133. console.log('smyhvae 图片上传成功:' + res.imageUrl);
  134. const imgItem = {
  135. uid: '1', // 注意,这个uid一定不能少,否则上传失败
  136. name: 'hehe.png',
  137. status: 'done',
  138. url: res.imageUrl, // url 是展示在页面上的绝对链接
  139. imgUrl: res.imageUrl, // imgUrl 是存到 db 里的相对链接
  140. // response: '{"status": "success"}',
  141. };
  142. this.setState({
  143. imgList: [imgItem],
  144. }); // 更新 imgList
  145. })
  146. .catch((e) => {
  147. console.log('smyhvae 图片上传失败:' + JSON.stringify(e || ''));
  148. message.error('图片上传失败,请重试');
  149. });
  150. };
  151. };
  152. handleChange = ({ file, fileList }) => {
  153. console.log('smyhvae handleChange file:' + JSON.stringify(file));
  154. console.log('smyhvae handleChange fileList:' + JSON.stringify(fileList));
  155. if (file.status == 'removed') {
  156. this.setState({
  157. imgList: [],
  158. });
  159. }
  160. };
  161. submit = (e) => {
  162. e.preventDefault();
  163. this.props.form.validateFields((err, fieldsValue) => {
  164. if (err) {
  165. return;
  166. }
  167. const { id, imgList } = this.state;
  168. const tempImgList = imgList.filter((item) => item.status == 'done'); // 筛选出 status = done 的图片
  169. const imgArr = [];
  170. tempImgList.forEach((item) => {
  171. imgArr.push(item.imgUrl);
  172. // imgArr.push(item.url);
  173. });
  174. submitData({
  175. id,
  176. img: imgArr[0] || '', // 1、暂时只传一张图片给后台。如果传多张图片,那么,upload组件需要进一步完善,比较麻烦,以后有需求再优化。2、如果图片字段是选填,那就用空字符串兜底
  177. })
  178. .then((res) => {
  179. if (res.ret == 0) {
  180. message.success(`${id ? '修改' : '新增'}成功,自动跳转中...`);
  181. } else if (res.ret == 201 || res.ret == 202 || res.ret == 203 || res.ret == 6) {
  182. return Promise.reject(res.msg);
  183. } else {
  184. return Promise.reject();
  185. }
  186. })
  187. .catch((e) => {
  188. message.error(e || '提交失败,请重试');
  189. });
  190. });
  191. };
  192. render() {
  193. const { id, imgList } = this.state;
  194. console.log('smyhvae render imgList:' + JSON.stringify(imgList));
  195. const { getFieldDecorator } = this.props.form;
  196. const formItemLayout = {
  197. labelCol: { span: 3 },
  198. wrapperCol: { span: 10 },
  199. };
  200. const buttonItemLayout = {
  201. wrapperCol: { span: 10, offset: 3 },
  202. };
  203. const uploadButton = (
  204. <div>
  205. <Icon type="plus" />
  206. <div className="ant-upload-text">Upload</div>
  207. </div>
  208. );
  209. return (
  210. <Card title={id ? '修改信息' : '新增信息'}>
  211. <Form onSubmit={this.submit} layout="horizontal">
  212. {/* 新建图片、编辑图片 */}
  213. <FormItem label="图片" {...formItemLayout}>
  214. {getFieldDecorator('img', {
  215. rules: [{ required: false, message: '请上传图片' }],
  216. })(
  217. <Upload
  218. action="2"
  219. customRequest={this.doImgUpload}
  220. listType="picture-card"
  221. fileList={imgList}
  222. onPreview={this.handlePreview}
  223. beforeUpload={this.handleBeforeUpload}
  224. onChange={this.handleChange}
  225. >
  226. {imgList.length >= 1 ? null : uploadButton}
  227. </Upload>
  228. )}
  229. </FormItem>
  230. <Row>
  231. <Col span={3} />
  232. <Col span={18} className={styles.graytext}>
  233. 注:图片支持JPGJPEGPNG格式,小于1M,最多上传1
  234. </Col>
  235. </Row>
  236. <FormItem {...buttonItemLayout}>
  237. <Button type="primary" htmlType="submit">
  238. 提交
  239. </Button>
  240. </FormItem>
  241. </Form>
  242. {/* 图片点开预览 */}
  243. <Modal visible={this.state.previewVisible} footer={null} onCancel={this.handleCancel}>
  244. <img alt="example" style={{ width: '100%' }} src={this.state.previewImage} />
  245. </Modal>
  246. </Card>
  247. );
  248. }
  249. }

参考链接

注意file的格式:https://www.lmonkey.com/t/oREQA5XE1

Demo在线演示:

fileList 格式在线演示:

ant design Upload组件的使用总结:https://www.jianshu.com/p/0aa4612af987

antd上传功能的CustomRequest:https://mlog.club/article/3832743