antd 批量上傳文件邏輯

来源:https://www.cnblogs.com/czzj/archive/2022/09/20/16711158.html
-Advertisement-
Play Games

通過 antd 框架的 Upload 控制項,採用手動上傳的方式,先選擇需要上傳的文件(控制文件數量以及大小),再根據所選的文件列表,迴圈上傳,期間通過 Spin 控制項提示上傳中。 ...


基本步驟

  通過 antd 框架的 Upload 控制項,採用手動上傳的方式,先選擇需要上傳的文件(控制文件數量以及大小),再根據所選的文件列表,迴圈上傳,期間通過 Spin 控制項提示上傳中。

效果展示

  

 控制項引用

  Upload 控制項配置:

 1 props : {
 2   multiple: true,
 3   maxCount:20,//限制最多顯示 20 個文件
 4   onRemove: (file) => {//刪除列表文件
 5     let fileListbatch_curr=this.state.fileListbatch;
 6     console.log("props-onRemove-fileListbatch_curr:",fileListbatch_curr);
 7     let index = fileListbatch_curr.findIndex(item=>item.uid==file.uid);
 8     console.log("props-onRemove-obj:",index,file.uid);
 9     if(index==-1){
10       //message.warning("未刪除成功!")
11       return;
12     }else{
13       const newFileList = fileListbatch_curr.slice();
14       newFileList.splice(index, 1);
15       console.log("props-onRemove-newFileList:",newFileList);
16       this.setState({fileListbatch:newFileList});
17       let uploadsuccesslist_curr=this.state.uploadsuccesslist;
18       let indexsuccess=uploadsuccesslist_curr.findIndex(item=>item.uid==file.uid);//根據唯一碼 uid 查找目標文件的索引
19       if(indexsuccess!=-1){
20         uploadsuccesslist_curr.splice(indexsuccess, 1);//刪除
21         console.log("props-onRemove-uploadsuccesslist_curr:",uploadsuccesslist_curr);
22         this.setState({uploadsuccesslist:uploadsuccesslist_curr});
23       }
24     }
25   },
26   beforeUpload: (file) => {//添加文件,將文件加入臨時列表,準備上傳
27     let fileListbatch_curr=this.state.fileListbatch;
28     let ff=fileListbatch_curr.find((item)=>item.name==file.name);
29     if(ff==undefined){
30       fileListbatch_curr.push(file);
31       this.setState({fileListbatch:fileListbatch_curr,uploadflag:false});
32       return false;
33     }
34     else{
35       message.warning("存在同名文件已選擇,請確認!");
36       return Upload.LIST_IGNORE;//列表中不顯示
37     }
38   },
39   fileListbatch_cc,
40 }

  界面元素排列:

 1 <Upload {...props}>
 2   <Button icon={<UploadOutlined />}>選擇文件(Max:20Pcs, Max:200MB)</Button>
 3 </Upload>
 4 <Button
 5   type="primary"
 6   onClick={this.batchUploadReports}
 7   disabled={fileListbatch.length === 0}
 8   //loading={uploading}
 9   style={{ marginTop: 20,width:180 }}
10 >
11   {/* {uploading ? 'Uploading' : 'Start Upload'} */}
12   開始上傳
13 </Button>
14 <label style={{ lineHeight: "0px", color: '#bfbfbf', fontSize: 10,float:'left',marginBottom:20 }}>支持擴展名:apk/exe/pdf/xls/doc/ppt等</label>

