feat: Add base project structure with API and web components

This commit is contained in:
Ke Sun
2025-12-02 20:28:01 +08:00
parent f3de6d6cc9
commit c1adc62ec6
817 changed files with 111226 additions and 106 deletions

View File

@@ -0,0 +1,307 @@
import { useState, useEffect, forwardRef, useImperativeHandle } from 'react';
import { Upload, Button, Modal, Progress, App } from 'antd';
import { UploadOutlined } from '@ant-design/icons';
import type { UploadProps, UploadFile } from 'antd';
// import { request } from '@/utils/request';
import type { UploadProps as RcUploadProps } from 'antd/es/upload/interface';
import CloudUploadOutlined from '@/assets/images/CloudUploadOutlined.png'
import { useTranslation } from 'react-i18next';
const { confirm } = Modal;
const { Dragger } = Upload;
interface UploadFilesProps extends Omit<UploadProps, 'onChange'> {
/** 上传接口地址 */
action?: string;
/** 是否支持多选 */
multiple?: boolean;
/** 已上传的文件列表 */
fileList?: UploadFile[];
/** 文件列表变化回调 */
onChange?: (fileList: UploadFile[]) => void;
customRequest?: RcUploadProps['customRequest'];
/** 自定义上传请求配置 */
requestConfig?: {
data?: Record<string, string | number | boolean>;
headers?: Record<string, string>;
};
/** 禁用上传 */
disabled?: boolean;
/** 文件大小限制MB */
fileSize?: number;
/** 文件类型限制 ['doc', 'xls', 'ppt', 'pdf'] */
fileType?: string[];
/** 是否自动上传默认为true */
isAutoUpload?: boolean;
/** 最大上传文件数 */
maxCount?: number;
/** 是否支持拖拽上传默认为false */
isCanDrag?: boolean;
}
const ALL_FILE_TYPE: {
[key: string]: string;
} = {
txt: 'text/plain',
pdf: 'application/pdf',
doc: 'application/msword',
docx: 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
xls: 'application/vnd.ms-excel',
xlsx: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
ppt: 'application/vnd.ms-powerpoint',
pptx: 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
csv: 'text/csv',
md: 'text/markdown',
htm: 'text/html',
html: 'text/html',
json: 'application/json',
}
export interface UploadFilesRef {
fileList: UploadFile[];
clearFiles: () => void;
}
/**
* 公共上传组件基于Ant Design Upload组件封装
* 支持单文件/多文件上传、拖拽上传、文件验证、预览等功能
*/
const UploadFiles = forwardRef<UploadFilesRef, UploadFilesProps>(({
action = '/api/upload',
multiple = false,
fileList: propFileList = [],
onChange,
// requestConfig = {},
disabled = false,
fileSize = 5,
fileType = ['doc', 'xls', 'ppt', 'pdf'],
isAutoUpload = true,
maxCount = 1,
isCanDrag = false,
...props
}, ref) => {
const { t } = useTranslation();
const { message } = App.useApp()
const [fileList, setFileList] = useState<UploadFile[]>(propFileList);
const [accept, setAccept] = useState<string | undefined>();
// 处理文件移除
const handleRemove = (file: UploadFile) => {
confirm({
title: '确定要删除此文件吗?',
okText: '确定',
okType: 'danger',
cancelText: '取消',
onOk: () => {
const newFileList = fileList.filter((item) => item.uid !== file.uid);
setFileList(newFileList);
onChange?.(newFileList);
},
});
return false; // 阻止默认删除行为由confirm控制
};
// 校验文件类型和大小
const beforeUpload: RcUploadProps['beforeUpload'] = (file) => {
// 校验文件大小
if (fileSize) {
const isLtMaxSize = (file.size / 1024 / 1024) < fileSize;
if (!isLtMaxSize) {
message.error(`文件大小不能超过 ${fileSize}MB`);
return Upload.LIST_IGNORE;
}
}
// 校验文件类型
if (fileType && fileType.length > 0) {
// 获取文件扩展名
const fileName = file.name.toLowerCase();
const fileExtension = fileName.substring(fileName.lastIndexOf('.') + 1);
// 检查扩展名是否在允许的类型列表中
const isValidExtension = fileType.some(type => type.toLowerCase() === fileExtension);
// 如果有 MIME 类型,也检查 MIME 类型(作为备选验证)
const isValidMimeType = file.type && accept ? accept.includes(file.type) : true;
if (!isValidExtension && !isValidMimeType) {
message.error(`不支持的文件类型: ${fileExtension || file.type}`);
return Upload.LIST_IGNORE;
}
}
if (!isAutoUpload) {
const newFileList = [...fileList, file as UploadFile];
setFileList(newFileList);
onChange?.(newFileList);
return Upload.LIST_IGNORE; // 阻止自动上传
}
return isAutoUpload;
};
// 自定义上传方法
/*
const customRequest: RcUploadProps['customRequest'] = ({ file, onSuccess, onError, onProgress }) => {
setLoading(true);
const formData = new FormData();
formData.append('file', file as RcFile);
// 添加额外的请求参数
const requestData = requestConfig.data;
if (requestData) {
Object.keys(requestData).forEach(key => {
const value = requestData[key];
formData.append(key, String(value));
});
}
request.post(action, formData, {
headers: {
'Content-Type': 'multipart/form-data',
...requestConfig.headers,
},
...requestConfig,
})
.then((response) => {
if (onSuccess) onSuccess(response);
})
.catch((error) => {
message.error('上传失败,请重试');
if (onError) onError(error);
// setFileList(fileList.filter((item) => item.uid !== (file as UploadFile).uid));
})
.finally(() => {
setLoading(false);
});
};
*/
// 处理上传状态变化
const handleChange: UploadProps['onChange'] = ({ fileList: newFileList, event }) => {
console.log('event', event)
setFileList(newFileList);
if (onChange) {
onChange(newFileList);
}
};
// 清空已上传文件
const clearFiles = () => {
setFileList([]);
if (onChange) {
onChange([]);
}
}
useEffect(() => {
if (fileType && fileType.length > 0) {
// 同时包含 MIME 类型和文件扩展名
const acceptArray: string[] = [];
fileType.forEach((type: string) => {
const lowerType = type.toLowerCase();
// 添加 MIME 类型(如果存在)
const mimeType = ALL_FILE_TYPE[lowerType];
if (mimeType) {
acceptArray.push(mimeType);
}
// 添加文件扩展名(.md, .html 等)
acceptArray.push(`.${lowerType}`);
});
setAccept(acceptArray.join(','));
} else {
setAccept(undefined);
}
}, [fileType])
// 生成上传组件配置
const uploadProps: UploadProps = {
action,
multiple: multiple && maxCount > 1,
fileList,
beforeUpload,
headers: {
authorization: localStorage.getItem('token') || '',
},
onRemove: handleRemove,
onChange: handleChange,
accept,
disabled,
showUploadList: {
showPreviewIcon: false,
showRemoveIcon: true,
showDownloadIcon: false,
},
itemRender: (_, file, __, actions) => {
return (
<div key={file.uid} className="rb:relative rb:w-full rb:pt-[8px] rb:pl-[10px] rb:pr-[10px] rb-pb-[10px] rb:border-1 rb:border-[#EBEBEB] rb:rounded rb:p-2 rb:mt-2 rb:bg-white">
<div className="rb:text-[12px] rb:flex rb:items-center rb:justify-between rb:mb-[2px]">
{file.name}
<span className="rb:text-[#5B6167]" onClick={() => actions?.remove()}>Cancel</span>
</div>
<Progress percent={file.percent || 0} strokeColor={file.status === 'error' ? '#FF5D34' : '#155EEF'} size="small" showInfo={false} />
</div>
);
},
...props,
};
// 暴露给父组件的方法
useImperativeHandle(ref, () => ({
fileList,
clearFiles
}));
const hasProgress = fileList.some((item) => item.percent !== 100);
if (isCanDrag) {
return (
<div className="rb:mb-[24px] rb:w-full">
<Dragger {...uploadProps} style={{ height: '270px' }}>
<div className="rb:flex rb:justify-center rb:flex-col rb:items-center">
<img className="rb:w-[48px] rb:h-[48px]" src={CloudUploadOutlined} />
{!hasProgress && (!fileList || !fileList.length) &&
<>
<div className="rb:text-base rb:text-[14px] rb:font-medium rb:flex rb:items-center rb:mt-[8px] rb:leading-[20px]">
{t('common.dragUploadTip')}<span className="rb:ml-[4px] rb:text-[#155EEF]">{t('common.uploadClickTip')}</span>
</div>
{fileType && <div className="rb:text-[12px] rb:text-[#A8A9AA] rb:leading-[14px] rb:mt-[8px] rb:cursor-pointer">{t('common.supportedFileTypes', { types: fileType.join(',') })}</div>}
{(fileSize || fileType || maxCount > 1) && (
<div className='rb:text-xs rb:mt-2 rb:text-[#A8A9AA]'>
{t('common.uploadFileTipMax', { max: fileSize, maxCount: maxCount })}
</div>
)}
</>
}
{hasProgress && <div className="rb:text-base rb:text-[14px] rb:font-medium rb:flex rb:items-center rb:mt-[8px] rb:mb-[24px] rb:leading-[20px]">{t('common.uploading')}</div>}
</div>
</Dragger>
</div>
);
}
return (
<Upload
{...uploadProps}
>
<div>
<Button
type="default"
icon={<UploadOutlined />}
className="rb:w-full"
disabled={fileList.length >= maxCount}
>
</Button>
{(fileSize || fileType || maxCount > 1) && (
<div>
{fileSize && <> <b style={{color: '#f56c6c'}}>{ fileSize }MB</b></>}
{fileType && <> <b style={{color: '#f56c6c'}}>{ fileType.join('、') }</b></>}
{multiple && maxCount > 1 && <> <b style={{color: '#f56c6c'}}>{ maxCount } </b> </>}
</div>
)}
</div>
</Upload>
);
});
export default UploadFiles;

