摘要:

最近一直在做sass系统,整个系统是基于阿里的Ant Design Pro框架来搭建的,里面经常有用到上传文件的一个功能,在开发时发现上传的逻辑都没有基于业务进行统一的封装,导致项目中一搜上传的出现一大堆一些重复的代码(可能开发时大家都觉得费事,所以直接负责粘贴了,虽然我有时也会这样😁),后面我就将一些常用的逻辑抽取出来,封装成一个通用的~

组件封装:

1、组件类型定义:

这部分主要就是定义这个上传组件需要的一些属性和值了,例如:上传的类型,文件列表、多选等,根据自己业务逻辑来就行了。

interface UploadTypes {
  acceptType: string[]; // 上传的文件类型,
  fileList: any[]; // 上传的文件列表
  setFileList: any;
  listType: "text" | "picture" | "picture-card"; // 上传卡片样式
  disabled?: boolean | false; // 是否禁用上传
  fileTag: number; // 上传文件类型标签
  anjianId: string | number;
  multiple?: boolean; // 是否允许文件多选,默认多选
}

const UploadCard: React.FC<UploadTypes> = ({ acceptType, fileList, setFileList, listType, disabled, anjianId = 0, fileTag = 4, multiple = true }) => {
  
  
  return (
  	  return (
        <Upload listType={listType} {...props} 
          fileList={fileList} 
          onPreview={handlePreview} 
          name="fashen_fujian" 					
          disabled={disabled}>
          {listType == "picture-card" && (
            <div>
              <CloudUploadOutlined style={{ fontSize: 30, color: "gray" }} />
            </div>
          )}
          {(listType == "picture" || listType == "text") &&
            <Button icon={<UploadOutlined />}>上传本地文件</Button>}
        </Upload>
  );
  )
}

2、文件类型检查和文件上传前检查:

文件类型检查:我这里列出了一些常见的文件上传类型,我这里默认允许pdf、jpg、jpeg、png类型文件的上传

// 常见的上传文件类型
const acceptTypeObj: AcceptType = {
  ".txt": "text/plain",
  ".html": "text/html",
  ".js": "text/javascript",
  ".css": "text/css",
  ".csv": "text/csv",
  ".json": "application/json",
  ".doc": "application/msword",
  ".docx": "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
  ".ppt": "application/vnd.ms-powerpoint",
  ".pptx": "application/vnd.openxmlformats-officedocument.presentationml.presentation",
  ".xls": "application/vnd.ms-excel",
  ".xlsx": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
  ".xml": "application/xml",
  ".pdf": "application/pdf",
  ".jpg": "image/jpeg",
  ".jpeg": "image/jpeg",
  ".png": "image/png",
  ".gif": "image/gif",
  ".bmp": "image/bmp",
  ".tif": "image/tiff",
  ".tiff": "image/tiff",
  ".ico": "image/x-icon",
  ".svg": "image/svg+xml",
  ".mp3": "audio/mpeg",
  ".wav": "audio/wav",
  ".mp4": "video/mp4",
  ".avi": "video/x-msvideo",
  ".wmv": "video/x-ms-wmv",
  ".mov": "video/quicktime",
  ".flv": "video/x-flv",
  ".webm": "video/webm",
  ".mpeg": "video/mpeg",
  ".zip": "application/zip",
  ".rar": "application/x-rar-compressed",
  ".tar": "application/x-tar",
  ".gz": "application/gzip",
  ".7z": "application/x-7z-compressed",
};

  // 文件上传
  const props: UploadProps = {
    name: "file",
    accept: _.map([...acceptType, ".pdf", ".jpg", ".jpeg", ".png"], (ext) => acceptTypeObj[ext])
      .filter(Boolean)
      .join(","), // 文件上传类型限制
    multiple: multiple,
    beforeUpload: (file) => {
      // 检查文件上传不能超过5g且文件不能重复
      const boolean = BeforeUpload(file, fileList);
      return boolean ? boolean : Upload.LIST_IGNORE;
    },
   
  };

