refactor: upload types, close #5047

pull/5057/head
tangjinzhou 2021-12-17 15:32:32 +08:00
parent 06516ec60d
commit 9784b429f5
16 changed files with 92 additions and 212 deletions

View File

@ -214,7 +214,7 @@ export {
TypographyTitle, TypographyTitle,
} from './typography'; } from './typography';
export type { UploadProps } from './upload'; export type { UploadProps, UploadListProps, UploadChangeParam } from './upload';
export { default as Upload, UploadDragger } from './upload'; export { default as Upload, UploadDragger } from './upload';

View File

@ -1,12 +1,12 @@
import { defineComponent } from 'vue'; import { defineComponent } from 'vue';
import { getOptionProps, getSlot } from '../_util/props-util'; import { getOptionProps, getSlot } from '../_util/props-util';
import Upload from './Upload'; import Upload from './Upload';
import { UploadProps } from './interface'; import { uploadProps } from './interface';
export default defineComponent({ export default defineComponent({
name: 'AUploadDragger', name: 'AUploadDragger',
inheritAttrs: false, inheritAttrs: false,
props: UploadProps, props: uploadProps,
render() { render() {
const props = getOptionProps(this); const props = getOptionProps(this);
const { height, ...restProps } = props; const { height, ...restProps } = props;

View File

@ -10,39 +10,19 @@ import defaultLocale from '../locale-provider/default';
import { defaultConfigProvider } from '../config-provider'; import { defaultConfigProvider } from '../config-provider';
import Dragger from './Dragger'; import Dragger from './Dragger';
import UploadList from './UploadList'; import UploadList from './UploadList';
import { UploadProps } from './interface'; import type { UploadFile } from './interface';
import { uploadProps } from './interface';
import { T, fileToObject, genPercentAdd, getFileItem, removeFileItem } from './utils'; import { T, fileToObject, genPercentAdd, getFileItem, removeFileItem } from './utils';
import { defineComponent, inject } from 'vue'; import { defineComponent, inject } from 'vue';
import { getDataAndAriaProps } from '../_util/util'; import { getDataAndAriaProps } from '../_util/util';
import { useInjectFormItemContext } from '../form/FormItemContext'; import { useInjectFormItemContext } from '../form/FormItemContext';
export type UploadFileStatus = 'error' | 'success' | 'done' | 'uploading' | 'removed';
export interface UploadFile<T = any> {
uid: string;
size?: number;
name: string;
fileName?: string;
lastModified?: number;
lastModifiedDate?: Date;
url?: string;
status?: UploadFileStatus;
percent?: number;
thumbUrl?: string;
originFileObj?: any;
response?: T;
error?: any;
linkProps?: any;
type?: string;
xhr?: T;
preview?: string;
}
export default defineComponent({ export default defineComponent({
name: 'AUpload', name: 'AUpload',
mixins: [BaseMixin], mixins: [BaseMixin],
inheritAttrs: false, inheritAttrs: false,
Dragger, Dragger,
props: initDefaultProps(UploadProps, { props: initDefaultProps(uploadProps, {
type: 'select', type: 'select',
multiple: false, multiple: false,
action: '', action: '',

View File

@ -20,12 +20,12 @@ import EyeOutlined from '@ant-design/icons-vue/EyeOutlined';
import Tooltip from '../tooltip'; import Tooltip from '../tooltip';
import Progress from '../progress'; import Progress from '../progress';
import classNames from '../_util/classNames'; import classNames from '../_util/classNames';
import { UploadListProps } from './interface'; import { uploadListProps } from './interface';
export default defineComponent({ export default defineComponent({
name: 'AUploadList', name: 'AUploadList',
mixins: [BaseMixin], mixins: [BaseMixin],
props: initDefaultProps(UploadListProps, { props: initDefaultProps(uploadListProps, {
listType: 'text', // or picture listType: 'text', // or picture
progressAttr: { progressAttr: {
strokeWidth: 2, strokeWidth: 2,

View File

@ -2,10 +2,10 @@ import { mount } from '@vue/test-utils';
import Upload from '..'; import Upload from '..';
import { T, fileToObject, genPercentAdd, getFileItem, removeFileItem } from '../utils'; import { T, fileToObject, genPercentAdd, getFileItem, removeFileItem } from '../utils';
import PropsTypes from '../../_util/vue-types'; import PropsTypes from '../../_util/vue-types';
import { UploadListProps } from '../interface'; import { uploadListProps } from '../interface';
import { setup, teardown } from './mock'; import { setup, teardown } from './mock';
UploadListProps.items = PropsTypes.any; uploadListProps.items = PropsTypes.any;
describe('Upload', () => { describe('Upload', () => {
beforeEach(() => setup()); beforeEach(() => setup());

View File

@ -3,11 +3,11 @@ import * as Vue from 'vue';
import Upload from '..'; import Upload from '..';
import { errorRequest, successRequest } from './requests'; import { errorRequest, successRequest } from './requests';
import PropsTypes from '../../_util/vue-types'; import PropsTypes from '../../_util/vue-types';
import { UploadListProps } from '../interface'; import { uploadListProps } from '../interface';
import { sleep } from '../../../tests/utils'; import { sleep } from '../../../tests/utils';
import { h } from 'vue'; import { h } from 'vue';
UploadListProps.items = PropsTypes.any; uploadListProps.items = PropsTypes.any;
const delay = timeout => new Promise(resolve => setTimeout(resolve, timeout)); const delay = timeout => new Promise(resolve => setTimeout(resolve, timeout));
const fileList = [ const fileList = [

View File

@ -42,22 +42,7 @@ Click to upload user's avatar, and validate size and format of picture with `bef
import { PlusOutlined, LoadingOutlined } from '@ant-design/icons-vue'; import { PlusOutlined, LoadingOutlined } from '@ant-design/icons-vue';
import { message } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadChangeParam, UploadProps } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
url?: string;
type?: string;
size: number;
originFileObj: any;
}
interface FileInfo {
file: FileItem;
fileList: FileItem[];
}
function getBase64(img: Blob, callback: (base64Url: string) => void) { function getBase64(img: Blob, callback: (base64Url: string) => void) {
const reader = new FileReader(); const reader = new FileReader();
@ -74,7 +59,7 @@ export default defineComponent({
const loading = ref<boolean>(false); const loading = ref<boolean>(false);
const imageUrl = ref<string>(''); const imageUrl = ref<string>('');
const handleChange = (info: FileInfo) => { const handleChange = (info: UploadChangeParam) => {
if (info.file.status === 'uploading') { if (info.file.status === 'uploading') {
loading.value = true; loading.value = true;
return; return;
@ -92,7 +77,7 @@ export default defineComponent({
} }
}; };
const beforeUpload = (file: FileItem) => { const beforeUpload = (file: UploadProps['fileList'][number]) => {
const isJpgOrPng = file.type === 'image/jpeg' || file.type === 'image/png'; const isJpgOrPng = file.type === 'image/jpeg' || file.type === 'image/png';
if (!isJpgOrPng) { if (!isJpgOrPng) {
message.error('You can only upload JPG file!'); message.error('You can only upload JPG file!');

View File

@ -34,26 +34,14 @@ Classic mode. File selection dialog pops up when upload button is clicked.
import { message } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { UploadOutlined } from '@ant-design/icons-vue'; import { UploadOutlined } from '@ant-design/icons-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadChangeParam } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
url?: string;
}
interface FileInfo {
file: FileItem;
fileList: FileItem[];
}
export default defineComponent({ export default defineComponent({
components: { components: {
UploadOutlined, UploadOutlined,
}, },
setup() { setup() {
const handleChange = (info: FileInfo) => { const handleChange = (info: UploadChangeParam) => {
if (info.file.status !== 'uploading') { if (info.file.status !== 'uploading') {
console.log(info.file, info.fileList); console.log(info.file, info.fileList);
} }

View File

@ -26,25 +26,14 @@ Use `defaultFileList` for uploaded files when page init.
<script lang="ts"> <script lang="ts">
import { UploadOutlined } from '@ant-design/icons-vue'; import { UploadOutlined } from '@ant-design/icons-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadChangeParam, UploadProps } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
url?: string;
}
interface FileInfo {
file: FileItem;
fileList: FileItem[];
}
export default defineComponent({ export default defineComponent({
components: { components: {
UploadOutlined, UploadOutlined,
}, },
setup() { setup() {
const fileList = ref<FileItem[]>([ const fileList = ref<UploadProps['fileList']>([
{ {
uid: '1', uid: '1',
name: 'xxx.png', name: 'xxx.png',
@ -67,7 +56,7 @@ export default defineComponent({
}, },
]); ]);
const handleChange = ({ file, fileList }: FileInfo) => { const handleChange = ({ file, fileList }: UploadChangeParam) => {
if (file.status !== 'uploading') { if (file.status !== 'uploading') {
console.log(file, fileList); console.log(file, fileList);
} }

View File

@ -41,26 +41,14 @@ We can upload serveral files at once in modern browsers by giving the input the
import { InboxOutlined } from '@ant-design/icons-vue'; import { InboxOutlined } from '@ant-design/icons-vue';
import { message } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadChangeParam } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
url?: string;
}
interface FileInfo {
file: FileItem;
fileList: FileItem[];
}
export default defineComponent({ export default defineComponent({
components: { components: {
InboxOutlined, InboxOutlined,
}, },
setup() { setup() {
const handleChange = (info: FileInfo) => { const handleChange = (info: UploadChangeParam) => {
const status = info.file.status; const status = info.file.status;
if (status !== 'uploading') { if (status !== 'uploading') {
console.log(info.file, info.fileList); console.log(info.file, info.fileList);

View File

@ -39,26 +39,14 @@ You can gain full control over filelist by configuring `fileList`. You can accom
<script lang="ts"> <script lang="ts">
import { UploadOutlined } from '@ant-design/icons-vue'; import { UploadOutlined } from '@ant-design/icons-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadChangeParam, UploadProps } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: Response;
url: string;
}
interface FileInfo {
file: FileItem;
fileList: FileItem[];
}
export default defineComponent({ export default defineComponent({
components: { components: {
UploadOutlined, UploadOutlined,
}, },
setup() { setup() {
const fileList = ref<FileItem[]>([ const fileList = ref<UploadProps['fileList']>([
{ {
uid: '-1', uid: '-1',
name: 'xxx.png', name: 'xxx.png',
@ -66,7 +54,7 @@ export default defineComponent({
url: 'http://www.baidu.com/xxx.png', url: 'http://www.baidu.com/xxx.png',
}, },
]); ]);
const handleChange = (info: FileInfo) => { const handleChange = (info: UploadChangeParam) => {
let resFileList = [...info.fileList]; let resFileList = [...info.fileList];
// 1. Limit the number of uploaded files // 1. Limit the number of uploaded files

View File

@ -36,6 +36,7 @@ After users upload picture, the thumbnail will be shown in list. The upload butt
<script lang="ts"> <script lang="ts">
import { PlusOutlined } from '@ant-design/icons-vue'; import { PlusOutlined } from '@ant-design/icons-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadChangeParam, UploadProps } from 'ant-design-vue';
function getBase64(file: File) { function getBase64(file: File) {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
@ -46,22 +47,6 @@ function getBase64(file: File) {
}); });
} }
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
percent?: number;
url?: string;
preview?: string;
originFileObj?: any;
}
interface FileInfo {
file: FileItem;
fileList: FileItem[];
}
export default defineComponent({ export default defineComponent({
components: { components: {
PlusOutlined, PlusOutlined,
@ -70,7 +55,7 @@ export default defineComponent({
const previewVisible = ref<boolean>(false); const previewVisible = ref<boolean>(false);
const previewImage = ref<string | undefined>(''); const previewImage = ref<string | undefined>('');
const fileList = ref<FileItem[]>([ const fileList = ref<UploadProps['fileList']>([
{ {
uid: '-1', uid: '-1',
name: 'image.png', name: 'image.png',
@ -105,14 +90,14 @@ export default defineComponent({
const handleCancel = () => { const handleCancel = () => {
previewVisible.value = false; previewVisible.value = false;
}; };
const handlePreview = async (file: FileItem) => { const handlePreview = async (file: UploadProps['fileList'][number]) => {
if (!file.url && !file.preview) { if (!file.url && !file.preview) {
file.preview = (await getBase64(file.originFileObj)) as string; file.preview = (await getBase64(file.originFileObj)) as string;
} }
previewImage.value = file.url || file.preview; previewImage.value = file.url || file.preview;
previewVisible.value = true; previewVisible.value = true;
}; };
const handleChange = ({ fileList: newFileList }: FileInfo) => { const handleChange = ({ fileList: newFileList }: UploadChangeParam) => {
fileList.value = newFileList; fileList.value = newFileList;
}; };

View File

@ -46,22 +46,14 @@ If uploaded file is a picture, the thumbnail can be shown. `IE8/9` do not suppor
<script lang="ts"> <script lang="ts">
import { UploadOutlined } from '@ant-design/icons-vue'; import { UploadOutlined } from '@ant-design/icons-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadProps } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
url?: string;
thumbUrl?: string;
}
export default defineComponent({ export default defineComponent({
components: { components: {
UploadOutlined, UploadOutlined,
}, },
setup() { setup() {
const fileList = ref<FileItem[]>([ const fileList = ref<UploadProps['fileList']>([
{ {
uid: '-1', uid: '-1',
name: 'xxx.png', name: 'xxx.png',
@ -77,7 +69,7 @@ export default defineComponent({
thumbUrl: 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png', thumbUrl: 'https://zos.alipayobjects.com/rmsportal/jkjgkEfvpUPVyRjUImniVslZfWPnJuuZ.png',
}, },
]); ]);
const fileList1 = ref<FileItem[]>([ const fileList1 = ref<UploadProps['fileList']>([
{ {
uid: '-1', uid: '-1',
name: 'xxx.png', name: 'xxx.png',

View File

@ -39,41 +39,31 @@ import request from 'umi-request';
import { UploadOutlined } from '@ant-design/icons-vue'; import { UploadOutlined } from '@ant-design/icons-vue';
import { message } from 'ant-design-vue'; import { message } from 'ant-design-vue';
import { defineComponent, ref } from 'vue'; import { defineComponent, ref } from 'vue';
import type { UploadProps } from 'ant-design-vue';
interface FileItem {
uid: string;
name?: string;
status?: string;
response?: string;
url?: string;
preview?: string;
originFileObj?: any;
file: string | Blob;
}
export default defineComponent({ export default defineComponent({
components: { components: {
UploadOutlined, UploadOutlined,
}, },
setup() { setup() {
const fileList = ref<FileItem[]>([]); const fileList = ref<UploadProps['fileList']>([]);
const uploading = ref<boolean>(false); const uploading = ref<boolean>(false);
const handleRemove = (file: FileItem) => { const handleRemove: UploadProps['onRemove'] = file => {
const index = fileList.value.indexOf(file); const index = fileList.value.indexOf(file);
const newFileList = fileList.value.slice(); const newFileList = fileList.value.slice();
newFileList.splice(index, 1); newFileList.splice(index, 1);
fileList.value = newFileList; fileList.value = newFileList;
}; };
const beforeUpload = (file: FileItem) => { const beforeUpload: UploadProps['beforeUpload'] = file => {
fileList.value = [...fileList.value, file]; fileList.value = [...fileList.value, file];
return false; return false;
}; };
const handleUpload = () => { const handleUpload = () => {
const formData = new FormData(); const formData = new FormData();
fileList.value.forEach((file: FileItem) => { fileList.value.forEach((file: UploadProps['fileList'][number]) => {
formData.append('files[]', file as any); formData.append('files[]', file as any);
}); });
uploading.value = true; uploading.value = true;

View File

@ -2,7 +2,7 @@ import type { App, Plugin } from 'vue';
import Upload from './Upload'; import Upload from './Upload';
import Dragger from './Dragger'; import Dragger from './Dragger';
export { UploadProps, UploadListProps, UploadChangeParam } from './interface'; export type { UploadProps, UploadListProps, UploadChangeParam } from './interface';
Upload.Dragger = Dragger; Upload.Dragger = Dragger;

View File

@ -1,3 +1,4 @@
import type { ExtractPropTypes, PropType } from 'vue';
import { tuple } from '../_util/type'; import { tuple } from '../_util/type';
import PropsTypes from '../_util/vue-types'; import PropsTypes from '../_util/vue-types';
@ -15,58 +16,51 @@ export interface VcFile extends File {
readonly webkitRelativePath: string; readonly webkitRelativePath: string;
} }
// export const UploadFile = PropsTypes.shape({ export type UploadFileStatus = 'error' | 'success' | 'done' | 'uploading' | 'removed';
// uid: PropsTypes.oneOfType([ export interface UploadFile<T = any> {
// PropsTypes.string, uid: string;
// PropsTypes.number, size?: number;
// ]), name: string;
// size: PropsTypes.number, fileName?: string;
// name: PropsTypes.string, lastModified?: number;
// filename: PropsTypes.string, lastModifiedDate?: Date;
// lastModified: PropsTypes.number, url?: string;
// lastModifiedDate: PropsTypes.any, status?: UploadFileStatus;
// url: PropsTypes.string, percent?: number;
// status: UploadFileStatus, thumbUrl?: string;
// percent: PropsTypes.number, originFileObj?: any;
// thumbUrl: PropsTypes.string, response?: T;
// originFileObj: PropsTypes.any, error?: any;
// response: PropsTypes.any, linkProps?: any;
// error: PropsTypes.any, type?: string;
// linkProps: PropsTypes.any, xhr?: T;
// type: PropsTypes.string, preview?: string;
// }).loose
function UploadFile({ uid, name }) {
if (!uid && uid !== 0) return false;
if (!['string', 'number'].includes(typeof uid)) return false;
if (name === '' || typeof name !== 'string') return false;
return true;
} }
export const UploadChangeParam = { export interface UploadChangeParam<T extends object = UploadFile> {
file: PropsTypes.custom(UploadFile), file: T;
fileList: PropsTypes.arrayOf(PropsTypes.custom(UploadFile)), fileList: UploadFile[];
event: PropsTypes.object, event?: { percent: number };
}; }
export const ShowUploadListInterface = PropsTypes.shape({ export const ShowUploadListInterface = PropsTypes.shape({
showRemoveIcon: PropsTypes.looseBool, showRemoveIcon: PropsTypes.looseBool,
showPreviewIcon: PropsTypes.looseBool, showPreviewIcon: PropsTypes.looseBool,
}).loose; }).loose;
export const UploadLocale = PropsTypes.shape({ export interface UploadLocale {
uploading: PropsTypes.string, uploading?: string;
removeFile: PropsTypes.string, removeFile?: string;
downloadFile: PropsTypes.string, downloadFile?: string;
uploadError: PropsTypes.string, uploadError?: string;
previewFile: PropsTypes.string, previewFile?: string;
}).loose; }
export const UploadProps = { export const uploadProps = {
type: PropsTypes.oneOf(tuple('drag', 'select')), type: PropsTypes.oneOf(tuple('drag', 'select')),
name: PropsTypes.string, name: PropsTypes.string,
defaultFileList: PropsTypes.arrayOf(PropsTypes.custom(UploadFile)), defaultFileList: { type: Array as PropType<UploadFile[]> },
fileList: PropsTypes.arrayOf(PropsTypes.custom(UploadFile)), fileList: { type: Array as PropType<UploadFile[]> },
action: PropsTypes.oneOfType([PropsTypes.string, PropsTypes.func]), action: PropsTypes.oneOfType([PropsTypes.string, PropsTypes.func]),
directory: PropsTypes.looseBool, directory: PropsTypes.looseBool,
data: PropsTypes.oneOfType([PropsTypes.object, PropsTypes.func]), data: PropsTypes.oneOfType([PropsTypes.object, PropsTypes.func]),
@ -86,36 +80,37 @@ export const UploadProps = {
customRequest: PropsTypes.func, customRequest: PropsTypes.func,
withCredentials: PropsTypes.looseBool, withCredentials: PropsTypes.looseBool,
openFileDialogOnClick: PropsTypes.looseBool, openFileDialogOnClick: PropsTypes.looseBool,
locale: UploadLocale, locale: { type: Object as PropType<UploadLocale> },
height: PropsTypes.number, height: PropsTypes.number,
id: PropsTypes.string, id: PropsTypes.string,
previewFile: PropsTypes.func, previewFile: PropsTypes.func,
transformFile: PropsTypes.func, transformFile: PropsTypes.func,
onChange: PropsTypes.func, onChange: { type: Function as PropType<(info: UploadChangeParam) => void> },
onPreview: PropsTypes.func, onPreview: { type: Function as PropType<(file: UploadFile) => void> },
onRemove: PropsTypes.func, onRemove: {
onDownload: PropsTypes.func, type: Function as PropType<(file: UploadFile) => void | boolean | Promise<void | boolean>>,
'onUpdate:fileList': PropsTypes.func, },
onDownload: { type: Function as PropType<(file: UploadFile) => void> },
'onUpdate:fileList': { type: Function as PropType<(files: UploadFile[]) => void> },
}; };
export const UploadState = { export type UploadProps = Partial<ExtractPropTypes<typeof uploadProps>>;
fileList: PropsTypes.arrayOf(PropsTypes.custom(UploadFile)), export const uploadListProps = {
dragState: PropsTypes.string,
};
export const UploadListProps = {
listType: PropsTypes.oneOf(tuple('text', 'picture', 'picture-card')), listType: PropsTypes.oneOf(tuple('text', 'picture', 'picture-card')),
// items: PropsTypes.arrayOf(UploadFile), // items: PropsTypes.arrayOf(UploadFile),
items: PropsTypes.arrayOf(PropsTypes.custom(UploadFile)), items: { type: Array as PropType<UploadFile[]> },
// items: PropsTypes.any,
progressAttr: PropsTypes.object, progressAttr: PropsTypes.object,
prefixCls: PropsTypes.string, prefixCls: PropsTypes.string,
showRemoveIcon: PropsTypes.looseBool, showRemoveIcon: PropsTypes.looseBool,
showDownloadIcon: PropsTypes.looseBool, showDownloadIcon: PropsTypes.looseBool,
showPreviewIcon: PropsTypes.looseBool, showPreviewIcon: PropsTypes.looseBool,
locale: UploadLocale, locale: { type: Object as PropType<UploadLocale> },
previewFile: PropsTypes.func, previewFile: PropsTypes.func,
onPreview: PropsTypes.func, onPreview: { type: Function as PropType<(file: UploadFile) => void> },
onRemove: PropsTypes.func, onRemove: {
onDownload: PropsTypes.func, type: Function as PropType<(file: UploadFile) => void | boolean>,
},
onDownload: { type: Function as PropType<(file: UploadFile) => void> },
}; };
export type UploadListProps = Partial<ExtractPropTypes<typeof uploadListProps>>;