• <fieldset id="8imwq"><menu id="8imwq"></menu></fieldset>
  • <bdo id="8imwq"><input id="8imwq"></input></bdo>
    最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
    問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
    當前位置: 首頁 - 科技 - 知識百科 - 正文

    js實現控制文件拖拽并獲取拖拽內容功能

    來源:懂視網 責編:小采 時間:2020-11-27 22:19:23
    文檔

    js實現控制文件拖拽并獲取拖拽內容功能

    js實現控制文件拖拽并獲取拖拽內容功能:在用戶拖拽文件到瀏覽器的某個元素上時,js可以監聽到與拖拽相關的事件,并對拖拽結果進行處理,本文討論下和拖拽文件相關的一些問題,不過沒有處理太多關于兼容性的問題。 拖拽事件 js能夠監聽到拖拽的事件有drag、dragend、dragenter、dragexit
    推薦度:
    導讀js實現控制文件拖拽并獲取拖拽內容功能:在用戶拖拽文件到瀏覽器的某個元素上時,js可以監聽到與拖拽相關的事件,并對拖拽結果進行處理,本文討論下和拖拽文件相關的一些問題,不過沒有處理太多關于兼容性的問題。 拖拽事件 js能夠監聽到拖拽的事件有drag、dragend、dragenter、dragexit

    在用戶拖拽文件到瀏覽器的某個元素上時,js可以監聽到與拖拽相關的事件,并對拖拽結果進行處理,本文討論下和拖拽文件相關的一些問題,不過沒有處理太多關于兼容性的問題。

    拖拽事件

    js能夠監聽到拖拽的事件有drag、dragend、dragenter、dragexit(沒有瀏覽器實現)、dragleave、dragover、dragstart、drop,詳細的內容可以看MDN。

    其中,與拖拽文件相關的事件有dragenter(文件拖拽進)、dragover(文件拖拽在懸浮)、dragleave(文件拖拽離開)、drop(文件拖拽放下)。

    拖拽事件可以綁定到指定的DOM元素上,可以綁定到整個頁面中。

    var dropEle = document.querySelector('#dropZone');
    dropEle.addEventListener('drop', function (e) {
     // 
    }, false);
    
    document.addEventListener('drop', function (e) {
     // 
    }, false);

    阻止默認行為

    一般來說,我們只需要把處理拖拽文件的業務邏輯寫到drop事件中就可以了,為什么還要綁定dragenter、dragover、dragleave這三個事件呢?

    因為當你拖拽一個文件到沒有對拖拽事件進行處理的瀏覽器中的時候,瀏覽器會打開這個文件,比如拖拽一張圖片瀏覽器會打開這個圖片,在沒有PDF閱讀器的時候也可以拖拽一個PDF到瀏覽器中,瀏覽器就會打開這個PDF文件。

    如果瀏覽器打開了拖拽的文件,頁面就跳走了,我們希望得到拖拽的文件,而不是讓頁面跳走。上面說到瀏覽器會打開拖拽的文件是瀏覽器的默認行為,我們需要阻止這個默認行為,就需要再上述的事件中進行阻止。

    dropZone.addEventListener("dragenter", function (e) {
     e.preventDefault();
     e.stopPropagation();
    }, false);
    
    dropZone.addEventListener("dragover", function (e) {
     e.preventDefault();
     e.stopPropagation();
    }, false);
    
    dropZone.addEventListener("dragleave", function (e) {
     e.preventDefault();
     e.stopPropagation();
    }, false);
    
    dropZone.addEventListener("drop", function (e) {
     e.preventDefault();
     e.stopPropagation();
     // 處理拖拽文件的邏輯
    }

    實際上dragenter不阻止默認行為也不會觸發瀏覽器打開文件,為了防止某些瀏覽器可能有的兼容性問題,把拖拽周期中的所有的事件都阻止默認行為并且阻止了事件冒泡。

    獲得拖拽的文件

    我們會在drop這個事件的回調中的事件對象能夠得到文件對象。

    在事件對象中,一個e.dataTransfer這樣的屬性,它是一個DataTransfer類型的數據,有如下的屬性

    屬性 類型 說明
    dropEffect String 用來hack某些兼容性問題
    effectAllowed String 暫時不用
    files FileList 拖拽的文件列表
    items DataTransferItemList 拖拽的數據(有可能是字符串)
    types Array 拖拽的數據類型 該屬性在Safari下比較混亂

    在Chrome中我們用items對象獲得文件,其他瀏覽器用files獲得文件,主要是為了處理拖拽文件夾的問題,最好不允許用戶拖拽文件夾,因為文件夾內可能還有文件夾,遞歸上傳文件會很久,如果不遞歸查找,只上傳目錄第一層級的文件,用戶可能以為上傳功能了,但是沒有上傳子目錄文件,所以還是禁止上傳文件夾比較好,后面我會說要怎么處理。

    Chrome獲取文件

    dropZone.addEventListener("drop", function (e) {
     e.preventDefault();
     e.stopPropagation();
     
     var df = e.dataTransfer;
     var dropFiles = []; // 存放拖拽的文件對象
     
     if(df.items !== undefined) {
     // Chrome有items屬性,對Chrome的單獨處理
     for(var i = 0; i < df.items.length; i++) {
     var item = df.items[i];
     // 用webkitGetAsEntry禁止上傳目錄
     if(item.kind === "file" && item.webkitGetAsEntry().isFile) {
     var file = item.getAsFile();
     dropFiles.push(file);
     }
     }
     }
    }

    其他瀏覽器獲取文件

    這里只測試了Safari,其他瀏覽器并沒有測試,不過看完本文一定也有思路處理其他瀏覽器的兼容情況。

    dropZone.addEventListener("drop", function (e) {
     e.preventDefault();
     e.stopPropagation();
     
     var df = e.dataTransfer;
     var dropFiles = []; // 存放拖拽的文件對象
     
     if(df.items !== undefined) {
     // Chrome拖拽文件邏輯
     } else {
     for(var i = 0; i < df.files.length; i++) {
     dropFiles.push(df.files[i]);
     }
     }
    }

    由于Safari沒有item,自然也沒有webkitGetAsEntry,所以在Safari無法確定拖拽的是否是文件還是文件夾。

    非Chrome內核瀏覽器判斷目錄的方法

    瀏覽器獲取到的每個file對象有四個屬性:lastModified、name、size、type,其中type是文件的MIME Type,文件夾的type是空的,但是有些文件沒有MIME Type,如果按照type是否為空判斷是不是拖拽的文件夾的話,會誤傷一部分文件,所以這個方法行。

    那么還有什么方法可以判斷呢,思路大概是這樣子的,用戶拖拽的文件和文件夾應該是不一樣的東西,用File API操作的時候應該會有區別,比如進行某些操作的時候,文件就能夠正常操作,但是文件夾就會報錯,通過錯誤的捕獲就能夠判斷是文件還是文件夾了,好我們根據這個思路來寫一下。

    dropZone.addEventListener("drop", function (e) {
     e.preventDefault();
     e.stopPropagation();
    
     var df = e.dataTransfer;
     var dropFiles = [];
     
     if(df.items !== undefined){
     // Chrome拖拽文件邏輯
     } else {
     for(var i = 0; i < df.files.length; i++){
     var dropFile = df.files[i];
     if ( dropFile.type ) {
     // 如果type不是空串,一定是文件
     dropFiles.push(dropFile);
     } else {
     try {
     var fileReader = new FileReader();
     fileReader.readAsDataURL(dropFile.slice(0, 3));
    
     fileReader.addEventListener('load', function (e) {
     console.log(e, 'load');
     dropFiles.push(dropFile);
     }, false);
    
     fileReader.addEventListener('error', function (e) {
     console.log(e, 'error,不可以上傳文件夾');
     }, false);
    
     } catch (e) {
     console.log(e, 'catch error,不可以上傳文件夾');
     }
     }
     }
     }
    }, false);

    上面代碼創建了一個FileReader實例,通過這個實例對文件進行讀取,我測試讀取一個1G多的文件要3S多,時間有點長,就用slice截取了前3個字符,為什么是前3個不是前2個或者前4個呢,因為代碼是我寫的,我開心這么寫唄~

    如果load事件觸發了,就說明拖拽過來的東西是文件,如果error事件觸發了,就說明是文件夾,為了防止其他可能的潛在錯誤,用try包起來這段代碼。

    三方應用的一點點小hack

    經過測試發現通過Mac的Finder拖拽文件沒有問題,但是有時候文件并不一定在Finder中,也可能在某些應用中,有一個應用叫做圈點,這個應用的用戶反饋文件拖拽失效,去看了其他開源文件上傳的源碼,發現了這樣一行代碼:

    dropZone.addEventListener("dragover", function (e) {
     e.dataTransfer.dropEffect = 'copy'; // 兼容某些三方應用,如圈點
     e.preventDefault();
     e.stopPropagation();
    }, false);

    需要把dropEffect置為copy,上網搜了下這個問題,源碼文檔中也沒有說為什么要加這個,有興趣的同學可以找一下為什么。

    可以拿來就用的代碼

    由于用了FileReader去讀取文件,這是一個異步IO操作,為了記錄當前處理了多少個文件,以及什么時候觸發拖拽結束的回調,寫了一個checkDropFinish的方法一直去比較處理的文件數量和文件總數,確定所有文件處理完了后就去調用完成的回調。

    另外,我在最后調試異步處理的時候,用的斷點調試,發現斷點調試在Safari中會導致異步回調不觸發,需要自己調試定制功能的同學注意下。

    // 獲得拖拽文件的回調函數
    function getDropFileCallBack (dropFiles) {
     console.log(dropFiles, dropFiles.length);
    }
    
    var dropZone = document.querySelector("#dropZone");
    dropZone.addEventListener("dragenter", function (e) {
     e.preventDefault();
     e.stopPropagation();
    }, false);
    
    dropZone.addEventListener("dragover", function (e) {
     e.dataTransfer.dropEffect = 'copy'; // 兼容某些三方應用,如圈點
     e.preventDefault();
     e.stopPropagation();
    }, false);
    
    dropZone.addEventListener("dragleave", function (e) {
     e.preventDefault();
     e.stopPropagation();
    }, false);
    
    dropZone.addEventListener("drop", function (e) {
     e.preventDefault();
     e.stopPropagation();
    
     var df = e.dataTransfer;
     var dropFiles = []; // 拖拽的文件,會放到這里
     var dealFileCnt = 0; // 讀取文件是個異步的過程,需要記錄處理了多少個文件了
     var allFileLen = df.files.length; // 所有的文件的數量,給非Chrome瀏覽器使用的變量
    
     // 檢測是否已經把所有的文件都遍歷過了
     function checkDropFinish () {
     if ( dealFileCnt === allFileLen-1 ) {
     getDropFileCallBack(dropFiles);
     }
     dealFileCnt++;
     }
    
     if(df.items !== undefined){
     // Chrome拖拽文件邏輯
     for(var i = 0; i < df.items.length; i++) {
     var item = df.items[i];
     if(item.kind === "file" && item.webkitGetAsEntry().isFile) {
     var file = item.getAsFile();
     dropFiles.push(file);
     console.log(file);
     }
     }
     } else {
     // 非Chrome拖拽文件邏輯
     for(var i = 0; i < allFileLen; i++) {
     var dropFile = df.files[i];
     if ( dropFile.type ) {
     dropFiles.push(dropFile);
     checkDropFinish();
     } else {
     try {
     var fileReader = new FileReader();
     fileReader.readAsDataURL(dropFile.slice(0, 3));
    
     fileReader.addEventListener('load', function (e) {
     console.log(e, 'load');
     dropFiles.push(dropFile);
     checkDropFinish();
     }, false);
    
     fileReader.addEventListener('error', function (e) {
     console.log(e, 'error,不可以上傳文件夾');
     checkDropFinish();
     }, false);
    
     } catch (e) {
     console.log(e, 'catch error,不可以上傳文件夾');
     checkDropFinish();
     }
     }
     }
     }
    }, false);

    聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

    文檔

    js實現控制文件拖拽并獲取拖拽內容功能

    js實現控制文件拖拽并獲取拖拽內容功能:在用戶拖拽文件到瀏覽器的某個元素上時,js可以監聽到與拖拽相關的事件,并對拖拽結果進行處理,本文討論下和拖拽文件相關的一些問題,不過沒有處理太多關于兼容性的問題。 拖拽事件 js能夠監聽到拖拽的事件有drag、dragend、dragenter、dragexit
    推薦度:
    標簽: js 拖拽 拖拽文件
    • 熱門焦點

    最新推薦

    猜你喜歡

    熱門推薦

    專題
    Top
    主站蜘蛛池模板: 久久激情亚洲精品无码?V| 亚洲精品456播放| 九九线精品视频在线观看| 国内精品伊人久久久久av一坑| 国产精品你懂的在线播放| 2021国产成人精品国产| 午夜精品久久久久久| 国产精品99久久久久久董美香| 国语自产精品视频在线区| 亚洲国产精品成人| 国产一区二区三区欧美精品| 国产成人精品久久一区二区三区| 一本色道久久88综合日韩精品| 国产欧美日本精品| 四虎精品免费永久在线| 69国产成人综合久久精品| 久久丫精品国产亚洲av不卡| 亚洲精品97久久中文字幕无码| 国产精品高清一区二区人妖| 亚洲精品无码专区在线播放| 国产精品视频全国免费观看| 久久久免费精品re6| 国产亚洲精品a在线观看| 欧美国产亚洲精品高清不卡| 国产精品成熟老女人视频| 亚洲麻豆精品国偷自产在线91| 精品久久久久国产免费| 韩国三级中文字幕hd久久精品 | 欲帝精品福利视频导航| 好湿好大硬得深一点动态图91精品福利一区二区 | 亚洲AV永久纯肉无码精品动漫| 无码AⅤ精品一区二区三区| 久久精品国产精品亚洲| 久久99精品久久久久久秒播| 久久九九久精品国产| 蜜臀精品国产高清在线观看| 久夜色精品国产一区二区三区| 欧美成人精品一区二区综合| 亚洲福利精品电影在线观看| 亚洲精品人成无码中文毛片 | 99国产精品久久|