代码演示
如何引入
import { Upload } from '@kousum/semi-ui-vue';基本
最基本的用法,在 children 内放置一个 Button,点击 children 内容(即放置的 Button)激活文件选择框,选择完成后自动开始上传
添加提示文本
通过 prompt 插槽,设置自定义提示文本
通过 promptPosition 设置插槽位置,可选 left、right、bottom,默认为 right
当 listType 为 picture 时,promptPosition 位置的参照对象为图片墙列表整体
点击头像触发上传
.avatar-upload .semi-upload-add {
border-radius: 50%; // 确保只有圆是点击热区
}自定义上传属性
通过设置 data、headers 可添加自定义上传属性
上传文件类型
通过 accept 属性(input的原生html` 属性)可以限制上传的文件类型。
accept 支持传入以下两种类型字符串:
- 文件后缀名集合(推荐),如 .jpg、.png 等;
- 文件类型的 MIME types 集合,可参考MDN 文档
例如只允许用户上传 PNG 和 PDF 文件,accept 可以这样写: accept = '.pdf,.png' 或 accept = 'application/pdf,image/png'(将 PNG 与 PDF 的 MIME type 通过,连接起来即可)。
上传文件夹
通过传入 directory 为 true,可以支持上传文件夹下的所有文件
一次选中多个文件
通过设置 multiple 属性可以支持同时选中多个文件上传。
限制文件总数量
通过设置 limit 属性可以限制最大可上传的文件数
当 limit 为1时,始终用最新上传的代替当前,并不会触发onExceed回调v1.5.0生效
import { Upload, Button } from '@kousum/semi-ui-vue';
import { IconUpload } from '@kousum/semi-icons-vue';
export default () => {
let action = 'https://api.semi.design/upload';
let limit = 1;
let onChange = props => {
console.log(props.fileList);
};
return (
<Upload
action={action}
limit={limit}
onChange={onChange}
>
<Button icon={<IconUpload />} theme="light">
点击上传(最多{limit}项)
</Button>
</Upload>
);
};
import { Upload, Button } from '@kousum/semi-ui-vue';
import { IconUpload } from '@kousum/semi-icons-vue';
export default () => {
let action = 'https://api.semi.design/upload';
let [disabled, setDisabled] = useState(false);
let limit = 2;
let onChange = props => {
let length = props.fileList.length;
if (length === limit) {
setDisabled(true);
} else {
setDisabled(false);
}
};
return (
<Upload
action={action}
limit={limit}
onExceed={() => Toast.warning(`最多只允许上传${limit}个文件`)}
onChange={onChange}
>
<Button icon={<IconUpload />} theme="light" disabled={disabled}>
点击上传(最多{limit}项)
</Button>
</Upload>
);
};照片墙模式下,当已上传文件数量等于 limit 时,会自动隐藏上传入口
限制上传文件大小
通过 maxSize 和 minSize 属性可以自定义上传文件大小的限制,通过设置 onSizeError 可以设置超出限制时的回调。
自定义预览逻辑
listType 为 list 时,可以通过传入 previewFile 览逻辑。
例如你不需要对图片类型进行缩略图预览时,可以在 previewFile 中恒定返回一个<IconFile />
自定义列表操作区
listType 为 list 时,可以通过传入 renderFileOperation 来自定义列表操作区
默认文件列表
通过 defaultFileList 可以展示已上传的文件。当需要预览默认文件的缩略图时,你可以将 defaultFileList 内对应 item 的 preview 属性设为 true
受控组件
当传入fileList时,作为受控组件使用。需要监听 onChange 回调,并且将 fileList 回传给 Upload(注意需传入一个新的数组对象)
图片墙
设置 listType = 'picture',用户可以上传图片并在列表中显示缩略图
如果通过 defaultFileList 或 fileList 设置已上传的文件列表时,会自动读取对象数组中的 url 属性用于展示图片
设置 showPicInfo,可以查看图片基础信息
图片墙放大预览
配合 Image 组件,通过 renderThumbnail API ,可以实现点击图片放大预览
可以通过 renderPicPreviewIcon,onPreviewClick 来自定义预览图标,当显示替换图标 showReplace 时,不会再显示预览图标
当需要自定义预览/替换功能时,需要关闭替换功能,使用 renderPicPreviewIcon 监听图标点击事件即可。onPreviewClick 监听的是单张图片容器的点击事件
图片墙设置宽高
通过设置 picHeight, picWidth ( v2.42 后提供),可以统一设置图片墙元素的宽高 如果同时使用 renderThumbnail return Image 组件来实现点击放大预览,你需要同时指定 Image 组件的 width 和 height
设置 hotSpotLocation 自定义点击热区的顺序,默认在照片墙列表结尾
禁用
手动触发上传
uploadTrigger='custom',选中文件后将不会自动触发上传。需要手动调用 ref 上的 upload 方法触发
拖拽上传
draggable='true',可以使用拖拽功能
可以通过 dragIcon、dragMainText、dragSubText 快捷设置拖拽区内容
还可以通过 children 传入 ReactNode,完全自定义拖拽区的显示
Scss 样式如下
.components-upload-demo-drag-area {
border-radius: var(--semi-border-radius-small);
border: 2px dashed var(--semi-color-border);
width: 100%;
padding: 12px;
background-color: var(--semi-color-tertiary-light-default);
display: flex;
cursor: pointer;
flex-wrap: wrap;
justify-content: center;
&:hover {
background-color: var(--semi-color-primary-light-default);
border-color: var(--semi-color-primary);
}
}上传前自定义校验
可通过 beforeUpload 钩子,对文件状态进行更新,这是在网络上传前,选择文件后进行校验,({ file: FileItem, fileList: Array<FileItem> }) => beforeUploadResult | Promise | boolean 同步校验时需返回 boolean(true 为校验通过,false 为校验失败,校验失败会阻止文件网络上传)或者一个 Object 对象,具体结构如下
// beforeUploadResult:
{
fileInstance?: File,
status?: 'success' | 'uploadFail' | 'validateFail' | 'validating' | 'uploading' | 'wait',
validateMessage?: React.ReactNode | string, // 文件的校验信息
shouldUpload: boolean, // 是否需要上传。默认为true,如果为false,该fileItem只会被展示在列表中,不会触发上传操作
autoRemove: boolean, // 是否从fileList中移除该文件,默认为false
}异步校验时,需返回 Promise,Promise resolve 代表检验通过,reject 代表校验失败,不会触发上传。
resolve/reject 时可以传入 object(结构同上 beforeUploadResult)
上传后更新文件信息
可以通过 afterUpload 钩子,对文件状态,校验信息,文件名进行更新。({ response: any, file: FileItem, fileList: Array<FileItem> }) => afterUploadResultafterUpload 在上传完成后(xhr.onload)且没有发生错误的情况下触发,需返回一个 Object 对象(不支持异步返回),具体结构如下
// afterUploadResult:
{
status?: 'success' | 'uploadFail' | 'validateFail' | 'validating' | 'uploading' | 'wait';
// 文件的校验信息
validateMessage?: React.ReactNode | string;
// 是否从fileList中移除该文件,默认为false
autoRemove?: boolean;
// 文件的名称
name?: string;
// 预览文件的url,一般为当次上传请求中 Server 接收到文件后返回的存储地址,v2.63后支持传入。
// 之前的版本也可以通过 onChange 中结合 status 手动更新受控 fileList 中的属性实现
url?: string
}自定义请求
当传入 customRequest 时, 相当于使用的自定义的请求方法替换了 upload 内置的 xhr 请求,用户需要自行接管上传行为。
可在入参中获取到当前操作的 file 对象,用户自行实现上传过程,并且在适当的时候调用 customRequest 入参中的 onProgress、onError、onSuccess 以更新 Upload 组件内部状态进而驱动 UI 更新
customRequest 包含以下入参
{
// 当前文件名称
fileName: string,
// 用户设置的props.data
data: object,
// FileItem,具体结构参考下面的文档
file: FileItem,
// original File Object which extends Blob, 浏览器实际获取到的文件对象(https://developer.mozilla.org/zh-CN/docs/Web/API/File)
fileInstance: File,
// 上传过程中应调用的函数,event需要包含 total、loaded属性
onProgress: (event: { total: number, loaded: number }) => any,
// 上传出错时应调用的函数
onError: (userXhr: { status: number }, e: event) => any,
// 上传成功后应调用的函数, response为上传成功后的请求结果
onSuccess: (response: any, e?: event) => any,
// 用户设置的props.withCredentials
withCredentials: boolean,
// 用户设置的props.action
action: string,
}API 参考
| 属性 | 说明 | 类型 | 默认值 | 版本 |
|---|---|---|---|---|
| accept | html 原生属性,接受上传的文件类型。accept 的值为你允许选择文件的MIME types 字符串或文件后缀(.jpg等) | string | ||
| action | 文件上传地址,必填 | string | ||
| addOnPasting | 按下 ctrl/command + v时,是否自动将剪贴板中的文件添加至 fileList,当前仅支持图片类型; 需用户授权同意 | boolean | false | 2.43.0 |
| afterUpload | 文件上传后的钩子,根据 return 的 object 更新文件状态 | function(auProps) => afterUploadResult | 1.0.0 | |
| beforeClear | 清空文件前回调,按照返回值来判断是否继续移除,返回false、Promise.resolve(false)、Promise.reject()会阻止移除 | (fileList: Array<FileItem >) => boolean|Promise | 1.31.0 | |
| beforeRemove | 移除文件前的回调,按照返回值来判断是否继续移除,返回false、Promise.resolve(false)、Promise.reject()会阻止移除 | (file: <FileItem>, fileList: Array<FileItem >) => boolean|Promise | 1.31.0 | |
| beforeUpload | 上传文件前的钩子,根据 return 的 object 更新文件状态,控制是否上传 | function(buProps) => beforeUploadResult | Promise | boolean | 1.0.0 | |
| capture | 文件上传控件中媒体拍摄的方式 | boolean|string|undefined | ||
| className | 类名 | string | ||
| customRequest | 自定义上传使用的异步请求方法 | (object: customRequestArgs) => void | 1.5.0 | |
| data | 上传时附带的额外参数或返回上传额外参数的方法 | object|(file: File) => object | {} | |
| defaultFileList | 已上传的文件列表 | Array<FileItem> | [] | |
| directory | 文件夹类型上传 | boolean | false | 1.20.0 |
| disabled | 是否禁用 | boolean | false | |
| dragIcon | 拖拽区左侧 Icon | ReactNode | <IconUpload /> | |
| dragMainText | 拖拽区主文本 | ReactNode | '点击上传文件或拖拽文件到这里' | |
| dragSubText | 拖拽区帮助文本 | ReactNode | '' | |
| draggable | 是否支持拖拽上传 | boolean | false | |
| fileList | 已上传的文件列表,传入该值时,upload 即为受控组件 | Array<FileItem> | 1.0.0 | |
| fileName | 作用与 name 相同,主要在 Form.Upload 中使用,为了避免与 Field 的 props.name 冲突,此处另外提供一个重命名的 props | string | 1.0.0 | |
| headers | 上传时附带的 headers 或返回上传额外 headers 的方法 | object|(file: File) => object | {} | |
| hotSpotLocation | 照片墙点击热区的放置位置,可选值 start, end | string | 'end' | 2.5.0 |
| itemStyle | fileCard 的内联样式 | CSSProperties | 1.0.0 | |
| limit | 最大允许上传文件个数 | number | ||
| listType | 文件列表展示类型,可选picture、list | string | 'list' | |
| maxSize | 文件体积最大限制,单位 KB | number | ||
| minSize | 文件体积最小限制,单位 KB | number | ||
| multiple | 是否允许单次选中多个文件 | boolean | false | |
| name | 上传时使用的文件名 | string | '' | |
| onAcceptInvalid | 当接收到的文件不符合accept规范时触发(一般是因为文件夹选择了全部类型文件/拖拽不符合格式的文件时触发) | (files: File[]) => void | 1.24.0 | |
| onChange | 文件状态发生变化时调用,包括上传成功,失败,上传中,回调入参为 Object,包含 fileList、currentFile 等值 | ({fileList: Array<FileItem\>, currentFile?: FileItem}) => void | 1.0.0 | |
| onClear | 点击清空时的回调 | () => void | 1.1.0 | |
| onDrop | 当拖拽的元素在拖拽区上被释放时触发 | (e, files: Array<File>, fileList: Array<FileItem>) => void | 1.9.0 | |
| onError | 上传错误时的回调 | (error: Error, file: File, fileList: Array<FileItem>, xhr: XMLHttpRequest) => void | ||
| onExceed | 上传文件总数超出 limit 时的回调 | (fileList:Array<FileItem>) => void | ||
| onFileChange | 选中文件后的回调 | (Array<File>) => void | ||
| onOpenFileDialog | 打开系统文系统文件选择弹窗时触发 | () => void | 1.18.0 | |
| onPreviewClick | 点击文件卡片时的回调 | (fileItem: FileItem) => void | 1.8.0 | |
| onProgress | 上传文件时的回调 | (percent: number, file: File, fileList: Array<FileItem>) => void | ||
| onPastingError | addOnPasting为true时,粘贴读取失败时的回调 | (Error|PermissionState) | 2.43.0 | |
| onRemove | 移除文件的回调 | (currentFile: File, fileList:Array<FileItem>, currentFileItem: FileItem) => void | ||
| onRetry | 上传重试的回调 | (file: <FileItem>) => void | 1.18.0 | |
| onSizeError | 文件尺寸非法的回调 | (file:File, fileList:Array<FileItem>) => void | ||
| onSuccess | 上传成功后的回调 | (responseBody: object, file: File, fileList:Array<FileItem>) => void | ||
| picHeight | 图片墙模式下,可通过该 API 定制图片展示高度 | string|number | 2.42.0 | |
| picWidth | 图片墙模式下,可通过该 API 定制图片展示宽度 | string|number | 2.42.0 | |
| previewFile | 自定义预览逻辑,该函数返回内容将会替换原缩略图 | (fileItem: FileItem) => ReactNode | ||
| prompt | 自定义插槽,可用于插入提示文本。与直接在 children 中写的区别时,prompt 的内容在点击时不会触发上传(图片墙模式下,v1.3.0 后才支持传入 prompt) | ReactNode | ||
| promptPosition | 提示文本的位置,当 listType 为 list 时,参照物为 children 元素;当 listType 为 picture 时,参照物为图片列表。可选值 left、right、bottom(图片墙模式下,v1.3.0 后才支持使用 promptPosition) | string | 'right' | |
| renderFileItem | fileCard 的自定义渲染 | (renderProps: RenderFileItemProps) => ReactNode | 1.0.0 | |
| renderFileOperation | 自定义列表项操作区 | (renderProps: RenderFileItemProps)=>ReactNode | 2.5.0 | |
| renderPicInfo | 自定义照片墙信息,只在照片墙模式下有效 | (renderProps: RenderFileItemProps)=>ReactNode | 2.2.0 | |
| renderPicPreviewIcon | 自定义照片墙hover时展示的预览图标,只在照片墙模式下有效 | (renderProps: RenderFileItemProps)=>ReactNode | 2.5.0 | |
| renderThumbnail | 自定义图片墙缩略图,只在照片墙模式下有效 | (renderProps: RenderFileItemProps)=>ReactNode | 2.2.0 | |
| showClear | 在 limit 不为 1 且当前已上传文件数大于 1 时,是否展示清空按钮 | boolean | true | 1.0.0 |
| showPicInfo | 是否显示图片信息,只在照片墙模式下有效 | boolean | false | 2.2.0 |
| showReplace | 上传成功时,是否展示在 fileCard 内部展示替换按钮 | boolean | false | 1.21.0 |
| showRetry | 上传失败时,是否展示在 fileCard 内部展示重试按钮 | boolean | true | 1.0.0 |
| showUploadList | 是否显示文件列表 | boolean | true | |
| style | 样式 | CSSProperties | ||
| transformFile | 选中文件后,上传文件前的回调函数,可用于对文件进行自定义转换处理 | (file:File) => FileItem | 1.0.0 | |
| uploadTrigger | 触发上传时机,可选值 auto、custom | string | 'auto' | |
| validateMessage | Upload 整体的错误信息 | ReactNode | 1.0.0 | |
| withCredentials | 是否带上 Cookie 信息 | boolean | false |
Interfaces
FileItem Interface
interface FileItem {
event? : event, // xhr event
fileInstance?: File, // original File Object which extends Blob, 浏览器实际获取到的文件对象(https://developer.mozilla.org/zh-CN/docs/Web/API/File)
name: string,
percent? : number, // 上传进度百分比
preview: boolean, // 是否根据url进行预览
response?: any, // xhr的response, 请求成功时为respoonse body,请求失败时为对应 error
shouldUpload?: boolean; // 是否应该继续上传
showReplace?: boolean, // 单独控制该file是否展示替换按钮
showRetry?: boolean, // 单独控制该file是否展示重试按钮
size: string, // 文件大小,单位kb
status: string, // 'success' | 'uploadFail' | 'validateFail' | 'validating' | 'uploading' | 'wait';
uid: string, // 文件唯一标识符,如果当前文件是通过upload选中添加的,会自动生成uid。如果是defaultFileList, 需要自行保证不会重复
url: string,
validateMessage?: ReactNode | string,
}Methods
绑定在组件实例上的方法,可以通过 ref 调用实现某些特殊交互
| 名称 | 描述 | 类型 | 版本 |
|---|---|---|---|
| insert | 上传文件,当index传入时,会插入到指定位置,不传则插入到最后 | (files: Array<File>, index?: number) => void | 2.2.0 |
| upload | 手动开始上传,配合uploadTrigger="custom"使用 | () => void | |
| openFileDialog | 打开文件选择窗口 | () => void | 2.21.0 |
Accessibility
Upload组件是一个可交互的控件,在点击或拖拽时触发文件选择,文件选中后会在文件列表内展示状态。
ARIA
- 为可点击元素添加
role="button" - 文件列表添加
role="list",并用aria-label描述
文案规范
- 上传按钮
- 关于表单按钮的文案规范,参考按钮Button组件的文案规范
- 帮助文本
- 帮助文本使用语句书写规范,首字母大写,可以不需要句号
- 用户出错提示
- 清晰地告诉用户为什么文件无法被上传,并且告知用户如何操作能够成功上传
- 帮助文本使用语句书写规范,首字母大写
- 简洁的用语让用户能够一眼读懂,比如
File size must be less than 20MB,File type must be .gif, .jpg, .png or .svg
设计变量
FAQ
- 什么时候会展示重试按钮?
- 当
showRetry为 true,且当前文件是由于网络原因错误导致的上传失败时,会展示重试按钮。其他如校验失败,上传成功等状态是不会展示重试按钮的。
- 当
- 什么时候会展示替换按钮?
- 当
showReplace为true,且当前文件状态为已上传时,会展示替换按钮。
- 当
- Semi Upload把图片存到哪里了?
- Semi Upload不负责图片的保存,当你使用 Upload 组件时需要自定义 action。你可以选择把 action 设置为自己的服务器地址或者图片服务地址。
- Form.Upload props.name无效?
- Form.Field 中有 props.name,Upload也有 props.name,同名 props 会冲突。使用 Form.Upload 时,可以转为使用 props.fileName,避免冲突
- 上传图片后没有调用 XXX 方法?
- 如果你设置了
accept,可以尝试把 accept 属性去掉,然后再看是否调用了改方法。去掉后调用了该方法说明,accept 在当前环境下获取的 file type 与设置的 accept 不符,上传行为提前终止。可以打个断点到 upload/foundation.js checkFileFormat 函数,看下获取的 file.type 真实值是否符合预期。
- 如果你设置了