beforeUpload方法: 上传文件前会执行这个方法,我们可以在这里对上传的文件做处理,我这里只限制了文件大小和不能重复;

/**
 * 上传前判断大小,是否相同
 *
 * @param file 要上传的文件
 * @param fileList 已上传的文件列表
 * @returns 能否上传
 */
export const BeforeUpload = (file: RcFile, fileList: any[]) => {
  const fileSizeInMB = file.size / 1024 / 1024; // 将文件大小转换为MB
  const maxSizeInMB = 5120; // 最大允许上传的文件大小(5GB)
  if (fileSizeInMB > maxSizeInMB) {
    message.error(`文件大小不能超过 ${maxSizeInMB}MB`);
    return false; // 返回 false 取消上传
  }
  const isFileExist = fileList.some(item =>
  (
    (item.name === file.name || item.name === file.name.slice(0, file.name.lastIndexOf('.'))) &&
    item.size === file.size &&
    item.type === file.type &&
    item.lastModified === file.lastModified)
  );
  if (isFileExist) {
    message.error('文件已存在,请选择其他文件');
    return false; // 阻止上传
  }
  return true; // 返回 true 允许上传
}

3、自定义上传逻辑:

使用customRequest方法来覆盖antd默认的上传行为,因为我这里上传文件时上传到腾讯云oss,需要先拿到上传的一个token验证,验证成功后再进行文件上传(其他oss应该都是大体一致的),这里我们主要处理的就是上传的onChange方法,因为这里如果上传的是图片类型的文件是没什么问题,主要的是上传一些pdf、word、excel等文件类型的是没有对应的缩列图显示,我在这里就根据对应的文件类型替换成相对应的缩列图显示了(缩列图地址要自己找的)

![image-20240429153320858](/Users/lance/Library/Application Support/typora-user-images/image-20240429153320858.png)

直接看方法实现:


// 文件图片样式地址,这里要自己找哈
const imgUrls = {
  ".doc": "https://xxx/images/applet/ls/wenjian/doc.png",
  ".docx": "https://xxx/images/applet/ls/wenjian/docx.png",
  ".csv": "https://xxx/images/applet/ls/wenjian/csv.png",
  ".html": "https://xxx/images/applet/ls/wenjian/html.png",
  ".jpg": "https://xxx/images/applet/ls/wenjian/jpg.png",
  ".jpeg": "https://xxx/images/applet/ls/wenjian/jpg.png",
  ".mp4": "https://xxx/images/applet/ls/wenjian/mp4.png",
  ".pdf": "https://xxx/images/applet/ls/wenjian/pdf.png",
  ".png": "https://xxx/images/applet/ls/wenjian/png.png",
  ".ppt": "https://xxx/images/applet/ls/wenjian/ppt.png",
  ".pptx": "https://xxx/images/applet/ls/wenjian/pptx.png",
  ".txt": "https://xxx/images/applet/ls/wenjian/txt.png",
  ".xls": "https://xxx/images/applet/ls/wenjian/xlsx.png",
  ".xlsx": "https://xxx/images/applet/ls/wenjian/xlsx.png",
  ".zip": "https://xxx/images/applet/ls/wenjian/zip.png",
  default: "https://xxx/images/applet/ls/wenjian/default.png",
};
// 文件类型匹配
const fileTypeMap = {
  "text/plain": {
    extType: ".txt",
    url: imgUrls[".txt"],
  },
  "application/zip": {
    extType: ".zip",
    url: imgUrls[".zip"],
  },
  "application/pdf": {
    extType: ".pdf",
    url: imgUrls[".pdf"],
  },
  "application/msword": {
    extType: ".doc",
    url: imgUrls[".doc"],
  },
  "application/vnd.openxmlformats-officedocument.wordprocessingml.document": {
    extType: ".docx",
    url: imgUrls[".docx"],
  },
  "application/vnd.ms-powerpoint": {
    extType: ".ppt",
    url: imgUrls[".ppt"],
  },
  "application/vnd.openxmlformats-officedocument.presentationml.presentation": {
    extType: ".pptx",
    url: imgUrls[".pptx"],
  },
  "application/vnd.ms-excel": {
    extType: ".xls",
    url: imgUrls[".xls"],
  },
  "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": {
    extType: ".xlsx",
    url: imgUrls[".xlsx"],
  },
  "image/jpeg": {
    extType: ".jpeg",
    url: imgUrls[".jpeg"],
  },
  "image/png": {
    extType: ".png",
    url: imgUrls[".png"],
  },
  "image/jpg": {
    extType: ".jpg",
    url: imgUrls[".jpg"],
  },
  default: {
    extType: "",
    url: "https://xxx/images/applet/ls/wenjian/default.png",
  },
};  

