sunui-upimg.vue 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. <template>
  2. <view class="s-add-list">
  3. <view class="s-add-list-items" :style="{
  4. height: size[0],
  5. width: size[1]
  6. }" v-for="(item, index) in imageList" :key="index">
  7. <image :src="item[filedImage]" :url="item.url" @tap="showImgs" class="s-add-list-img" :mode="imgMode">
  8. </image>
  9. <view class="s-add-list-remove s-icons icon-close" @tap.stop="removeImg" :id="'s-items-img-' + index">
  10. <icon type="clear" :color="closeColor"></icon>
  11. </view>
  12. <view class="upload-progress" :style="{ width: size[1] }">
  13. <progress :percent="item.progress" :stroke-width="progressSize" :activeColor="progressColor"
  14. :backgroundColor="progressBgColor" />
  15. </view>
  16. <view class="s-add-list-reup" @tap.stop="retry" :data-index="index" v-if="item.error">
  17. <text class="s-add-list-reup-icon s-icons icon-retry"></text>
  18. <text class="s-add-list-reup-text">失败重试</text>
  19. </view>
  20. </view>
  21. <view class="s-add-list-items s-add-list-btn" :style="{
  22. height: size[0],
  23. width: size[1],
  24. backgroundColor: backgroundColor
  25. }" :class="disabled ? 's-disabled' : ''" @tap="addImg" v-if="imageList.length < number">
  26. <slot name="icon"></slot>
  27. <view class="s-add-list-btn-text">{{ title }}</view>
  28. </view>
  29. </view>
  30. </template>
  31. <script>
  32. export default {
  33. props: {
  34. // 图片字段
  35. filedImage: {
  36. type: String,
  37. default: 'url'
  38. },
  39. // 背景颜色
  40. backgroundColor: {
  41. type: String,
  42. default: '#f7f7f7'
  43. },
  44. // 是否禁用
  45. disabled: {
  46. type: Boolean,
  47. default: false
  48. },
  49. // 上传数量
  50. number: {
  51. type: Number,
  52. default: 9
  53. },
  54. // 按钮名称
  55. title: {
  56. type: String,
  57. default: '添加照片'
  58. },
  59. // 上传文字颜色
  60. titleColor: {
  61. type: String,
  62. default: "#999"
  63. },
  64. // 图片大小
  65. size: {
  66. type: Array,
  67. default: () => ['222rpx', '222rpx']
  68. },
  69. // 上传前钩子
  70. beforeUpload: {
  71. type: Function
  72. },
  73. // 关闭按钮颜色
  74. closeColor: {
  75. type: String,
  76. default: "#999"
  77. },
  78. // 服务器地址
  79. url: {
  80. type: String,
  81. default: ''
  82. },
  83. // 进度条进度
  84. progressSize: {
  85. type: Number,
  86. default: 1
  87. },
  88. // 进度条颜色
  89. progressColor: {
  90. type: String,
  91. default: "#09A0F7"
  92. },
  93. // 进度条背景颜色
  94. progressBgColor: {
  95. type: String,
  96. default: "#999"
  97. },
  98. // 上传文件名称
  99. fileName: {
  100. type: String,
  101. default: 'img'
  102. },
  103. // 携带的form数据
  104. formData: {
  105. type: Object,
  106. default: () => {
  107. return {};
  108. }
  109. },
  110. // 图片模式
  111. imgMode: {
  112. type: String,
  113. default: 'widthFix'
  114. },
  115. // 携带的请求头
  116. header: {
  117. type: Object,
  118. default: () => {
  119. return {};
  120. }
  121. },
  122. // 返回状态码,默认0
  123. statusCode: {
  124. type: Number,
  125. default: 0
  126. },
  127. // 返回状态判断字段
  128. statusField: {
  129. type: String,
  130. default: 'errno'
  131. },
  132. },
  133. data() {
  134. return {
  135. imageList: [],
  136. upDate: false
  137. };
  138. },
  139. watch: {
  140. imageList(newVal, oldVal) {
  141. if (!this.upDate) {
  142. this.$emit('change', newVal);
  143. }
  144. }
  145. },
  146. methods: {
  147. clearAllImgs() {
  148. this.imageList = [];
  149. },
  150. addImg() {
  151. if (this.disabled) return;
  152. let num = this.number - this.imageList.length;
  153. if (num < 1) {
  154. return false;
  155. }
  156. uni.chooseImage({
  157. count: num,
  158. sizeType: ['compressed'],
  159. success: async res => {
  160. let file = res.tempFiles;
  161. for (let i = 0; i < res.tempFilePaths.length; i++) {
  162. if (this.beforeUpload) {
  163. const valid = await this.beforeUpload(file[i], i);
  164. if (valid === false) {
  165. return false;
  166. }
  167. }
  168. this.imageList.push({
  169. url: res.tempFilePaths[i],
  170. progress: 0,
  171. error: false
  172. });
  173. }
  174. }
  175. });
  176. },
  177. removeImg(e) {
  178. let index = e.currentTarget.id.replace('s-items-img-', '');
  179. let removeImg = this.imageList.splice(index, 1);
  180. this.$emit('remove', removeImg[0]);
  181. },
  182. showImgs(e) {
  183. let currentImg = e.currentTarget.dataset.url;
  184. let imgs = [];
  185. for (let i = 0; i < this.imageList.length; i++) {
  186. imgs.push(this.imageList[i][this.filedImage]);
  187. }
  188. uni.previewImage({
  189. urls: imgs,
  190. current: currentImg
  191. });
  192. },
  193. upload(index) {
  194. if (this.upDate) {
  195. return;
  196. }
  197. this.upDate = true;
  198. if (!index) {
  199. index = 0;
  200. }
  201. uni.showLoading({
  202. title: '图片上传中'
  203. });
  204. this.uploadBase(index);
  205. },
  206. retry(e) {
  207. let index = e.currentTarget.dataset.index;
  208. this.upload(index);
  209. },
  210. uploadBase(index) {
  211. // 全部上传完成
  212. if (index > this.imageList.length - 1) {
  213. uni.hideLoading();
  214. this.upDate = false;
  215. this.$emit('upload', this.imageList);
  216. return;
  217. }
  218. // 验证后端
  219. if (this.url == '') {
  220. uni.showToast({
  221. title: '请设置上传服务器地址',
  222. icon: 'none'
  223. });
  224. return;
  225. }
  226. // 检查是否是默认值
  227. if (this.imageList[index].progress >= 1) {
  228. this.uploadBase(index + 1);
  229. return;
  230. }
  231. this.imageList[index].error = false;
  232. // 创建上传对象
  233. const upTask = uni.uploadFile({
  234. url: this.url,
  235. filePath: this.imageList[index].url,
  236. name: 'file' || this.fileName,
  237. formData: this.formData,
  238. header: this.header,
  239. // #ifdef MP-ALIPAY
  240. fileType: 'image',
  241. // #endif
  242. success: uploadRes => {
  243. uploadRes = JSON.parse(uploadRes.data);
  244. if (uploadRes[this.statusField] != this.statusCode) {
  245. uni.showToast({
  246. title: '上传失败 : ' + uploadRes.data,
  247. icon: 'none'
  248. });
  249. this.error(index);
  250. } else {
  251. //上传图片成功
  252. this.imageList[index].progress = 100;
  253. this.imageList[index].url = uploadRes.data;
  254. this.imageList[index].result = uploadRes;
  255. this.uploadBase(index + 1);
  256. }
  257. },
  258. fail: e => {
  259. uni.showToast({
  260. title: '上传失败,请点击图片重试',
  261. icon: 'none'
  262. });
  263. this.error(index);
  264. }
  265. });
  266. upTask.onProgressUpdate(res => {
  267. if (res.progress > 0) {
  268. this.imageList[index].progress = res.progress;
  269. this.imageList.splice(index, 1, this.imageList[index]);
  270. }
  271. });
  272. },
  273. // 上传错误
  274. error(index) {
  275. this.upDate = false;
  276. setTimeout(() => {
  277. this.imageList[index].progress = 0;
  278. this.imageList[index].error = true;
  279. this.$emit('uploaderror');
  280. }, 500);
  281. },
  282. // 设置默认值
  283. setItems(items) {
  284. if (items.length) {
  285. this.imageList = [];
  286. for (let i = 0; i < items.length; i++) {
  287. this.imageList.push({
  288. url: items[i],
  289. progress: 100
  290. });
  291. }
  292. }
  293. }
  294. }
  295. };
  296. </script>
  297. <style lang="scss" scoped>
  298. .s-disabled {
  299. cursor: not-allowed;
  300. }
  301. .s-add-list {
  302. display: flex;
  303. flex-wrap: wrap;
  304. }
  305. .s-add-list-btn {
  306. display: flex;
  307. flex-direction: column;
  308. align-items: center;
  309. justify-content: center;
  310. }
  311. .s-add-list-btn-text {
  312. font-size: 1.875rem;
  313. line-height: 36rpx;
  314. text-align: center;
  315. color: #999;
  316. width: 100%;
  317. }
  318. .s-add-list-items {
  319. width: 222rpx;
  320. height: 222rpx;
  321. overflow: hidden;
  322. margin-bottom: 10rpx;
  323. margin-right: 11rpx;
  324. /* background: #f6f7f8; */
  325. font-size: 0;
  326. position: relative;
  327. border-radius: 10rpx;
  328. }
  329. .s-add-list-image {
  330. width: 222rpx;
  331. }
  332. .s-add-list-remove {
  333. position: absolute;
  334. z-index: 15;
  335. right: 10rpx;
  336. top: 0;
  337. color: #888888;
  338. }
  339. .upload-progress {
  340. position: absolute;
  341. z-index: 99;
  342. left: 0;
  343. bottom: 10rpx;
  344. // width: 180rpx;
  345. padding: 0 21rpx;
  346. }
  347. .s-add-list-reup {
  348. position: absolute;
  349. z-index: 3;
  350. left: 0;
  351. top: 0rpx;
  352. width: 100%;
  353. height: 222rpx;
  354. display: flex;
  355. justify-content: center;
  356. align-items: center;
  357. background-color: rgba(0, 0, 0, 0.3);
  358. flex-direction: column;
  359. }
  360. .s-add-list-reup-icon {
  361. text-align: center;
  362. width: 100%;
  363. color: #ffffff;
  364. display: block;
  365. font-size: 80rpx;
  366. line-height: 100rpx;
  367. }
  368. .s-add-list-reup-text {
  369. text-align: center;
  370. width: 100%;
  371. color: #ffffff;
  372. display: block;
  373. font-size: 20rpx;
  374. line-height: 30rpx;
  375. }
  376. .s-add-list-img {
  377. width: 100%;
  378. height: 100%;
  379. }
  380. </style>