| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- <template>
- <view :class="classes">
- <view class="preview" v-for="item in fileList" :key="item.uid">
- <view class="preview-img">
- <nut-icon
- v-if="isDeletable"
- color="rgba(0,0,0,0.6)"
- @click="onDelete(item, index)"
- class="close"
- name="mask-close"
- ></nut-icon>
- <img v-if="item.type.includes('image') && item.url" :src="item.url" />
- <view class="tips" v-if="item.status != 'success'">{{
- item.status
- }}</view>
- </view>
- </view>
- <view class="upload" v-if="maxCount - fileList.length">
- <nut-icon color="#808080" :name="uploadIcon"></nut-icon>
- <input
- type="file"
- :capture="capture"
- :accept="acceptType"
- :multiple="multiple"
- :name="name"
- :disabled="disabled"
- @change="onChange"
- />
- </view>
- </view>
- </template>
- <script lang="ts">
- import { computed, reactive } from 'vue';
- import { createComponent } from '@/utils/create';
- import { Uploader, UploadOptions } from './uploader';
- const { componentName, create } = createComponent('uploader');
- export type FileItemStatus =
- | 'ready'
- | 'uploading'
- | 'success'
- | 'error'
- | 'removed';
- export class FileItem {
- status: FileItemStatus = 'ready';
- uid: string = new Date().getTime().toString();
- name?: string;
- url?: string;
- type?: string;
- formData: FormData = new FormData();
- }
- export default create({
- props: {
- name: { type: String, default: 'file' },
- url: { type: String, default: '' },
- // defaultFileList: { type: Array, default: () => new Array<FileItem>() },
- fileList: { type: Array, default: () => [] },
- isPreview: { type: Boolean, default: true },
- isDeletable: { type: Boolean, default: true },
- method: { type: String, default: 'post' },
- capture: { type: String, default: 'camera' },
- maxSize: { type: [Number, String], default: Number.MAX_VALUE },
- maxCount: { type: [Number, String], default: 1 },
- clearInput: { type: Boolean, default: false },
- acceptType: { type: String, default: '*' },
- headers: { type: Object, default: {} },
- formData: { type: Object, default: {} },
- uploadIcon: { type: String, default: 'photograph' },
- xhrState: { type: [Number, String], default: 200 },
- withCredentials: { type: Boolean, default: false },
- multiple: { type: Boolean, default: false },
- disabled: { type: Boolean, default: false },
- beforeUpload: {
- type: Function,
- default: null
- },
- beforeDelete: {
- type: Function,
- default: (file: FileItem, files: FileItem[]) => {
- return true;
- }
- },
- onChange: { type: Function }
- // customRequest: { type: Function }
- },
- emits: [
- 'start',
- 'progress',
- 'oversize',
- 'success',
- 'failure',
- 'change',
- 'delete'
- ],
- setup(props, { emit }) {
- const fileList = reactive(props.fileList) as Array<FileItem>;
- const classes = computed(() => {
- const prefixCls = componentName;
- return {
- [prefixCls]: true
- };
- });
- const clearInput = (el: HTMLInputElement) => {
- el.value = '';
- };
- const executeUpload = (fileItem: FileItem) => {
- const uploadOption = new UploadOptions();
- uploadOption.url = props.url;
- for (const [key, value] of Object.entries(props.formData)) {
- fileItem.formData.append(key, value);
- }
- uploadOption.formData = fileItem.formData;
- uploadOption.method = props.method;
- uploadOption.xhrState = props.xhrState as number;
- uploadOption.headers = props.headers;
- uploadOption.withCredentials = props.withCredentials;
- uploadOption.onStart = (option: UploadOptions) => {
- fileItem.status = 'ready';
- emit('start', option);
- };
- uploadOption.onProgress = (
- e: ProgressEvent<XMLHttpRequestEventTarget>,
- option: UploadOptions
- ) => {
- fileItem.status = 'uploading';
- emit('progress', { e, option });
- };
- uploadOption.onSuccess = (
- responseText: XMLHttpRequest['responseText'],
- option: UploadOptions
- ) => {
- fileItem.status = 'success';
- emit('success', {
- responseText,
- option
- });
- };
- uploadOption.onFailure = (
- responseText: XMLHttpRequest['responseText'],
- option: UploadOptions
- ) => {
- fileItem.status = 'error';
- emit('failure', {
- responseText,
- option
- });
- };
- new Uploader(uploadOption).upload();
- };
- const readFile = (files: File[]) => {
- files.forEach((file: File) => {
- const formData = new FormData();
- formData.append(props.name, file);
- const fileItem = new FileItem();
- fileItem.name = file.name;
- fileItem.status = 'uploading';
- fileItem.type = file.type;
- fileItem.formData = formData;
- executeUpload(fileItem);
- if (props.isPreview && file.type.includes('image')) {
- const reader = new FileReader();
- reader.onload = (event: ProgressEvent<FileReader>) => {
- fileItem.url = (event.target as FileReader).result as string;
- fileList.push(fileItem);
- };
- reader.readAsDataURL(file);
- } else {
- fileList.push(fileItem);
- }
- });
- };
- const filterFiles = (files: File[]) => {
- const maxCount = (props.maxCount as number) * 1;
- const maxSize = (props.maxSize as number) * 1;
- const oversizes = new Array<File>();
- files = files.filter((file: File) => {
- if (file.size > maxSize) {
- oversizes.push(file);
- return false;
- } else {
- return true;
- }
- });
- if (oversizes.length) {
- emit('oversize', oversizes);
- }
- if (files.length > maxCount) {
- files.splice(maxCount - 1, files.length - maxCount);
- }
- return files;
- };
- const onDelete = (file: FileItem, index: number) => {
- if (props.beforeDelete(file, fileList)) {
- fileList.splice(index, 1);
- emit('delete', {
- file,
- fileList
- });
- } else {
- console.log('用户阻止了删除!');
- }
- };
- const onChange = (event: InputEvent) => {
- if (props.disabled) {
- return;
- }
- const $el = event.target as HTMLInputElement;
- let { files } = $el;
- if (props.clearInput) {
- clearInput($el);
- }
- if (props.beforeUpload) {
- props.beforeUpload(files).then((f: Array<File>) => {
- const _files: File[] = filterFiles(new Array<File>().slice.call(f));
- readFile(_files);
- });
- } else {
- const _files: File[] = filterFiles(new Array<File>().slice.call(files));
- readFile(_files);
- }
- emit('change', {
- fileList,
- event
- });
- };
- return {
- onChange,
- onDelete,
- fileList,
- classes
- };
- }
- });
- </script>
- <style lang="scss">
- @import 'index.scss';
- </style>
|