mirror of
				https://gitee.com/vant-contrib/vant-weapp.git
				synced 2025-10-26 01:22:07 +08:00 
			
		
		
		
	
							parent
							
								
									0af61992a3
								
							
						
					
					
						commit
						f088e3e6da
					
				| @ -2,7 +2,7 @@ import Page from '../../common/page'; | ||||
| 
 | ||||
| Page({ | ||||
|   data: { | ||||
|     fileList: [], | ||||
|     fileList1: [], | ||||
|     fileList2: [ | ||||
|       { url: 'https://img.yzcdn.cn/vant/leaf.jpg' }, | ||||
|       { url: 'https://img.yzcdn.cn/vant/tree.jpg' } | ||||
|  | ||||
| @ -522,22 +522,29 @@ | ||||
| // Uploader | ||||
| @uploader-size: 80px; | ||||
| @uploader-icon-size: 24px; | ||||
| @uploader-icon-color: @gray-6; | ||||
| @uploader-icon-color: @gray-4; | ||||
| @uploader-text-color: @gray-6; | ||||
| @uploader-text-font-size: @font-size-sm; | ||||
| @uploader-upload-border-color: @gray-3; | ||||
| @uploader-upload-border-radius: 4px; | ||||
| @uploader-upload-background-color: @white; | ||||
| @uploader-upload-border-radius: 8px; | ||||
| @uploader-upload-background-color: @gray-1; | ||||
| @uploader-upload-active-color: @active-color; | ||||
| @uploader-delete-color: @gray-6; | ||||
| @uploader-delete-icon-size: 18px; | ||||
| @uploader-delete-background-color: @white; | ||||
| @uploader-file-background-color: @background-color; | ||||
| @uploader-file-icon-size: 20px; | ||||
| @uploader-file-icon-color: @gray-7; | ||||
| @uploader-file-name-padding: 0 @padding-base; | ||||
| @uploader-file-name-margin-top: @padding-xs; | ||||
| @uploader-file-name-font-size: @font-size-sm; | ||||
| @uploader-file-name-text-color: @gray-7; | ||||
| @uploader-mask-background-color: fade(@gray-8, 88%); | ||||
| @uploader-mask-icon-size: 22px; | ||||
| @uploader-mask-message-font-size: @font-size-sm; | ||||
| @uploader-mask-message-line-height: 14px; | ||||
| @uploader-loading-icon-size: 22px; | ||||
| @uploader-loading-icon-color: @white; | ||||
| @uploader-disabled-opacity: @disabled-opacity; | ||||
| 
 | ||||
| // DropdownMenu | ||||
| @dropdown-menu-height: 50px; | ||||
| @dropdown-menu-background-color: @white; | ||||
|  | ||||
| @ -29,13 +29,13 @@ | ||||
|     height: @uploader-size; | ||||
|     margin: 0 @padding-xs @padding-xs 0; | ||||
|     background-color: @uploader-upload-background-color; | ||||
|     border: 1px dashed @uploader-upload-border-color; | ||||
|     border-radius: @uploader-upload-border-radius; | ||||
| 
 | ||||
|     &:active { | ||||
|       background-color: @uploader-upload-active-color; | ||||
|     } | ||||
| 
 | ||||
|     &-icon { | ||||
|       display: inline-block; | ||||
|       width: 24px; | ||||
|       height: 24px; | ||||
|       color: @uploader-icon-color; | ||||
|       font-size: @uploader-icon-size; | ||||
|     } | ||||
| @ -54,11 +54,13 @@ | ||||
|   &__preview { | ||||
|     position: relative; | ||||
|     margin: 0 @padding-xs @padding-xs 0; | ||||
|     cursor: pointer; | ||||
| 
 | ||||
|     &-image { | ||||
|       display: block; | ||||
|       width: @uploader-size; | ||||
|       height: @uploader-size; | ||||
|       overflow: hidden; | ||||
|       border-radius: @uploader-upload-border-radius; | ||||
|     } | ||||
| 
 | ||||
| @ -84,9 +86,6 @@ | ||||
|     border-radius: @uploader-upload-border-radius; | ||||
| 
 | ||||
|     &-icon { | ||||
|       display: inline-block; | ||||
|       width: 20px; | ||||
|       height: 20px; | ||||
|       color: @uploader-file-icon-color; | ||||
|       font-size: @uploader-file-icon-size; | ||||
|     } | ||||
| @ -94,8 +93,8 @@ | ||||
|     &-name { | ||||
|       box-sizing: border-box; | ||||
|       width: 100%; | ||||
|       margin-top: @padding-xs; | ||||
|       padding: 0 5px; | ||||
|       margin-top: @uploader-file-name-margin-top; | ||||
|       padding: @uploader-file-name-padding; | ||||
|       color: @uploader-file-name-text-color; | ||||
|       font-size: @uploader-file-name-font-size; | ||||
|       text-align: center; | ||||
|  | ||||
| @ -1,5 +1,6 @@ | ||||
| import { VantComponent } from '../common/component'; | ||||
| import { isImageFile, isVideo } from './utils'; | ||||
| import { isImageFile, isVideo, chooseFile, isPromise } from './utils'; | ||||
| import { chooseImageProps, chooseVideoProps } from './shared'; | ||||
| 
 | ||||
| VantComponent({ | ||||
|   props: { | ||||
| @ -7,6 +8,8 @@ VantComponent({ | ||||
|     multiple: Boolean, | ||||
|     uploadText: String, | ||||
|     useBeforeRead: Boolean, | ||||
|     afterRead: null, | ||||
|     beforeRead: null, | ||||
|     previewSize: { | ||||
|       type: null, | ||||
|       value: 90 | ||||
| @ -19,14 +22,6 @@ VantComponent({ | ||||
|       type: String, | ||||
|       value: 'image' | ||||
|     }, | ||||
|     sizeType: { | ||||
|       type: Array, | ||||
|       value: ['original', 'compressed'] | ||||
|     }, | ||||
|     capture: { | ||||
|       type: Array, | ||||
|       value: ['album', 'camera'] | ||||
|     }, | ||||
|     fileList: { | ||||
|       type: Array, | ||||
|       value: [], | ||||
| @ -60,27 +55,16 @@ VantComponent({ | ||||
|       type: String, | ||||
|       value: 'scaleToFill' | ||||
|     }, | ||||
|     camera: { | ||||
|       type: String, | ||||
|       value: 'back' | ||||
|     }, | ||||
|     compressed: { | ||||
|       type: Boolean, | ||||
|       value: true | ||||
|     }, | ||||
|     maxDuration: { | ||||
|       type: Number, | ||||
|       value: 60 | ||||
|     }, | ||||
|     uploadIcon: { | ||||
|       type: String, | ||||
|       value: 'plus' | ||||
|     } | ||||
|       value: 'photograph' | ||||
|     }, | ||||
|     ...chooseImageProps, | ||||
|     ...chooseVideoProps | ||||
|   }, | ||||
| 
 | ||||
|   data: { | ||||
|     lists: [], | ||||
|     computedPreviewSize: '', | ||||
|     isInCount: true | ||||
|   }, | ||||
| 
 | ||||
| @ -95,128 +79,116 @@ VantComponent({ | ||||
|       this.setData({ lists, isInCount: lists.length < maxCount }); | ||||
|     }, | ||||
| 
 | ||||
|     getDetail(index) { | ||||
|       return { | ||||
|         name: this.data.name, | ||||
|         index: index == null ? this.data.fileList.length : index | ||||
|       }; | ||||
|     }, | ||||
| 
 | ||||
|     startUpload() { | ||||
|       if (this.data.disabled) return; | ||||
|       const { | ||||
|         name = '', | ||||
|         capture, | ||||
|         maxCount, | ||||
|         multiple, | ||||
|         maxSize, | ||||
|         accept, | ||||
|         sizeType, | ||||
|         lists, | ||||
|         camera, | ||||
|         compressed, | ||||
|         maxDuration, | ||||
|         useBeforeRead = false // 是否定义了 beforeRead
 | ||||
|       } = this.data; | ||||
|       const { maxCount, multiple, accept, lists, disabled } = this.data; | ||||
| 
 | ||||
|       let chooseFile = null; | ||||
|       const newMaxCount = maxCount - lists.length; | ||||
|       // 设置为只选择图片的时候使用 chooseImage 来实现
 | ||||
|       if (accept === 'image') { | ||||
|         chooseFile = new Promise((resolve, reject) => { | ||||
|           wx.chooseImage({ | ||||
|             count: multiple ? (newMaxCount > 9 ? 9 : newMaxCount) : 1, // 最多可以选择的数量,如果不支持多选则数量为1
 | ||||
|             sourceType: capture, // 选择图片的来源,相册还是相机
 | ||||
|             sizeType, | ||||
|             success: resolve, | ||||
|             fail: reject | ||||
|           }); | ||||
|         }); | ||||
|       } else if (accept === 'video') { | ||||
|         chooseFile = new Promise((resolve, reject) => { | ||||
|           wx.chooseVideo({ | ||||
|             sourceType: capture, | ||||
|             compressed, | ||||
|             maxDuration, | ||||
|             camera, | ||||
|             success: resolve, | ||||
|             fail: reject | ||||
|           }); | ||||
|         }); | ||||
|       } else { | ||||
|         chooseFile = new Promise((resolve, reject) => { | ||||
|           wx.chooseMessageFile({ | ||||
|             count: multiple ? newMaxCount : 1, // 最多可以选择的数量,如果不支持多选则数量为1
 | ||||
|             type: 'file', | ||||
|             success: resolve, | ||||
|             fail: reject | ||||
|           }); | ||||
|         }); | ||||
|       } | ||||
|       if (disabled) return; | ||||
| 
 | ||||
|       chooseFile | ||||
|         .then( | ||||
|           ( | ||||
|             res: | ||||
|               | WechatMiniprogram.ChooseImageSuccessCallbackResult | ||||
|               | WechatMiniprogram.ChooseMessageFileSuccessCallbackResult | ||||
|               | WechatMiniprogram.ChooseVideoSuccessCallbackResult | ||||
|           ) => { | ||||
|             let file = null; | ||||
|       chooseFile({ | ||||
|         ...this.data, | ||||
|         maxCount: maxCount - lists.length | ||||
|       }) | ||||
|         .then(res => { | ||||
|           let file = null; | ||||
| 
 | ||||
|             if (isVideo(res, accept)) { | ||||
|               file = { | ||||
|                 path: res.tempFilePath, | ||||
|                 ...res | ||||
|               }; | ||||
|             } else { | ||||
|               file = multiple ? res.tempFiles : res.tempFiles[0]; | ||||
|             } | ||||
| 
 | ||||
|             // 检查文件大小
 | ||||
|             if (file instanceof Array) { | ||||
|               const sizeEnable = file.every(item => item.size <= maxSize); | ||||
|               if (!sizeEnable) { | ||||
|                 this.$emit('oversize', { name }); | ||||
|                 return; | ||||
|               } | ||||
|             } else if (file.size > maxSize) { | ||||
|               this.$emit('oversize', { name }); | ||||
|               return; | ||||
|             } | ||||
| 
 | ||||
|             // 触发上传之前的钩子函数
 | ||||
|             if (useBeforeRead) { | ||||
|               this.$emit('before-read', { | ||||
|                 file, | ||||
|                 name, | ||||
|                 callback: (result: boolean) => { | ||||
|                   if (result) { | ||||
|                     // 开始上传
 | ||||
|                     this.$emit('after-read', { file, name }); | ||||
|                   } | ||||
|                 } | ||||
|               }); | ||||
|             } else { | ||||
|               this.$emit('after-read', { file, name }); | ||||
|             } | ||||
|           if (isVideo(res, accept)) { | ||||
|             file = { | ||||
|               path: res.tempFilePath, | ||||
|               ...res | ||||
|             }; | ||||
|           } else { | ||||
|             file = multiple ? res.tempFiles : res.tempFiles[0]; | ||||
|           } | ||||
|         ) | ||||
| 
 | ||||
|           this.onBeforeRead(file); | ||||
|         }) | ||||
|         .catch(error => { | ||||
|           this.$emit('error', error); | ||||
|         }); | ||||
|     }, | ||||
| 
 | ||||
|     deleteItem(event) { | ||||
|       const { index } = event.currentTarget.dataset; | ||||
|       this.$emit('delete', { index, name: this.data.name }); | ||||
|     onBeforeRead(file) { | ||||
|       const { beforeRead, useBeforeRead } = this.data; | ||||
|       let res: boolean | Promise<any> = true; | ||||
| 
 | ||||
|       if (typeof beforeRead === 'function') { | ||||
|         res = beforeRead(file, this.getDetail()); | ||||
|       } | ||||
| 
 | ||||
|       if (useBeforeRead) { | ||||
|         res = new Promise((resolve, reject) => { | ||||
|           this.$emit('before-read', { | ||||
|             file, | ||||
|             ...this.getDetail(), | ||||
|             callback: (ok: boolean) => { | ||||
|               ok ? resolve() : reject(); | ||||
|             } | ||||
|           }); | ||||
|         }); | ||||
|       } | ||||
| 
 | ||||
|       if (!res) { | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       if (isPromise(res)) { | ||||
|         res.then((data: any) => this.onAfterRead(data || file)); | ||||
|       } else { | ||||
|         this.onAfterRead(file); | ||||
|       } | ||||
|     }, | ||||
| 
 | ||||
|     doPreviewImage(event) { | ||||
|       if (!this.data.previewFullImage) return; | ||||
|       const curUrl = event.currentTarget.dataset.url; | ||||
|       const images = this.data.lists | ||||
|         .filter(item => item.isImage) | ||||
|         .map(item => item.url || item.path); | ||||
|     onAfterRead(file) { | ||||
|       const { maxSize } = this.data; | ||||
|       const oversize = Array.isArray(file) | ||||
|         ? file.some(item => item.size > maxSize) | ||||
|         : file.size > maxSize; | ||||
| 
 | ||||
|       this.$emit('click-preview', { url: curUrl, name: this.data.name }); | ||||
|       if (oversize) { | ||||
|         this.$emit('oversize', { file, ...this.getDetail() }); | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       if (typeof this.data.afterRead === 'function') { | ||||
|         this.data.afterRead(file, this.getDetail()); | ||||
|       } | ||||
| 
 | ||||
|       this.$emit('after-read', { file, ...this.getDetail() }); | ||||
|     }, | ||||
| 
 | ||||
|     deleteItem(event) { | ||||
|       const { index } = event.currentTarget.dataset; | ||||
| 
 | ||||
|       this.$emit('delete', { | ||||
|         ...this.getDetail(index), | ||||
|         file: this.data.fileList[index] | ||||
|       }); | ||||
|     }, | ||||
| 
 | ||||
|     onPreviewImage(event) { | ||||
|       const { index } = event.currentTarget.dataset; | ||||
|       const { lists } = this.data; | ||||
|       const item = lists[index]; | ||||
| 
 | ||||
|       this.$emit('click-preview', { | ||||
|         url: item.url || item.path, | ||||
|         ...this.getDetail(index) | ||||
|       }); | ||||
| 
 | ||||
|       if (!this.data.previewFullImage) return; | ||||
| 
 | ||||
|       wx.previewImage({ | ||||
|         urls: images, | ||||
|         current: curUrl, | ||||
|         urls: lists | ||||
|           .filter(item => item.isImage) | ||||
|           .map(item => item.url || item.path), | ||||
|         current: item.url || item.path, | ||||
|         fail() { | ||||
|           wx.showToast({ title: '预览图片失败', icon: 'none' }); | ||||
|         } | ||||
|  | ||||
| @ -16,8 +16,8 @@ | ||||
|         alt="{{ item.name || ('图片' + index) }}" | ||||
|         class="van-uploader__preview-image" | ||||
|         style="width: {{ utils.addUnit(previewSize) }}; height: {{ utils.addUnit(previewSize) }};" | ||||
|         data-url="{{ item.url || item.path }}" | ||||
|         bind:tap="doPreviewImage" | ||||
|         data-index="{{ index }}" | ||||
|         bind:tap="onPreviewImage" | ||||
|       /> | ||||
|       <view | ||||
|         wx:else | ||||
|  | ||||
							
								
								
									
										31
									
								
								packages/uploader/shared.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								packages/uploader/shared.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | ||||
| // props for choose image
 | ||||
| export const chooseImageProps = { | ||||
|   sizeType: { | ||||
|     type: Array, | ||||
|     value: ['original', 'compressed'] | ||||
|   }, | ||||
|   capture: { | ||||
|     type: Array, | ||||
|     value: ['album', 'camera'] | ||||
|   } | ||||
| }; | ||||
| 
 | ||||
| // props for choose video
 | ||||
| export const chooseVideoProps = { | ||||
|   capture: { | ||||
|     type: Array, | ||||
|     value: ['album', 'camera'] | ||||
|   }, | ||||
|   compressed: { | ||||
|     type: Boolean, | ||||
|     value: true | ||||
|   }, | ||||
|   maxDuration: { | ||||
|     type: Number, | ||||
|     value: 60 | ||||
|   }, | ||||
|   camera: { | ||||
|     type: String, | ||||
|     value: 'back' | ||||
|   } | ||||
| }; | ||||
| @ -31,8 +31,67 @@ export function isImageFile(item: File): boolean { | ||||
| } | ||||
| 
 | ||||
| export function isVideo( | ||||
|   res, | ||||
|   accept | ||||
|   res: any, | ||||
|   accept: string | ||||
| ): res is WechatMiniprogram.ChooseVideoSuccessCallbackResult { | ||||
|   return accept === 'video'; | ||||
| } | ||||
| 
 | ||||
| export function chooseFile({ | ||||
|   accept, | ||||
|   multiple, | ||||
|   capture, | ||||
|   compressed, | ||||
|   maxDuration, | ||||
|   sizeType, | ||||
|   camera, | ||||
|   maxCount | ||||
| }): Promise< | ||||
|   | WechatMiniprogram.ChooseImageSuccessCallbackResult | ||||
|   | WechatMiniprogram.ChooseVideoSuccessCallbackResult | ||||
|   | WechatMiniprogram.ChooseMessageFileSuccessCallbackResult | ||||
| > { | ||||
|   if (accept === 'image') { | ||||
|     return new Promise((resolve, reject) => { | ||||
|       wx.chooseImage({ | ||||
|         count: multiple ? Math.min(maxCount, 9) : 1, // 最多可以选择的数量,如果不支持多选则数量为1
 | ||||
|         sourceType: capture, // 选择图片的来源,相册还是相机
 | ||||
|         sizeType, | ||||
|         success: resolve, | ||||
|         fail: reject | ||||
|       }); | ||||
|     }); | ||||
|   } | ||||
|   if (accept === 'video') { | ||||
|     return new Promise((resolve, reject) => { | ||||
|       wx.chooseVideo({ | ||||
|         sourceType: capture, | ||||
|         compressed, | ||||
|         maxDuration, | ||||
|         camera, | ||||
|         success: resolve, | ||||
|         fail: reject | ||||
|       }); | ||||
|     }); | ||||
|   } | ||||
|   return new Promise((resolve, reject) => { | ||||
|     wx.chooseMessageFile({ | ||||
|       count: multiple ? maxCount : 1, // 最多可以选择的数量,如果不支持多选则数量为1
 | ||||
|       type: 'file', | ||||
|       success: resolve, | ||||
|       fail: reject | ||||
|     }); | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| export function isFunction(val: unknown): val is Function { | ||||
|   return typeof val === 'function'; | ||||
| } | ||||
| 
 | ||||
| export function isObject(val: any): val is Record<any, any> { | ||||
|   return val !== null && typeof val === 'object'; | ||||
| } | ||||
| 
 | ||||
| export function isPromise<T = any>(val: unknown): val is Promise<T> { | ||||
|   return isObject(val) && isFunction(val.then) && isFunction(val.catch); | ||||
| } | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user