// 文件上传
  const props: UploadProps = {
    name: "file",
    accept: _.map([...acceptType, ".pdf", ".jpg", ".jpeg", ".png"], (ext) => acceptTypeObj[ext])
      .filter(Boolean)
      .join(","),
    multiple: multiple,
    beforeUpload: (file) => {
      // 检查文件上传不能超过5g且文件不能重复
      const boolean = BeforeUpload(file, fileList);
      return boolean ? boolean : Upload.LIST_IGNORE;
    },
    customRequest: async (option: any) => {
      try {
        // 后缀点的下标
        const index = option.file.name.lastIndexOf(".");
        //文件名称
        const name = option.file.name.slice(0, index);
        // 文件后缀类型 pdf,jpg,jpeg,....
        const extension = option.file.name.slice(index + 1);
        // 获取文件上传权限
        const {
          code,
          message: msg,
          data: obj,
        } = await getUploadToken({
          fileExt: extension,
          fileName: name,
          fileTag: fileTag,
          anjianId: anjianId,
        });
        if (code != 20000) {
          message.error(msg);
          return option.onError ? option.onError(msg) : null;
        }
        // 上传oss
        const uploadResult = await uploadCos(obj, option);
        if (uploadResult.err) {
          message.error(`上传文件失败,原因:${uploadResult.err}`);
          return option.onError ? option.onError(uploadResult.err) : null;
        } else {
          return option.onSuccess
            ? option.onSuccess({
                ...uploadResult.data,
                filePath: obj.filePath,
              })
            : null;
        }
      } catch (error) {
        message.error("上传失败");
      }
    },
    onChange: ({ fileList: changeFileList }) => {
      // 文件处理,如果是pdf、word、excel等文件,用缩列图展示
      try {
        const uploadList = _.map(changeFileList, (item: any) => {
          const fileInfo = fileTypeMap[item.type] || fileTypeMap.default;
          const extType = fileInfo.extType || "";
          let url = "";
          if (_.startsWith(item.type, "image/")) {
            url = item?.thumbUrl;
          } else {
            url = fileInfo.url;
          }
          // 上传错误的图像
          if (item?.status == "error") {
            url = errorImg;
          }
          return {
            ...item,
            thumbUrl: url,
            extType: extType,
          };
        });
        setFileList(uploadList);
      } catch (error) {
        message.error("上传出错!");
      }
    },
  };

4、自定义文件预览