處理邏輯

  根據文件列表,迴圈上傳全部文件。

  【後端採用 .Net 5.0 WebAPI 詳見:大文件分片上傳 中的“後端部分”】

  1 //批量上傳文件
  2 batchUploadReports = () => {
  3 this.setState({uploadsuccesslist:[]})
  4 let uploadsuccesslist_curr=[];
  5 this.formRef_upload.current.validateFields()
  6 .then(formrefcurr => {
  7   if(formrefcurr["baogaolb"]==0||formrefcurr["baogaolx"].length==0){
  8     message.warning("請檢查必填項!");
  9     return null;
 10   }
 11   this.setState({fileuploading:true,uploadsuccesslist:[],tipContent:"報告文件上傳中,請耐心等待..."});
 12   let fileListbatch_curr2=this.state.fileListbatch;
 13   let filecount=fileListbatch_curr2.length;
 14   let filecount_success=0;
 15   if(filecount>20){//只取前 20 個文件
 16     filecount=20;
 17     fileListbatch_curr2=fileListbatch_curr2.splice(0,20);
 18   }
 19   try{//通過拋出異常,來中斷 foreach
 20     fileListbatch_curr2.forEach(element => {//文件 list 迴圈上傳
 21       if(element.size/(1024*1024)>200){
 22         this.setState({fileuploading:false,uploadsuccesslist:[],tipContent:"報告文件上傳中,請耐心等待..."})
 23         message.warning("單個報告大小不允許超過 200MB,請檢查後繼續上傳!");
 24         // fileListbatch_curr2.length=0;
 25         throw new Error(element)//若有不符合條件的文件,拋出異常中斷迴圈
 26       }
 27       else{
 28         let filename=element.name;
 29         let chunklistcurr=this.createFileChunk(element).map(({ file, name }, index) => {//createFileChunk:創建文件切片,可處理大文件
 30             return {
 31               chunk: file,
 32               size: file.size,
 33               percent: 0,
 34               //filename:file.
 35               name: name,// + "-" + (index + 1),
 36               index,
 37             };
 38           });
 39         let filecountchunk=chunklistcurr.length;
 40         let ii=0;
 41         chunklistcurr.forEach(element => {//分片傳輸
 42           const formData = new FormData()
 43           formData.append('file', element.chunk);
 44           formData.append('index', element.index);
 45           formData.append('name', element.name);
 46           formData.append('size', element.size);
 47           //formData.append('filecount', filecount);
 48           axios({
 49             method: 'post',
 50             url: '/api/system/System/UploadFileAttachmentChunk?zhuti='+this.state.zhuti,
 51             data: formData,
 52             headers: { "Content-Type": "multipart/form-data"}
 53           }).then(({data}) => {
 54             if(data.code==200){
 55               ii++;
 56               if(ii==filecountchunk){//分塊全部上傳完成
 57                 let indata={"name":chunklistcurr[0].name,"filecount":filecountchunk,"filename":filename
 58                   }
 59                 axios({//傳輸完成,通知拼接
 60                   method: 'post',
 61                   url: '/api/system/System/CombineChunkToFileBatch',
 62                   data: indata,
 63                   headers: { "Content-Type": "application/json"}
 64                 }).then(({data}) => {
 65                   if(data.code==200){
 66                     let listbatchupload={}
 67                     uploadsuccesslist_curr.push(listbatchupload);
 68                     this.setState({uploadsuccesslist:uploadsuccesslist_curr});
 69                     filecount_success++;
 70                   }
 71                   else if(data.code==202){
 72                     window.location.href="/wellcome";
 73                   }
 74                   else{
 75                     message.error("上傳失敗,請稍後重試!詳情:"+data.desc);
 76                     filecount_success++;
 77                   }
 78                 }).catch((err) =>{
 79                     console.log(err);
 80                     message.error("上傳失敗,請稍後重試!");
 81                 }).finally(() =>{
 82                   if(filecount_success==filecount){
 83                     this.setState({uploadflag:true})
 84                     message.success("全部上傳完成,請進一步確認上傳是否全部成功");
 85                     this.setState({fileuploading:false,tipContent:"載入中..."});
 86                   }
 87                 })
 88               }
 89             }
 90             else if(data.code==202){
 91               window.location.href="/wellcome";
 92             }
 93             else{
 94               message.error("上傳失敗,請稍後重試!詳情:"+data.desc);
 95               chunklistcurr.length=0;
 96               return;
 97             }
 98           }).catch((err) =>{
 99               console.log(err);
100               message.error("上傳失敗,請稍後重試!");
101               return;
102           }).finally(() =>{ })
103         })
104       }
105     });
106   }
107   catch(e){        
108     console.log('catch-e:'+e)
109   }
110 })
111 }
 1 //創建文件切片
 2 createFileChunk = (file, size = 20*1024*1024) => {
 3     const fileChunkList = [];
 4     let cur = 0;
 5     while (cur < file.size) {
 6       fileChunkList.push({ file: file.slice(cur, cur + size), name: file.uid });//uid:文件唯一標識
 7       cur += size;
 8     }
 9     return fileChunkList;
10 };

 註:本文代碼已在項目中實用,有疑問歡迎指正。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 2022 年 9 月 16 日,由中國信息通信研究院(以下簡稱“信通院”)主辦的“2022 OSCAR 開源產業大會"活動於北京成功舉辦。會上宣佈,StoneDB 發起廠商杭州石原子科技有限公司正式加入信通院“科技製造開源社區(TMOSC)”,未來石原子將與信通院及各成員單位一起聚焦可信開源全景,推 ...
  • 本篇為Redis性能問題診斷系列的第四篇,也是最後一篇,主要從應用程式、系統、伺服器硬體及網路系統等層面上進行講解,重點分享了哪些配置需要重點關註和調整優化,才能最大程度的發揮Redis的處理能力; ...
  • 一.起步 1.1 配置uni-app開發環境 什麼是uni-app,就是基於vue的一個開發框架,可以將我們寫的一套代碼,同時發佈到ios、安卓、小程式等多個平臺 ==官方推薦使用Hbuilderx來寫uni-app項目== 下載之後可以將預設改為vscode 進入hbuilder插件市場下載scs ...
  • 前端技術的發展不斷融入了很多後端的思想,逐步形成前端的 ”四個現代化“:工程化、模塊化、規範化、流程化。這個主題介紹 *模塊化* ,主要內容包括模塊化前傳(早期模塊化的實現)、模塊化的四個規範(Common JS、AMD、CMD、ESM)。本文就聊聊早期的模塊化。 ...
  • 現在的很多程式應用,基本上都是需要多端覆蓋,因此基於一個Web API的後端介面,來構建多端應用,如微信、H5、APP、WInForm、BS的Web管理端等都是常見的應用。本篇隨筆繼續分析總結一下項目開發的經驗,針對頁面組件化開發經驗方面進行一些梳理總結,內容包括組件的概念介紹,簡單頁面組件的抽取開... ...
  • 每日3題 1 以下代碼執行後,控制臺中的輸出內容為? // index.js console.log(1); import { sum } from "./sum.js"; console.log(sum(1, 2)); //sum.js console.log(2); export const s ...
  • 導讀:面對多種多樣的跨端訴求,有哪些跨端方案?跨端的本質是什麼?作為業務技術開發者,應該怎麼做?本文分享阿裡巴巴ICBU技術部在跨端開發上的一些思考,介紹了當前主流的跨端方案,以及跨端開發的經驗心得。 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 最近在研究一個基於TP6的框架CRMEB,這裡分享下我的開發心得 首先在上篇文章中,我們安裝了CRMEBphp介面項目,需要可以看這一篇 TP6框架--CRMEB學習筆記:項目初始化+環境配置 1.獲取項目 這裡是git地址 https: ...
