upload-images.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. <template>
  2. <view class="upload-content">
  3. <block v-for="(item, index) in imageList" :key="index">
  4. <view class="upload-item">
  5. <image class="upload-img" :src="item.filePath" mode="aspectFill" @click="previewImage(index)"></image>
  6. <image class="upload-del-btn"
  7. @click="delImage(index)"
  8. src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACYAAAAmCAYAAACoPemuAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAyJpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMy1jMDExIDY2LjE0NTY2MSwgMjAxMi8wMi8wNi0xNDo1NjoyNyAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvIiB4bWxuczp4bXBNTT0iaHR0cDovL25zLmFkb2JlLmNvbS94YXAvMS4wL21tLyIgeG1sbnM6c3RSZWY9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9zVHlwZS9SZXNvdXJjZVJlZiMiIHhtcDpDcmVhdG9yVG9vbD0iQWRvYmUgUGhvdG9zaG9wIENTNiAoV2luZG93cykiIHhtcE1NOkluc3RhbmNlSUQ9InhtcC5paWQ6RjNBODgzQjUwNDM5MTFFOUJDMjlGN0UwRTJGMjVCNjQiIHhtcE1NOkRvY3VtZW50SUQ9InhtcC5kaWQ6RjNBODgzQjYwNDM5MTFFOUJDMjlGN0UwRTJGMjVCNjQiPiA8eG1wTU06RGVyaXZlZEZyb20gc3RSZWY6aW5zdGFuY2VJRD0ieG1wLmlpZDpGM0E4ODNCMzA0MzkxMUU5QkMyOUY3RTBFMkYyNUI2NCIgc3RSZWY6ZG9jdW1lbnRJRD0ieG1wLmRpZDpGM0E4ODNCNDA0MzkxMUU5QkMyOUY3RTBFMkYyNUI2NCIvPiA8L3JkZjpEZXNjcmlwdGlvbj4gPC9yZGY6UkRGPiA8L3g6eG1wbWV0YT4gPD94cGFja2V0IGVuZD0iciI/PuYQTIAAAAHNSURBVHjazJgxasMwFIbjnsD4BDmCLxDQDZrNQ5bcILlBvXvpDRLIWmiHFkyH2t0LyVbwEkPpbE/eYlUCGYQax096UuIffkIgij7ryU/vyZuYy2e+Z56K70R8lsJcOfPn5AriMCvmZ2YKdMW8EQ/hRA9iEorwXlpZtObMRySQ6kysvrFWloHUEIcmUBuHUDLccmxQskErt7wyVLdy00tQ4Q2g5De2V9kNwWjffiNDA33f/42i6E13Qj5uNptlwJD+SyODuaqqqi/KVBTFOxSKA7VtW/NxSZI8AcbE2nuraZpvKgSBk6G4drsdBOwogz1CwAghH/JEl+BUKJ1VltPHHjoIAoeE4l53YFqb+RKcBSgqyqXhtxEKZwkKB3YOThYCqjPuCDoHZwGK3mGLtdPp5LmqTI1Dqe4p26EkNqA4iE6eg2z+iQ0okyQMATvYgLIIt9Y6knTyFBIudHqIq3BpmkL60VJ9O0sXZY8MBwSLtdNGEAQ/i8XiVXcz83EcEPDbuq/fzMdYWt+6GTmMsX2rh9q3TtsxNrzXhKt1rwg6rR1Dhdjqo7QMlWOvoWTF4imxQMRF7eaL8L5ohmwrLgDB8pCQc8DlcG7y538CDABJNGPqfaJgfgAAAABJRU5ErkJggg=="
  9. mode="scaleToFill">
  10. </image>
  11. <view class="upload-progress" v-if="item.progress < 100">{{item.progress}}%</view>
  12. </view>
  13. </block>
  14. <view class="upload-add-btn" v-if="rduLength > 0" @click="chooseImage"></view>
  15. </view>
  16. </template>
  17. <script>
  18. export default {
  19. data() {
  20. return {
  21. imageList: []
  22. };
  23. },
  24. props: {
  25. url: {
  26. type: String,
  27. value: '' //上传接口地址
  28. },
  29. count: {
  30. type: Number,
  31. value: 4 //单次可选择的图片数量
  32. },
  33. length: {
  34. type: Number,
  35. value: 50 //可上传总数量
  36. }
  37. },
  38. computed: {
  39. rduLength(){
  40. return this.length - this.imageList.length;
  41. }
  42. },
  43. methods: {
  44. //选择图片
  45. chooseImage: function(){
  46. uni.chooseImage({
  47. count: this.rduLength < this.count ? this.rduLength : this.count, //最多可以选择的图片张数,默认9
  48. sizeType: ['original', 'compressed'], //original 原图,compressed 压缩图,默认二者都有
  49. sourceType: ['album'], //album 从相册选图,camera 使用相机,默认二者都有
  50. success: (res)=> {
  51. const images = res.tempFilePaths;
  52. this.uploadFiles(images);
  53. }
  54. });
  55. },
  56. //上传图片
  57. async uploadFiles(images){
  58. this.imageList.push({
  59. filePath: images[0],
  60. progress: 0
  61. });
  62. uni.showLoading({
  63. title: '请稍后..',
  64. mask: true,
  65. })
  66. try{
  67. const uploadUrl = await this.uploadImage(images[0]);
  68. }catch(err){
  69. return;
  70. }
  71. if(uploadUrl !== false){
  72. images.splice(0, 1);
  73. this.imageList[this.imageList.length - 1].src = uploadUrl;
  74. //判断是否需要继续上传
  75. if(images.length > 0 && this.rduLength > 0){
  76. this.uploadFiles(images);
  77. }else{
  78. uni.hideLoading();
  79. }
  80. }else{
  81. //上传失败处理
  82. this.imageList.pop();
  83. uni.hideLoading();
  84. uni.showToast({
  85. title: '上传中出现问题,已终止上传',
  86. icon: 'none',
  87. mask: true,
  88. duration: 2000
  89. });
  90. }
  91. },
  92. uploadImage: function(file){
  93. return new Promise((resolve, reject)=> {
  94. //发送给后端的附加参数
  95. const formData = {
  96. thumb_mode: 1,
  97. };
  98. this.uploadTask = uni.uploadFile({
  99. url: this.url,
  100. filePath: file,
  101. name: 'file',
  102. formData: formData,
  103. success(uploadFileResult){
  104. const uploadFileRes = JSON.parse(uploadFileResult.data) || {};
  105. if(uploadFileRes.status === 1 && uploadFileRes.data){
  106. resolve(uploadFileRes.data);
  107. }else{
  108. reject('接口返回错误');
  109. }
  110. },
  111. fail(){
  112. reject('网络链接错误');
  113. }
  114. });
  115. //上传进度
  116. this.uploadTask.onProgressUpdate((progressRes)=> {
  117. this.imageList[this.imageList.length - 1].progress = progressRes.progress;
  118. });
  119. });
  120. },
  121. //删除图片
  122. delImage: function(index){
  123. uni.showModal({
  124. content: '确定要放弃这张图片么?',
  125. success: (confirmRes)=> {
  126. if (confirmRes.confirm) {
  127. this.imageList.splice(index, 1);
  128. }
  129. }
  130. });
  131. },
  132. //预览图片
  133. previewImage: function(index){
  134. const urls = [];
  135. this.imageList.forEach((item)=> {
  136. urls.push(item.filePath);
  137. })
  138. uni.previewImage({
  139. current: urls[index],
  140. urls: urls,
  141. indicator: "number"
  142. })
  143. }
  144. }
  145. }
  146. </script>
  147. <style lang="scss">
  148. .upload-content{
  149. padding:24upx 0 0 28upx;
  150. background-color: #fff;
  151. overflow:hidden;
  152. }
  153. .upload-item{
  154. position: relative;
  155. float:left;
  156. width:150upx;
  157. height:150upx;
  158. margin-right:30upx;
  159. margin-bottom:30upx;
  160. &:nth-child(4n){
  161. margin-right:0;
  162. }
  163. .upload-img{
  164. width:100%;
  165. height:100%;
  166. border-radius:8upx;
  167. }
  168. .upload-del-btn{
  169. position: absolute;
  170. right:-16upx;
  171. top:-14upx;
  172. width:36upx;
  173. height:36upx;
  174. border: 4upx solid #fff;
  175. border-radius: 100px;
  176. }
  177. .upload-progress{
  178. position: absolute;
  179. left:0;
  180. top:0;
  181. display:flex;
  182. align-items:center;
  183. justify-content: center;
  184. width:100%;
  185. height:100%;
  186. background-color: rgba(0,0,0,.4);
  187. color:#fff;
  188. font-size:24upx;
  189. border-radius:8upx;
  190. }
  191. }
  192. .upload-add-btn {
  193. position: relative;
  194. float:left;
  195. width: 150upx;
  196. height: 150upx;
  197. z-index: 99;
  198. border-radius:8upx;
  199. background:#f9f9f9;
  200. &:before,
  201. &:after {
  202. content: " ";
  203. position: absolute;
  204. top: 50%;
  205. left: 50%;
  206. -webkit-transform: translate(-50%, -50%);
  207. transform: translate(-50%, -50%);
  208. width: 4upx;
  209. height: 60upx;
  210. background-color: #d6d6d6;
  211. }
  212. &:after {
  213. width: 60upx;
  214. height: 4upx;
  215. }
  216. &:active {
  217. background-color: #f7f7f7;
  218. }
  219. }
  220. </style>