我这里预览是打开了一个新窗口然后把地址放到一个ifram中去进行预览,把文件的url、name、type存储到sessionStorage(之前存储在localStorage,后面发现有个问题,每次预览不同的文件打开新的标签,只要一手动刷新窗口,标签里面拿到的是同一份文件,用sessionStorage就不会存在这个问题了),这里还有个问题就是pdf文件是可以通过将文件转换成一个可预览的url放到ifram中显示的,但是word、xls文件就不行(可能是浏览器本身不支持这两种文件预览,因为我这里没有用插件,都是用浏览器ifram去显示,后面我直接提示用户上传后再预览,因为后端对地址做了处理,预览文件是调用腾讯文档提供的在线预览的api),哪位大佬有什么好的方法在前端实现word、xls在线预览(不是使用了插件)可以告诉一下哈

  // 自定义预览
  const handlePreview = (file: any) => {
    // 判断上传完成的 status == done
    if (file?.status == "done") {
      // 图片的可以直接预览
      if ([".jpg", ".png", ".jpeg"].includes(file?.extType)) {
        const fileInfo = {
          url: file?.thumbUrl,
          fileName: file?.name,
          fileType: file?.extType,
        };
        saveLocalInfo("fileInfo", fileInfo);
        window.open("/preview", "_blank");
        return;
      }
      // pdf 的可以转换成可以预览的url
      if ([".pdf"].includes(file?.extType)) {
        const fileURL = URL.createObjectURL(file?.originFileObj);
        const fileInfo = {
          url: fileURL,
          fileName: file?.name,
          fileType: file?.extType,
        };
        saveLocalInfo("fileInfo", fileInfo);
        window.open("/preview", "_blank");
        return;
      }
      // 文档类型的要上传后才能预览
      if ([".doc", ".docx", ".xls", ".xls"].includes(file?.extType)) {
        return message.warn("该文档需上传后才能进行预览");
      }
      return message.warn("当前文件不支持预览");
    }

    // 上传错误的文件
    if (file?.status == "error") {
      return message.error("预览失败");
    }
    return message.error("预览失败");
  };

预览效果:

![image-20240429160826876](/Users/lance/Library/Application Support/typora-user-images/image-20240429160826876.png)

5、完整代码:

import { getUploadToken } from "@/services/auth/service";
import { saveLocalInfo } from "@/utils/tool";
import { BeforeUpload, uploadCos } from "@/utils/upload";
import { CloudUploadOutlined, UploadOutlined } from "@ant-design/icons";
import { Button, Upload, UploadProps, message } from "antd";
import _ from "lodash";

const errorImg =
  "";



interface AcceptType {
  [extension: string]: string;
}

// 常见的上传文件类型
const acceptTypeObj: AcceptType = {
  ".txt": "text/plain",
  ".html": "text/html",
  ".js": "text/javascript",
  ".css": "text/css",
  ".csv": "text/csv",
  ".json": "application/json",
  ".doc": "application/msword",
  ".docx": "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
  ".ppt": "application/vnd.ms-powerpoint",
  ".pptx": "application/vnd.openxmlformats-officedocument.presentationml.presentation",
  ".xls": "application/vnd.ms-excel",
  ".xlsx": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
  ".xml": "application/xml",
  ".pdf": "application/pdf",
  ".jpg": "image/jpeg",
  ".jpeg": "image/jpeg",
  ".png": "image/png",
  ".gif": "image/gif",
  ".bmp": "image/bmp",
  ".tif": "image/tiff",
  ".tiff": "image/tiff",
  ".ico": "image/x-icon",
  ".svg": "image/svg+xml",
  ".mp3": "audio/mpeg",
  ".wav": "audio/wav",
  ".mp4": "video/mp4",
  ".avi": "video/x-msvideo",
  ".wmv": "video/x-ms-wmv",
  ".mov": "video/quicktime",
  ".flv": "video/x-flv",
  ".webm": "video/webm",
  ".mpeg": "video/mpeg",
  ".zip": "application/zip",
  ".rar": "application/x-rar-compressed",
  ".tar": "application/x-tar",
  ".gz": "application/gzip",
  ".7z": "application/x-7z-compressed",
};