一周排行
    -Advertisement-
    Play Games
  • 人臉識別技術在現代社會中扮演著越來越重要的角色,比如人臉識別門禁、人臉識別支付、甚至人臉識別網站登錄等。 最近有群友問.NET有沒有人臉識別的組件,小編查閱相關資料介紹下麵幾種.NET人臉識別組件供大家參考。 **1、Microsoft Azure Face API** 簡介:Microsoft A ...
  • # 1. 與 .NET Core 緩存的關係和差異 ABP 框架中的緩存系統核心包是 [Volo.Abp.Caching](https://www.nuget.org/packages/Volo.Abp.Caching) ,而對於分散式緩存的支持,abp 官方提供了基於 Redis 的方案,需要安裝 ...
  • 最近ET做熱更重載dll的時候,返回登陸會重新檢測新的dll,首次登錄之前已經Assembly.Load()過一次dll,第二次返回登陸再次load dll到記憶體中,Invoke執行方法的時候,異常了,有些方法執行了,有些未執行,於是查資料,看到些老資料說Assembly.Load重覆載入同名dll ...
  • 1. 擴展方法 擴展方法使你能夠向現有類型“添加”方法,而無需創建新的派生類型、重新編譯或以其他方式修改原始類型。 擴展方法是一種靜態方法,但可以像擴展類型上的實例方法一樣進行調用。 對於用 C#、F# 和 Visual Basic 編寫的客戶端代碼,調用擴展方法與調用在類型中定義的方法沒有明顯區別 ...
  • 以前在隨筆《Winform開發框架之客戶關係管理系統(CRM)的開發總結系列1-界面功能展示 》的幾篇隨筆中介紹過基於WInform開發框架開發的CRM系統,系統的功能主要也是圍繞著客戶相關信息來進行管理的。本篇隨筆介紹在最新的《SqlSugar開發框架》中整合CRM系統模塊的功能。 ...
  • 隨著技術的發展,ASP.NET Core MVC也推出了好長時間,經過不斷的版本更新迭代,已經越來越完善,本系列文章主要講解ASP.NET Core MVC開發B/S系統過程中所涉及到的相關內容,適用於初學者,在校畢業生,或其他想從事ASP.NET Core MVC 系統開發的人員。 經過前幾篇文章... ...
  • [toc] 這篇文章是我之前總結的一篇文章,因為整理博客的原因,原有博客已經註銷,但這篇文章對一些讀者很有用,所以現在新瓶裝舊酒重新整理回來分享給大家。 最近一段時間生產環境頻繁出問題,每次都會生成一個hs_err_pid*.log文件,因為工作內容的原因,在此之前並沒有瞭解過相關內容,趁此機會學習 ...
  • # 前言 在上一篇文章中,給大家講解了泛型的概念、作用、使用場景,以及泛型集合、泛型介面和泛型類的用法,但受限於篇幅,並沒有把泛型的內容講解完畢。所以今天我們會繼續學習泛型方法、泛型擦除,以及通配符等的內容,希望大家繼續做好學習的準備哦。 *** 全文大約【**4600】** 字,不說廢話,只講可以 ...
  • 昨天遇到參數key大小寫不一致導致校驗簽名失敗的問題,查了很長時間才找到原因。看了一下FastJson源碼,發現JSON.toObject中轉換成對象的時候會忽略大小寫。 所以,當使用了JSON.toObject將json轉成Java對象後,再用JSON.toObject轉成json,key值就變了 ...
  • 基於java的線上商城設計與實現,線上購物平臺,校園購物商城,商品銷售平臺,基於Java的電商平臺;電商平臺,買家和賣家可以在此平臺上進行銷售和交易,節約了大量的線下時間成本,購物車的功能,校園交易平臺等等; ...