View File

@@ -0,0 +1,222 @@
import { useState, useEffect, forwardRef, useImperativeHandle } from 'react';
import { Upload, Modal, Image, App } from 'antd';
import type { GetProp, UploadFile, UploadProps } from 'antd';
// import { UploadOutlined, } from '@ant-design/icons';
import type { UploadProps as RcUploadProps } from 'antd/es/upload/interface';
import { useTranslation } from 'react-i18next';
import PlusIcon from '@/assets/images/plus.svg'
const { confirm } = Modal;
interface UploadImagesProps extends Omit<UploadProps, 'onChange'> {
/** 上传接口地址 */
action?: string;
/** 是否支持多选 */
multiple?: boolean;
/** 已上传的文件列表 */
fileList?: UploadFile[];
/** 文件列表变化回调 */
onChange?: (fileList: UploadFile[]) => void;
/** 禁用上传 */
disabled?: boolean;
/** 文件大小限制MB */
fileSize?: number;
/** 文件类型限制 */
fileType?: string[];
/** 是否自动上传默认为true */
isAutoUpload?: boolean;
/** 最大上传文件数 */
maxCount?: number;
}
const ALL_FILE_TYPE: {
[key: string]: string;
} = {
jpg: 'image/jpeg',
jpeg: 'image/jpeg',
png: 'image/png',
gif: 'image/gif',
bmp: 'image/bmp',
webp: 'image/webp',
svg: 'image/svg+xml',
}
interface UploadImagesRef {
fileList: UploadFile[];
clearFiles: () => void;
}
type FileType = Parameters<GetProp<UploadProps, 'beforeUpload'>>[0];
const getBase64 = (file: FileType): Promise<string> => {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.readAsDataURL(file);
reader.onload = () => resolve(reader.result as string);
reader.onerror = (error) => reject(error);
});
}
/**
* 公共上传组件基于Ant Design Upload组件封装
* 支持单文件/多文件上传、拖拽上传、文件验证、预览等功能
*/
const UploadImages = forwardRef<UploadImagesRef, UploadImagesProps>(({
action = '/api/upload',
multiple = false,
fileList: propFileList = [],
onChange,
disabled = false,
fileSize,
fileType = ['png', 'jpg', 'gif'],
isAutoUpload = true,
maxCount = 1,
...props
}, ref) => {
const { t } = useTranslation();
const { message } = App.useApp()
const [fileList, setFileList] = useState<UploadFile[]>(propFileList);
const [accept, setAccept] = useState<string | undefined>();
// const [loading, setLoading] = useState(false);
const [previewOpen, setPreviewOpen] = useState(false);
const [previewImage, setPreviewImage] = useState('');
// 处理文件移除
const handleRemove = (file: UploadFile) => {
confirm({
title: '确定要删除此文件吗?',
okText: '确定',
okType: 'danger',
cancelText: '取消',
onOk: () => {
const newFileList = fileList.filter((item) => item.uid !== file.uid);
setFileList(newFileList);
onChange?.(newFileList);
},
});
return false; // 阻止默认删除行为由confirm控制
};
// 校验文件类型和大小
const beforeUpload: RcUploadProps['beforeUpload'] = async (file: UploadFile) => {
// 校验文件大小
if (fileSize && file.size) {
const isLtMaxSize = (file.size / 1024 / 1024) < fileSize;
if (!isLtMaxSize) {
message.error(`文件大小不能超过 ${fileSize}MB`);
return Upload.LIST_IGNORE;
}
}
// 校验文件类型
if (accept && accept.length > 0 && file.type) {
const isAccept = accept.includes(file.type);
if (!isAccept) {
message.error(`不支持的文件类型: ${file.type}`);
return Upload.LIST_IGNORE;
}
}
if (!isAutoUpload) {
if (!file.url && !file.preview) {
file.url = await getBase64(file.originFileObj as FileType);
}
const newFileList = [...fileList, file];
setFileList(newFileList);
onChange?.(newFileList);
return Upload.LIST_IGNORE; // 阻止自动上传
}
return isAutoUpload;
};
// 处理上传状态变化
const handleChange: UploadProps['onChange'] = ({ fileList: newFileList }) => {
setFileList(newFileList);
if (onChange) {
onChange(newFileList);
}
};
// 清空已上传文件
const clearFiles = () => {
setFileList([]);
if (onChange) {
onChange([]);
}
}
const handlePreview = async (file: UploadFile) => {
if (!file.thumbUrl && !file.url && !file.preview) {
file.preview = await getBase64(file.originFileObj as FileType);
}
setPreviewImage(file.thumbUrl || file.url || (file.preview as string));
setPreviewOpen(true);
};
useEffect(() => {
if (fileType && fileType.length > 0) {
const acceptArray = fileType.map((type: string) => ALL_FILE_TYPE[type.toLowerCase()]).filter(Boolean);
setAccept(acceptArray.join(','));
} else {
setAccept(undefined);
}
}, [fileType])
// 生成上传组件配置
const uploadProps: UploadProps = {
action,
multiple: multiple && maxCount > 1,
fileList,
beforeUpload,
headers: {
authorization: localStorage.getItem('token') || '',
},
onPreview: handlePreview,
onRemove: handleRemove,
onChange: handleChange,
accept,
disabled,
listType: 'picture-card',
showUploadList: {
showPreviewIcon: true,
showRemoveIcon: true,
showDownloadIcon: false,
},
...props,
};
// 暴露给父组件的方法
useImperativeHandle(ref, () => ({
fileList,
clearFiles
}));
return (
<>
<Upload
{...uploadProps}
style={{
width: '136px',
height: '136px',
}}
>
{fileList.length < maxCount && (
<div className="rb:flex rb:flex-wrap rb:items-center rb:justify-center">
<img src={PlusIcon} className="rb:w-[32px] rb:h-[32px]" />
<div className="rb:mt-[12px] rb:text-[12px] rb:text-[#5B6167] rb:leading-[16px]">{t('common.clickUploadIcon')}</div>
</div>
)}
</Upload>
{previewImage && (
<Image
wrapperStyle={{ display: 'none' }}
preview={{
visible: previewOpen,
onVisibleChange: (visible) => setPreviewOpen(visible),
afterOpenChange: (visible) => !visible && setPreviewImage(''),
}}
src={previewImage}
/>
)}
</>
);
});
export default UploadImages;