// 文件图片样式地址,这里要自己找哈
const imgUrls = {
  ".doc": "https://xxx/images/applet/ls/wenjian/doc.png",
  ".docx": "https://xxx/images/applet/ls/wenjian/docx.png",
  ".csv": "https://xxx/images/applet/ls/wenjian/csv.png",
  ".html": "https://xxx/images/applet/ls/wenjian/html.png",
  ".jpg": "https://xxx/images/applet/ls/wenjian/jpg.png",
  ".jpeg": "https://xxx/images/applet/ls/wenjian/jpg.png",
  ".mp4": "https://xxx/images/applet/ls/wenjian/mp4.png",
  ".pdf": "https://xxx/images/applet/ls/wenjian/pdf.png",
  ".png": "https://xxx/images/applet/ls/wenjian/png.png",
  ".ppt": "https://xxx/images/applet/ls/wenjian/ppt.png",
  ".pptx": "https://xxx/images/applet/ls/wenjian/pptx.png",
  ".txt": "https://xxx/images/applet/ls/wenjian/txt.png",
  ".xls": "https://xxx/images/applet/ls/wenjian/xlsx.png",
  ".xlsx": "https://xxx/images/applet/ls/wenjian/xlsx.png",
  ".zip": "https://xxx/images/applet/ls/wenjian/zip.png",
  default: "https://xxx/images/applet/ls/wenjian/default.png",
};
// 文件类型匹配
const fileTypeMap = {
  "text/plain": {
    extType: ".txt",
    url: imgUrls[".txt"],
  },
  "application/zip": {
    extType: ".zip",
    url: imgUrls[".zip"],
  },
  "application/pdf": {
    extType: ".pdf",
    url: imgUrls[".pdf"],
  },
  "application/msword": {
    extType: ".doc",
    url: imgUrls[".doc"],
  },
  "application/vnd.openxmlformats-officedocument.wordprocessingml.document": {
    extType: ".docx",
    url: imgUrls[".docx"],
  },
  "application/vnd.ms-powerpoint": {
    extType: ".ppt",
    url: imgUrls[".ppt"],
  },
  "application/vnd.openxmlformats-officedocument.presentationml.presentation": {
    extType: ".pptx",
    url: imgUrls[".pptx"],
  },
  "application/vnd.ms-excel": {
    extType: ".xls",
    url: imgUrls[".xls"],
  },
  "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": {
    extType: ".xlsx",
    url: imgUrls[".xlsx"],
  },
  "image/jpeg": {
    extType: ".jpeg",
    url: imgUrls[".jpeg"],
  },
  "image/png": {
    extType: ".png",
    url: imgUrls[".png"],
  },
  "image/jpg": {
    extType: ".jpg",
    url: imgUrls[".jpg"],
  },
  default: {
    extType: "",
    url: "https://static.zhongbaisubao.com/images/applet/ls/wenjian/default.png",
  },
};

interface UploadTypes {
  acceptType: string[]; // 上传的文件类型,
  fileList: any[]; // 上传的文件列表
  setFileList: any;
  listType: "text" | "picture" | "picture-card"; // 上传卡片样式
  disabled?: boolean | false; // 是否禁用上传
  fileTag: number; // 上传文件类型标签
  anjianId: string | number;
  multiple?: boolean; // 是否允许文件多选,默认多选
}

