¶Ô±ÈÐÂÎļþ |
| | |
| | | <template> |
| | | <view class="u-upload" :style="[$u.addStyle(customStyle)]"> |
| | | <view class="u-upload__wrap" > |
| | | <template v-if="previewImage"> |
| | | <view |
| | | class="u-upload__wrap__preview" |
| | | v-for="(item, index) in lists" |
| | | :key="index" |
| | | > |
| | | <image |
| | | v-if="item.isImage || (item.type && item.type === 'image')" |
| | | :src="item.thumb || item.url" |
| | | :mode="imageMode" |
| | | class="u-upload__wrap__preview__image" |
| | | @tap="onPreviewImage(item)" |
| | | :style="[{ |
| | | width: $u.addUnit(width), |
| | | height: $u.addUnit(height) |
| | | }]" |
| | | /> |
| | | <view |
| | | v-else |
| | | class="u-upload__wrap__preview__other" |
| | | > |
| | | <u-icon |
| | | color="#80CBF9" |
| | | size="26" |
| | | :name="item.isVideo || (item.type && item.type === 'video') ? 'movie' : 'folder'" |
| | | ></u-icon> |
| | | <text class="u-upload__wrap__preview__other__text">{{item.isVideo || (item.type && item.type === 'video') ? 'è§é¢' : 'æä»¶'}}</text> |
| | | </view> |
| | | <view |
| | | class="u-upload__status" |
| | | v-if="item.status === 'uploading' || item.status === 'failed'" |
| | | > |
| | | <view class="u-upload__status__icon"> |
| | | <u-icon |
| | | v-if="item.status === 'failed'" |
| | | name="close-circle" |
| | | color="#ffffff" |
| | | size="25" |
| | | /> |
| | | <u-loading-icon |
| | | size="22" |
| | | mode="circle" |
| | | color="#ffffff" |
| | | v-else |
| | | /> |
| | | </view> |
| | | <text |
| | | v-if="item.message" |
| | | class="u-upload__status__message" |
| | | >{{ item.message }}</text> |
| | | </view> |
| | | <view |
| | | class="u-upload__deletable" |
| | | v-if="item.status !== 'uploading' && (deletable || item.deletable)" |
| | | @tap.stop="deleteItem(index)" |
| | | > |
| | | <view class="u-upload__deletable__icon"> |
| | | <u-icon |
| | | name="close" |
| | | color="#ffffff" |
| | | size="10" |
| | | ></u-icon> |
| | | </view> |
| | | </view> |
| | | <view |
| | | class="u-upload__success" |
| | | v-if="item.status === 'success'" |
| | | > |
| | | <!-- #ifdef APP-NVUE --> |
| | | <image |
| | | :src="successIcon" |
| | | class="u-upload__success__icon" |
| | | ></image> |
| | | <!-- #endif --> |
| | | <!-- #ifndef APP-NVUE --> |
| | | <view class="u-upload__success__icon"> |
| | | <u-icon |
| | | name="checkmark" |
| | | color="#ffffff" |
| | | size="12" |
| | | ></u-icon> |
| | | </view> |
| | | <!-- #endif --> |
| | | </view> |
| | | </view> |
| | | |
| | | </template> |
| | | |
| | | <template v-if="isInCount"> |
| | | <view |
| | | v-if="$slots.default || $slots.$default" |
| | | @tap="chooseFile" |
| | | > |
| | | <slot /> |
| | | </view> |
| | | <view |
| | | v-else |
| | | class="u-upload__button" |
| | | :hover-class="!disabled ? 'u-upload__button--hover' : ''" |
| | | hover-stay-time="150" |
| | | @tap="chooseFile" |
| | | :class="[disabled && 'u-upload__button--disabled']" |
| | | :style="[{ |
| | | width: $u.addUnit(width), |
| | | height: $u.addUnit(height) |
| | | }]" |
| | | > |
| | | <u-icon |
| | | :name="uploadIcon" |
| | | size="26" |
| | | :color="uploadIconColor" |
| | | ></u-icon> |
| | | <text |
| | | v-if="uploadText" |
| | | class="u-upload__button__text" |
| | | >{{ uploadText }}</text> |
| | | </view> |
| | | </template> |
| | | </view> |
| | | |
| | | </view> |
| | | </template> |
| | | |
| | | <script> |
| | | import { |
| | | chooseFile |
| | | } from './utils'; |
| | | import mixin from './mixin.js'; |
| | | import props from './props.js'; |
| | | |
| | | /** |
| | | * upload ä¸ä¼ |
| | | * @description 该ç»ä»¶ç¨äºä¸ä¼ å¾çåºæ¯ |
| | | * @tutorial https://uviewui.com/components/upload.html |
| | | * @property {String} accept æ¥åçæä»¶ç±»å, å¯éå¼ä¸ºall media image file video ï¼é»è®¤ 'image' ï¼ |
| | | * @property {String | Array} capture å¾çæè§é¢æ¾å模å¼ï¼å½accept为imageç±»åæ¶è®¾ç½®captureå¯éé¢å¤cameraå¯ä»¥ç´æ¥è°èµ·æå头ï¼é»è®¤ ['album', 'camera'] ï¼ |
| | | * @property {Boolean} compressed å½accept为videoæ¶çæï¼æ¯å¦å缩è§é¢ï¼é»è®¤ä¸ºtrueï¼é»è®¤ true ï¼ |
| | | * @property {String} camera å½accept为videoæ¶çæï¼å¯éå¼ä¸ºbackæfrontï¼é»è®¤ 'back' ï¼ |
| | | * @property {Number} maxDuration å½accept为videoæ¶çæï¼ææè§é¢æé¿æææ¶é´ï¼åä½ç§ï¼é»è®¤ 60 ï¼ |
| | | * @property {String} uploadIcon ä¸ä¼ åºåç徿 ï¼åªè½å
ç½®å¾æ ï¼é»è®¤ 'camera-fill' ï¼ |
| | | * @property {String} uploadIconColor ä¸ä¼ åºåç徿 çåä½é¢è²ï¼åªè½å
ç½®å¾æ ï¼é»è®¤ #D3D4D6 ï¼ |
| | | * @property {Boolean} useBeforeRead æ¯å¦å¼å¯æä»¶è¯»ååäºä»¶ï¼é»è®¤ false ï¼ |
| | | * @property {Boolean} previewFullImage æ¯å¦æ¾ç¤ºç»ä»¶èªå¸¦çå¾çé¢è§åè½ï¼é»è®¤ true ï¼ |
| | | * @property {String | Number} maxCount æå¤§ä¸ä¼ æ°éï¼é»è®¤ 52 ï¼ |
| | | * @property {Boolean} disabled æ¯å¦å¯ç¨ï¼é»è®¤ false ï¼ |
| | | * @property {String} imageMode é¢è§ä¸ä¼ çå¾çæ¶çè£åªæ¨¡å¼ï¼åimageç»ä»¶mode屿§ä¸è´ï¼é»è®¤ 'aspectFill' ï¼ |
| | | * @property {String} name æ è¯ç¬¦ï¼å¯ä»¥å¨åè°å½æ°ç第äºé¡¹åæ°ä¸è·å |
| | | * @property {Array} sizeType æéçå¾çç尺寸, å¯éå¼ä¸ºoriginal compressedï¼é»è®¤ ['original', 'compressed'] ï¼ |
| | | * @property {Boolean} multiple æ¯å¦å¼å¯å¾çå¤éï¼é¨åå®åæºå䏿¯æ ï¼é»è®¤ false ï¼ |
| | | * @property {Boolean} deletable æ¯å¦å±ç¤ºå 餿é®ï¼é»è®¤ true ï¼ |
| | | * @property {String | Number} maxSize æä»¶å¤§å°éå¶ï¼åä½ä¸ºbyte ï¼é»è®¤ Number.MAX_VALUE ï¼ |
| | | * @property {Array} fileList æ¾ç¤ºå·²ä¸ä¼ çæä»¶å表 |
| | | * @property {String} uploadText ä¸ä¼ åºåçæç¤ºæå |
| | | * @property {String | Number} width å
é¨é¢è§å¾çåºååéæ©å¾çæé®çåºå宽度ï¼é»è®¤ 80 ï¼ |
| | | * @property {String | Number} height å
é¨é¢è§å¾çåºååéæ©å¾çæé®çåºåé«åº¦ï¼é»è®¤ 80 ï¼ |
| | | * @property {Object} customStyle ç»ä»¶çæ ·å¼ï¼å¯¹è±¡å½¢å¼ |
| | | * @event {Function} afterRead 读ååçå¤ç彿° |
| | | * @event {Function} beforeRead 读ååçå¤ç彿° |
| | | * @event {Function} oversize æä»¶è¶
åºå¤§å°éå¶ |
| | | * @event {Function} clickPreview ç¹å»é¢è§å¾ç |
| | | * @event {Function} delete å é¤å¾ç |
| | | * @example <u-upload :action="action" :fileList="fileList" ></u-upload> |
| | | */ |
| | | export default { |
| | | name: "u-upload", |
| | | mixins: [uni.$u.mpMixin, uni.$u.mixin, mixin,props], |
| | | data() { |
| | | return { |
| | | // #ifdef APP-NVUE |
| | | successIcon: '', |
| | | // #endif |
| | | lists: [], |
| | | isInCount: true, |
| | | } |
| | | }, |
| | | watch: { |
| | | // ç嬿件å表çååï¼éæ°æ´çå
鍿°æ® |
| | | fileList: { |
| | | immediate: true, |
| | | handler() { |
| | | this.formatFileList() |
| | | } |
| | | }, |
| | | }, |
| | | methods: { |
| | | formatFileList() { |
| | | const { |
| | | fileList = [], maxCount |
| | | } = this; |
| | | const lists = fileList.map((item) => |
| | | Object.assign(Object.assign({}, item), { |
| | | // 妿item.url为æ¬å°éæ©çblobæä»¶çè¯ï¼æ æ³å¤æå
¶ä¸ºvideoè¿æ¯imageï¼æ¤å¤ä¼å
éè¿acceptå夿å¤ç |
| | | isImage: this.accept === 'image' || uni.$u.test.image(item.url || item.thumb), |
| | | isVideo: this.accept === 'video' || uni.$u.test.video(item.url || item.thumb), |
| | | deletable: typeof(item.deletable) === 'boolean' ? item.deletable : this.deletable, |
| | | }) |
| | | ); |
| | | this.lists = lists |
| | | this.isInCount = lists.length < maxCount |
| | | }, |
| | | chooseFile() { |
| | | const { |
| | | maxCount, |
| | | multiple, |
| | | lists, |
| | | disabled |
| | | } = this; |
| | | if (disabled) return; |
| | | // å¦æç¨æ·ä¼ å
¥çæ¯å符串ï¼éè¦æ ¼å¼åææ°ç» |
| | | let capture; |
| | | try { |
| | | capture = uni.$u.test.array(this.capture) ? this.capture : this.capture.split(','); |
| | | }catch(e) { |
| | | capture = []; |
| | | } |
| | | chooseFile( |
| | | Object.assign({ |
| | | accept: this.accept, |
| | | multiple: this.multiple, |
| | | capture: capture, |
| | | compressed: this.compressed, |
| | | maxDuration: this.maxDuration, |
| | | sizeType: this.sizeType, |
| | | camera: this.camera, |
| | | }, { |
| | | maxCount: maxCount - lists.length, |
| | | }) |
| | | ) |
| | | .then((res) => { |
| | | this.onBeforeRead(multiple ? res : res[0]); |
| | | }) |
| | | .catch((error) => { |
| | | this.$emit('error', error); |
| | | }); |
| | | }, |
| | | // æä»¶è¯»åä¹å |
| | | onBeforeRead(file) { |
| | | const { |
| | | beforeRead, |
| | | useBeforeRead, |
| | | } = this; |
| | | let res = true |
| | | // beforeReadæ¯å¦ä¸ºä¸ä¸ªæ¹æ³ |
| | | if (uni.$u.test.func(beforeRead)) { |
| | | // å¦æç¨æ·å®ä¹äºæ¤æ¹æ³ï¼å廿§è¡æ¤æ¹æ³ï¼å¹¶ä¼ å
¥è¯»åçæä»¶åè° |
| | | res = beforeRead(file, this.getDetail()); |
| | | } |
| | | if (useBeforeRead) { |
| | | res = new Promise((resolve, reject) => { |
| | | this.$emit( |
| | | 'beforeRead', |
| | | Object.assign(Object.assign({ |
| | | file |
| | | }, this.getDetail()), { |
| | | callback: (ok) => { |
| | | ok ? resolve() : reject(); |
| | | }, |
| | | }) |
| | | ); |
| | | }); |
| | | } |
| | | if (!res) { |
| | | return; |
| | | } |
| | | if (uni.$u.test.promise(res)) { |
| | | res.then((data) => this.onAfterRead(data || file)); |
| | | } else { |
| | | this.onAfterRead(file); |
| | | } |
| | | }, |
| | | getDetail(index) { |
| | | return { |
| | | name: this.name, |
| | | index: index == null ? this.fileList.length : index, |
| | | }; |
| | | }, |
| | | onAfterRead(file) { |
| | | const { |
| | | maxSize, |
| | | afterRead |
| | | } = this; |
| | | const oversize = Array.isArray(file) ? |
| | | file.some((item) => item.size > maxSize) : |
| | | file.size > maxSize; |
| | | if (oversize) { |
| | | this.$emit('oversize', Object.assign({ |
| | | file |
| | | }, this.getDetail())); |
| | | return; |
| | | } |
| | | if (typeof afterRead === 'function') { |
| | | afterRead(file, this.getDetail()); |
| | | } |
| | | this.$emit('afterRead', Object.assign({ |
| | | file |
| | | }, this.getDetail())); |
| | | }, |
| | | deleteItem(index) { |
| | | this.$emit( |
| | | 'delete', |
| | | Object.assign(Object.assign({}, this.getDetail(index)), { |
| | | file: this.fileList[index], |
| | | }) |
| | | ); |
| | | }, |
| | | // é¢è§å¾ç |
| | | onPreviewImage(item) { |
| | | if (!item.isImage || !this.previewFullImage) return |
| | | uni.previewImage({ |
| | | // å
filteræ¾åºä¸ºå¾ççitemï¼åè¿åfilterç»æä¸çå¾çurl |
| | | urls: this.lists.filter((item) => this.accept === 'image' || uni.$u.test.image(item.url || item.thumb)).map((item) => item.url || item.thumb), |
| | | current: item.url || item.thumb, |
| | | fail() { |
| | | uni.$u.toast('é¢è§å¾ç失败') |
| | | }, |
| | | }); |
| | | }, |
| | | onPreviewVideo(event) { |
| | | if (!this.data.previewFullImage) return; |
| | | const { |
| | | index |
| | | } = event.currentTarget.dataset; |
| | | const { |
| | | lists |
| | | } = this.data; |
| | | wx.previewMedia({ |
| | | sources: lists |
| | | .filter((item) => isVideoFile(item)) |
| | | .map((item) => |
| | | Object.assign(Object.assign({}, item), { |
| | | type: 'video' |
| | | }) |
| | | ), |
| | | current: index, |
| | | fail() { |
| | | uni.$u.toast('é¢è§è§é¢å¤±è´¥') |
| | | }, |
| | | }); |
| | | }, |
| | | onClickPreview(event) { |
| | | const { |
| | | index |
| | | } = event.currentTarget.dataset; |
| | | const item = this.data.lists[index]; |
| | | this.$emit( |
| | | 'clickPreview', |
| | | Object.assign(Object.assign({}, item), this.getDetail(index)) |
| | | ); |
| | | } |
| | | } |
| | | } |
| | | </script> |
| | | |
| | | <style lang="scss" scoped> |
| | | @import '../../libs/css/components.scss'; |
| | | $u-upload-preview-border-radius: 2px !default; |
| | | $u-upload-preview-margin: 0 8px 8px 0 !default; |
| | | $u-upload-image-width:80px !default; |
| | | $u-upload-image-height:$u-upload-image-width; |
| | | $u-upload-other-bgColor: rgb(242, 242, 242) !default; |
| | | $u-upload-other-flex:1 !default; |
| | | $u-upload-text-font-size:11px !default; |
| | | $u-upload-text-color:$u-tips-color !default; |
| | | $u-upload-text-margin-top:2px !default; |
| | | $u-upload-deletable-right:0 !default; |
| | | $u-upload-deletable-top:0 !default; |
| | | $u-upload-deletable-bgColor:rgb(55, 55, 55) !default; |
| | | $u-upload-deletable-height:14px !default; |
| | | $u-upload-deletable-width:$u-upload-deletable-height; |
| | | $u-upload-deletable-boder-bottom-left-radius:100px !default; |
| | | $u-upload-deletable-zIndex:3 !default; |
| | | $u-upload-success-bottom:0 !default; |
| | | $u-upload-success-right:0 !default; |
| | | $u-upload-success-border-style:solid !default; |
| | | $u-upload-success-border-top-color:transparent !default; |
| | | $u-upload-success-border-left-color:transparent !default; |
| | | $u-upload-success-border-bottom-color: $u-success !default; |
| | | $u-upload-success-border-right-color:$u-upload-success-border-bottom-color; |
| | | $u-upload-success-border-width:9px !default; |
| | | $u-upload-icon-top:0px !default; |
| | | $u-upload-icon-right:0px !default; |
| | | $u-upload-icon-h5-top:1px !default; |
| | | $u-upload-icon-h5-right:0 !default; |
| | | $u-upload-icon-width:16px !default; |
| | | $u-upload-icon-height:$u-upload-icon-width; |
| | | $u-upload-success-icon-bottom:-10px !default; |
| | | $u-upload-success-icon-right:-10px !default; |
| | | $u-upload-status-right:0 !default; |
| | | $u-upload-status-left:0 !default; |
| | | $u-upload-status-bottom:0 !default; |
| | | $u-upload-status-top:0 !default; |
| | | $u-upload-status-bgColor:rgba(0, 0, 0, 0.5) !default; |
| | | $u-upload-status-icon-Zindex:1 !default; |
| | | $u-upload-message-font-size:12px !default; |
| | | $u-upload-message-color:#FFFFFF !default; |
| | | $u-upload-message-margin-top:5px !default; |
| | | $u-upload-button-width:80px !default; |
| | | $u-upload-button-height:$u-upload-button-width; |
| | | $u-upload-button-bgColor:rgb(244, 245, 247) !default; |
| | | $u-upload-button-border-radius:2px !default; |
| | | $u-upload-botton-margin: 0 8px 8px 0 !default; |
| | | $u-upload-text-font-size:11px !default; |
| | | $u-upload-text-color:$u-tips-color !default; |
| | | $u-upload-text-margin-top: 2px !default; |
| | | $u-upload-hover-bgColor:rgb(230, 231, 233) !default; |
| | | $u-upload-disabled-opacity:.5 !default; |
| | | |
| | | .u-upload { |
| | | @include flex(column); |
| | | flex: 1; |
| | | |
| | | &__wrap { |
| | | @include flex; |
| | | flex-wrap: wrap; |
| | | flex: 1; |
| | | |
| | | &__preview { |
| | | border-radius: $u-upload-preview-border-radius; |
| | | margin: $u-upload-preview-margin; |
| | | position: relative; |
| | | overflow: hidden; |
| | | @include flex; |
| | | |
| | | &__image { |
| | | width: $u-upload-image-width; |
| | | height: $u-upload-image-height; |
| | | } |
| | | |
| | | &__other { |
| | | width: $u-upload-image-width; |
| | | height: $u-upload-image-height; |
| | | background-color: $u-upload-other-bgColor; |
| | | flex: $u-upload-other-flex; |
| | | @include flex(column); |
| | | justify-content: center; |
| | | align-items: center; |
| | | |
| | | &__text { |
| | | font-size: $u-upload-text-font-size; |
| | | color: $u-upload-text-color; |
| | | margin-top: $u-upload-text-margin-top; |
| | | } |
| | | } |
| | | } |
| | | } |
| | | |
| | | &__deletable { |
| | | position: absolute; |
| | | top: $u-upload-deletable-top; |
| | | right: $u-upload-deletable-right; |
| | | background-color: $u-upload-deletable-bgColor; |
| | | height: $u-upload-deletable-height; |
| | | width: $u-upload-deletable-width; |
| | | @include flex; |
| | | border-bottom-left-radius: $u-upload-deletable-boder-bottom-left-radius; |
| | | align-items: center; |
| | | justify-content: center; |
| | | z-index: $u-upload-deletable-zIndex; |
| | | |
| | | &__icon { |
| | | position: absolute; |
| | | transform: scale(0.7); |
| | | top: $u-upload-icon-top; |
| | | right: $u-upload-icon-right; |
| | | /* #ifdef H5 */ |
| | | top: $u-upload-icon-h5-top; |
| | | right: $u-upload-icon-h5-right; |
| | | /* #endif */ |
| | | } |
| | | } |
| | | |
| | | &__success { |
| | | position: absolute; |
| | | bottom: $u-upload-success-bottom; |
| | | right: $u-upload-success-right; |
| | | @include flex; |
| | | // ç±äºweex(nvue)为é¿éå·´å·´çKPI(é¨é¨ä¸ç»©èæ ¸)çlaji产ç©ï¼ä¸æ¯æcssç»å¶ä¸è§å½¢ |
| | | // æä»¥å¨nvueä¸ä½¿ç¨å¾çï¼énvueä¸ä½¿ç¨csså®ç° |
| | | /* #ifndef APP-NVUE */ |
| | | border-style: $u-upload-success-border-style; |
| | | border-top-color: $u-upload-success-border-top-color; |
| | | border-left-color: $u-upload-success-border-left-color; |
| | | border-bottom-color: $u-upload-success-border-bottom-color; |
| | | border-right-color: $u-upload-success-border-right-color; |
| | | border-width: $u-upload-success-border-width; |
| | | align-items: center; |
| | | justify-content: center; |
| | | /* #endif */ |
| | | |
| | | &__icon { |
| | | /* #ifndef APP-NVUE */ |
| | | position: absolute; |
| | | transform: scale(0.7); |
| | | bottom: $u-upload-success-icon-bottom; |
| | | right: $u-upload-success-icon-right; |
| | | /* #endif */ |
| | | /* #ifdef APP-NVUE */ |
| | | width: $u-upload-icon-width; |
| | | height: $u-upload-icon-height; |
| | | /* #endif */ |
| | | } |
| | | } |
| | | |
| | | &__status { |
| | | position: absolute; |
| | | top: $u-upload-status-top; |
| | | bottom: $u-upload-status-bottom; |
| | | left: $u-upload-status-left; |
| | | right: $u-upload-status-right; |
| | | background-color: $u-upload-status-bgColor; |
| | | @include flex(column); |
| | | align-items: center; |
| | | justify-content: center; |
| | | |
| | | &__icon { |
| | | position: relative; |
| | | z-index: $u-upload-status-icon-Zindex; |
| | | } |
| | | |
| | | &__message { |
| | | font-size: $u-upload-message-font-size; |
| | | color: $u-upload-message-color; |
| | | margin-top: $u-upload-message-margin-top; |
| | | } |
| | | } |
| | | |
| | | &__button { |
| | | @include flex(column); |
| | | align-items: center; |
| | | justify-content: center; |
| | | width: $u-upload-button-width; |
| | | height: $u-upload-button-height; |
| | | background-color: $u-upload-button-bgColor; |
| | | border-radius: $u-upload-button-border-radius; |
| | | margin: $u-upload-botton-margin; |
| | | /* #ifndef APP-NVUE */ |
| | | box-sizing: border-box; |
| | | /* #endif */ |
| | | |
| | | &__text { |
| | | font-size: $u-upload-text-font-size; |
| | | color: $u-upload-text-color; |
| | | margin-top: $u-upload-text-margin-top; |
| | | } |
| | | |
| | | &--hover { |
| | | background-color: $u-upload-hover-bgColor; |
| | | } |
| | | |
| | | &--disabled { |
| | | opacity: $u-upload-disabled-opacity; |
| | | } |
| | | } |
| | | } |
| | | </style> |