const UploadCard: React.FC<UploadTypes> = ({ acceptType, fileList, setFileList, listType, disabled, anjianId = 0, fileTag = 4, multiple = true }) => {
  // 文件上传
  const props: UploadProps = {
    name: "file",
    accept: _.map([...acceptType, ".pdf", ".jpg", ".jpeg", ".png"], (ext) => acceptTypeObj[ext])
      .filter(Boolean)
      .join(","),
    multiple: multiple,
    beforeUpload: (file) => {
      // 检查文件上传不能超过5g且文件不能重复
      const boolean = BeforeUpload(file, fileList);
      return boolean ? boolean : Upload.LIST_IGNORE;
    },
    customRequest: async (option: any) => {
      try {
        // 后缀点的下标
        const index = option.file.name.lastIndexOf(".");
        //文件名称
        const name = option.file.name.slice(0, index);
        // 文件后缀类型 pdf,jpg,jpeg,....
        const extension = option.file.name.slice(index + 1);
        // 获取文件上传权限
        const {
          code,
          message: msg,
          data: obj,
        } = await getUploadToken({
          fileExt: extension,
          fileName: name,
          fileTag: fileTag,
          anjianId: anjianId,
        });
        if (code != 20000) {
          message.error(msg);
          return option.onError ? option.onError(msg) : null;
        }
        // 上传oss
        const uploadResult = await uploadCos(obj, option);
        console.log(uploadResult,'uploadResult....')
        if (uploadResult.err) {
          message.error(`上传文件失败,原因:${uploadResult.err}`);
          return option.onError ? option.onError(uploadResult.err) : null;
        } else {
          return option.onSuccess
            ? option.onSuccess({
                ...uploadResult.data,
                filePath: obj.filePath,
              })
            : null;
        }
      } catch (error) {
        message.error("上传失败");
      }
    },
    onChange: ({ fileList: changeFileList }) => {
      // 文件处理,如果是pdf、word、excel等文件,用缩列图展示
      try {
        const uploadList = _.map(changeFileList, (item: any) => {
          const fileInfo = fileTypeMap[item.type] || fileTypeMap.default;
          const extType = fileInfo.extType || "";
          let url = "";
          if (_.startsWith(item.type, "image/")) {
            url = item?.thumbUrl;
          } else {
            url = fileInfo.url;
          }
          // 上传错误的图像
          if (item?.status == "error") {
            url = errorImg;
          }
          return {
            ...item,
            thumbUrl: url,
            extType: extType,
          };
        });
        setFileList(uploadList);
      } catch (error) {
        message.error("上传出错!");
      }
    },
  };

  

  // 自定义预览
  const handlePreview = (file: any) => {
    // 判断上传完成的 status == done
    if (file?.status == "done") {
      // 图片的可以直接预览
      if ([".jpg", ".png", ".jpeg"].includes(file?.extType)) {
        const fileInfo = {
          url: file?.thumbUrl,
          fileName: file?.name,
          fileType: file?.extType,
        };
        saveLocalInfo("fileInfo", fileInfo);
        window.open("/preview", "_blank");
        return;
      }
      // pdf 的可以转换成可以预览的url
      if ([".pdf"].includes(file?.extType)) {
        const fileURL = URL.createObjectURL(file?.originFileObj);
        const fileInfo = {
          url: fileURL,
          fileName: file?.name,
          fileType: file?.extType,
        };
        saveLocalInfo("fileInfo", fileInfo);
        window.open("/preview", "_blank");
        return;
      }
      // 文档类型的要上传后才能预览
      if ([".doc", ".docx", ".xls", ".xls"].includes(file?.extType)) {
        return message.warn("该文档需上传后才能进行预览");
      }
      return message.warn("当前文件不支持预览");
    }

    // 上传错误的文件
    if (file?.status == "error") {
      return message.error("预览失败");
    }
    return message.error("预览失败");
  };

  return (
    <Upload listType={listType} {...props} fileList={fileList} onPreview={handlePreview} name="fashen_fujian" disabled={disabled}>
      {listType == "picture-card" && (
        <div>
          <CloudUploadOutlined style={{ fontSize: 30, color: "gray" }} />
        </div>
      )}
      {(listType == "picture" || listType == "text") && <Button icon={<UploadOutlined />}>上传本地文件</Button>}
    </Upload>
  );
};

export default UploadCard;

最后:

希望对大家有所帮助哈, 我觉得还是要勤做笔记,像我觉得之前明明做过类似的功能,再使用时又忘记怎么做了😭,笔记没做,还是得百度,最后还是白白浪费了时间。。