天天看點

批量上傳js插件的使用案例

檔案上傳無疑是web應用中一個非常常用的功能,不管是PHP、jsp還是aspx、mvc等都會需要檔案上傳,但是衆所周知當使用自帶的檔案上傳功能時總會出現頁面重新整理的情況。當然現在有了html5這個好東西,我們可以調用它的新的api來做檔案的異步上傳。但是非常可惜,這個新的api并非每個浏覽器都支援。

如果你會flash這當然很好,你可以自己寫一個flash的上傳插件來支援上傳,不過本文不會對flash這個技術做任何的讨論。

好了言歸正傳,我們還是來讨論下隻使用js的情況下如何才能異步無重新整理的上傳檔案,首先估計大家會想到ajax,不過很不幸在目前沒有html5支援的浏覽器中使用ajax上傳檔案是不行的,而在國内使用支援html5浏覽器的使用者還不是絕大多數,那麼這種方案隻能放棄。

還有沒有辦法喃?當然有那就是使用iframe,看下面的html代碼:

[html] 

view

plain

copy https://code.csdn.net/snippets/74821 https://code.csdn.net/snippets/74821/fork

  1. <body>  
  2.     <form action="WebForm1.aspx" target="dynamic_creation_upload_iframe" method="POST" enctype="multipart/form-data">  
  3.     <input type="file" name="upload1" />  
  4.     </form>  
  5.     <iframe name="dynamic_creation_upload_iframe"></iframe>  
  6. </body>  

這是一段我們經常使用的上傳代碼,如果你使用的是asp.net控件,它最終也會生成這樣的html代碼,其中的enctype就是指定需要上傳檔案的屬性(action和method屬性就不需要解釋了吧)。不過在這個form上我添加了一個target屬性,并且指定了它的值為下面那個iframe的name屬性的值,這是為什麼?其實很簡單,就是當你送出這個form時(這樣就開始了檔案上傳),等服務端接收到檔案并儲存成功響應用戶端時,将響應的内容直接在這個iframe中重新整理,這樣這個iframe就起到了一個隔離的作用,此時我們隻需要在這個iframe上綁定一個onload事件,那麼當它重新整理時這個事件将被觸發,呵呵

我們就可以在這個load事件中做事情了(例如:在load的時候擷取服務端響應的結果,進而得知上傳是否成功)

有了這個思路事情就好辦了,以下是根據這種思路建構的一個js插件

[javascript] 

https://code.csdn.net/snippets/74821 https://code.csdn.net/snippets/74821/fork
  1. /* 
  2.     無重新整理異步上傳插件 
  3.     2013-10-16 Devotion Created 
  4. */  
  5. (function ($) {  
  6.     var defaultSettings = {  
  7.         url: "",                                 //上傳位址  
  8.         buttonFeature: true,                    //true:點選按鈕時僅選擇檔案; false:選擇完檔案後立即上傳  
  9.         fileSuffixs: ["jpg", "png"],             //允許上傳的檔案字尾名清單  
  10.         errorText: "不能上傳字尾為 {0} 的檔案!", //錯誤提示文本,其中{0}将會被上傳檔案的字尾名替換  
  11.         onCheckUpload: function (text) { //上傳時檢查檔案字尾名不包含在fileSuffixs屬性中時觸發的回調函數,(text為錯誤提示文本)  
  12.             alert(text);  
  13.         },  
  14.         onComplete: function (msg) { //上傳完成後的回調函數[不管成功或失敗,它都将被觸發](msg為服務端的傳回字元串)  
  15.         onChosen: function (file, obj) { //選擇檔案後的回調函數,(file為選中檔案的本地路徑;obj為目前的上傳控件執行個體)  
  16.             //alert(file);  
  17.         maximumFilesUpload: 5,//最大檔案上傳數(當此屬性大于1時,buttonFeature屬性隻能為true)  
  18.         onSubmitHandle: function (uploadFileNumber) { //送出上傳時的回調函數,uploadFileNumber為目前上傳的檔案數量  
  19.             //在此回調中傳回false上傳送出将被阻止  
  20.             return true;  
  21.         onSameFilesHandle: function (file) { //當重複選擇相同的檔案時觸發  
  22.             //在此回調中傳回false目前選擇的檔案将從上傳隊列中取消  
  23.         perviewImageElementId: "",//用于預覽上傳圖檔的元素id(請傳入一個div元素的id)  
  24.         perviewImgStyle: null//用于設定圖檔預覽時的樣式(可不設定,在不設定的情況下多檔案上傳時隻能顯示一張圖檔),如{ width: '100px', height: '100px', border: '1px solid #ebebeb' }  
  25.     };  
  26.     $.fn.uploadFile = function (settings) {  
  27.         settings = $.extend({}, defaultSettings, settings || {});  
  28.         if (settings.perviewImageElementId) {  
  29.             //設定圖檔預覽元素的必須樣式  
  30.             if (!settings.perviewImgStyle) {  
  31.                 var perviewImg = document.getElementById(settings.perviewImageElementId);  
  32.                 perviewImg.style.overflow = "hidden";  
  33.             }  
  34.         }  
  35.         return this.each(function () {  
  36.             var self = $(this);  
  37.             var upload = new UploadAssist(settings);  
  38.             upload.createIframe(this);  
  39.             //綁定目前按鈕點選事件  
  40.             self.bind("click", function (e) {  
  41.                 upload.chooseFile();  
  42.             });  
  43.             //将上傳輔助類的執行個體,存放到目前對象中,友善外部擷取  
  44.             self.data("uploadFileData", upload);  
  45.             //建立的iframe中的那個iframe,它的事件需要延遲綁定  
  46.             window.setTimeout(function () {  
  47.                 //為建立的iframe内部的iframe綁定load事件  
  48.                 $(upload.getIframeContentDocument().body.lastChild).on("load", function () {  
  49.                     var dcmt = upload.getInsideIframeContentDocument();  
  50.                     if (dcmt.body.innerHTML) {  
  51.                         if (settings.onComplete) {  
  52.                             settings.onComplete(dcmt.body.innerHTML);  
  53.                         }  
  54.                         dcmt.body.innerHTML = "";  
  55.                     }  
  56.                 });  
  57.             }, 100);  
  58.         });  
  59. })(jQuery);  
  60. //上傳輔助類  
  61. function UploadAssist(settings) {  
  62.     //儲存設定  
  63.     this.settings = settings;  
  64.     //已選擇檔案的路徑集合  
  65.     this.choseFilePath = [];  
  66.     //建立的iframe唯一名稱  
  67.     this.iframeName = "upload" + this.getInputFileName();  
  68.     return this;  
  69. }  
  70. UploadAssist.prototype = {  
  71.     //輔助類構造器  
  72.     constructor: UploadAssist,  
  73.     //建立iframe  
  74.     createIframe: function (/*插件中指定的dom對象*/elem) {  
  75.         var html = "<html>"  
  76.                 + "<head>"  
  77.                 + "<title>upload</title>"  
  78.                 + "<script>"  
  79.                 + "function getDCMT(){return window.frames['dynamic_creation_upload_iframe'].document;}"  
  80.                 + "</" + "script>"  
  81.                 + "</head>"  
  82.                 + "<body>"  
  83.                 + "<form method='post' target='dynamic_creation_upload_iframe' enctype='multipart/form-data' action='" + this.settings.url + "'>"  
  84.                 + "</form>"  
  85.                 + "<iframe name='dynamic_creation_upload_iframe'></iframe>"  
  86.                 + "</body>"  
  87.                 + "</html>";  
  88.         this.iframe = $("<iframe name='" + this.iframeName + "'></iframe>")[0];  
  89.         this.iframe.style.width = "0px";  
  90.         this.iframe.style.height = "0px";  
  91.         this.iframe.style.border = "0px solid #fff";  
  92.         this.iframe.style.margin = "0px";  
  93.         elem.parentNode.insertBefore(this.iframe, elem);  
  94.         var iframeDocument = this.getIframeContentDocument();  
  95.         iframeDocument.write(html);  
  96.     },  
  97.     //擷取上傳控件名稱  
  98.     getInputFileName: function () {  
  99.         return (new Date()).valueOf();  
  100.     //建立上傳控件到建立的iframe中  
  101.     createInputFile: function () {  
  102.         var that = this;  
  103.         var dcmt = this.getIframeContentDocument();  
  104.         var input = dcmt.createElement("input");  
  105.         input.type = "file";  
  106.         input.setAttribute("name", "input" + this.getInputFileName());  
  107.         input.onchange = function () {  
  108.             var fileSuf = this.value.substring(this.value.lastIndexOf(".") + 1);  
  109.             //檢查是否為允許上傳的檔案  
  110.             if (!that.checkFileIsUpload(fileSuf, that.settings.fileSuffixs)) {  
  111.                 that.settings.onCheckUpload(that.settings.errorText.replace("{0}", fileSuf));  
  112.                 return;  
  113.             //選中後的回調  
  114.             that.settings.onChosen(this.value, this);  
  115.             if (that.checkFileIsExist(this.value)) {  
  116.                 //儲存已經選擇的檔案路徑  
  117.                 that.choseFilePath.push({ "name": this.name, "value": this.value });  
  118.                 var status = that.settings.onSameFilesHandle(this.value);  
  119.                 if (typeof status === "boolean" && !status) {  
  120.                     that.removeFile(this.value);  
  121.                     return;  
  122.                 }  
  123.             } else {  
  124.             //是否開啟了圖檔預覽  
  125.             if (that.settings.perviewImageElementId) {  
  126.                 if (!that.settings.perviewImgStyle) {  
  127.                     perviewImage.beginPerview(this, that.settings.perviewImageElementId);  
  128.                 } else {  
  129.                     var ul = perviewImage.getPerviewRegion(that.settings.perviewImageElementId);  
  130.                     var main = perviewImage.createPreviewElement(this.value);  
  131.                     var li = document.createElement("li");  
  132.                     //li.style.float = "left";  
  133.                     if ($.browser.msie) {  
  134.                         li.style.styleFloat = "left";  
  135.                     else {  
  136.                         li.style.cssFloat = "left";  
  137.                     li.style.margin = "5px";  
  138.                     li.appendChild(main);  
  139.                     ul.appendChild(li);  
  140.                     var div = $(main).children("div").get(0);  
  141.                     $(main).children("img").hover(function () {  
  142.                         this.src = perviewImage.closeImg.after;  
  143.                     }, function () {  
  144.                         this.src = perviewImage.closeImg.before;  
  145.                     }).click(function () {  
  146.                         that.removeFile($(this).attr("filepath"));  
  147.                         $(this).parents("li").remove("li");  
  148.                     });  
  149.                     perviewImage.beginPerview(this, div, dcmt);  
  150.             if (!that.settings.buttonFeature) {  
  151.                 that.submitUpload();  
  152.         };  
  153.         dcmt.forms[0].appendChild(input);  
  154.         return input;  
  155.     //擷取建立的iframe中的document對象  
  156.     getIframeContentDocument: function () {  
  157.         return this.iframe.contentDocument || this.iframe.contentWindow.document;  
  158.     //擷取建立的iframe所在的window對象  
  159.     getIframeWindow: function () {  
  160.         return this.iframe.contentWindow || this.iframe.contentDocument.parentWindow;  
  161.     //擷取建立的iframe内部iframe的document對象  
  162.     getInsideIframeContentDocument: function () {  
  163.         return this.getIframeWindow().getDCMT();  
  164.     //擷取上傳input控件  
  165.     getUploadInput: function () {  
  166.         var inputs = this.getIframeContentDocument().getElementsByTagName("input");  
  167.         var len = inputs.length;  
  168.         if (len > 0) {  
  169.             if (!inputs[len - 1].value) {  
  170.                 return inputs[len - 1];  
  171.                 return this.createInputFile();  
  172.         return this.createInputFile();  
  173.     //forEach疊代函數  
  174.     forEach: function (/*數組*/arr, /*代理函數*/fn) {  
  175.         var len = arr.length;  
  176.         for (var i = 0; i < len; i++) {  
  177.             var tmp = arr[i];  
  178.             if (fn.call(tmp, i, tmp) == false) {  
  179.                 break;  
  180.     //送出上傳  
  181.     submitUpload: function () {  
  182.         var status = this.settings.onSubmitHandle(this.choseFilePath.length);  
  183.         if (typeof status === "boolean") {  
  184.             if (!status) {  
  185.         this.clearedNotChooseFile();  
  186.         dcmt.forms[0].submit();  
  187.     //檢查檔案是否可以上傳  
  188.     checkFileIsUpload: function (fileSuf, suffixs) {  
  189.         var status = false;  
  190.         this.forEach(suffixs, function (i, n) {  
  191.             if (fileSuf.toLowerCase() === n.toLowerCase()) {  
  192.                 status = true;  
  193.                 return false;  
  194.         return status;  
  195.     //檢查上傳的檔案是否已經存在上傳隊列中  
  196.     checkFileIsExist: function (/*目前上傳的檔案*/file) {  
  197.         this.forEach(this.choseFilePath, function (i, n) {  
  198.             if (n.value == file) {  
  199.     //清除未選擇檔案的上傳控件  
  200.     clearedNotChooseFile: function () {  
  201.         var files = this.getIframeContentDocument().getElementsByTagName("input");  
  202.         this.forEach(files, function (i, n) {  
  203.             if (!n.value) {  
  204.                 n.parentNode.removeChild(n);  
  205.     //将指定上傳的檔案從上傳隊列中删除  
  206.     removeFile: function (file) {  
  207.                 that.forEach(files, function (j, m) {  
  208.                     if (m.name == n.name) {  
  209.                         m.parentNode.removeChild(m);  
  210.                         return false;  
  211.                 that.choseFilePath.splice(i, 1);  
  212.     //清空上傳隊列  
  213.     clearUploadQueue: function () {  
  214.         this.choseFilePath.length = 0;  
  215.         this.getIframeContentDocument().forms[0].innerHTML = "";  
  216.     //選擇上傳檔案  
  217.     chooseFile: function () {  
  218.         var uploadfile;  
  219.         if (this.choseFilePath.length == this.settings.maximumFilesUpload) {  
  220.             if (this.settings.maximumFilesUpload <= 1) {  
  221.                 this.choseFilePath.length = 0;  
  222.                 var files = this.getIframeContentDocument().getElementsByTagName("input");  
  223.                 if (!files.length) {  
  224.                     uploadfile = this.getUploadInput();  
  225.                     $(uploadfile).click();  
  226.                     uploadfile = files[0];  
  227.         uploadfile = this.getUploadInput();  
  228.         $(uploadfile).click();  
  229.     }  
  230. };  
  231. //圖檔預覽操作  
  232. var perviewImage = {  
  233.     timers: [],  
  234.     closeImg: {  
  235.         before: "",  
  236.         after: ""  
  237.     //擷取預覽元素  
  238.     getElementObject: function (elem) {  
  239.         if (elem.nodeType && elem.nodeType === 1) {  
  240.             return elem;  
  241.         } else {  
  242.             return document.getElementById(elem);  
  243.     //開始圖檔預覽  
  244.     beginPerview: function (/*檔案上傳控件執行個體*/file, /*需要顯示的元素id或元素執行個體*/perviewElemId,dcmt) {  
  245.         for (var t = 0; t < this.timers.length; t++) {  
  246.             window.clearInterval(this.timers[t]);  
  247.         this.timers.length = 0;  
  248.         var preview_div = this.getElementObject(perviewElemId);  
  249.         var MAXWIDTH = preview_div.clientWidth;  
  250.         var MAXHEIGHT = preview_div.clientHeight;  
  251.         if (file.files && file.files[0]) { //此處為Firefox,Chrome以及IE10的操作  
  252.             preview_div.innerHTML = "";  
  253.             var img = document.createElement("img");  
  254.             preview_div.appendChild(img);  
  255.             img.style.visibility = "hidden";  
  256.             img.onload = function () {  
  257.                 var rect = perviewImage.clacImgZoomParam(MAXWIDTH, MAXHEIGHT, img.offsetWidth, img.offsetHeight);  
  258.                 img.style.width = rect.width + 'px';  
  259.                 img.style.height = rect.height + 'px';  
  260.                 img.style.marginLeft = rect.left + 'px';  
  261.                 img.style.marginTop = rect.top + 'px';  
  262.                 img.style.visibility = "visible";  
  263.             var reader = new FileReader();  
  264.             reader.onload = function (evt) {  
  265.                 img.src = evt.target.result;  
  266.             reader.readAsDataURL(file.files[0]);  
  267.         else {//此處為IE6,7,8,9的操作  
  268.             file.select();  
  269.             var src = dcmt.selection.createRange().text;  
  270.             var div_sFilter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod='scale',src='" + src + "')";  
  271.             var img_sFilter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(sizingMethod='image',src='" + src + "')";  
  272.             var img = document.createElement("div");  
  273.             img.style.filter = img_sFilter;  
  274.             img.style.width = "100%";  
  275.             img.style.height = "100%";  
  276.             function setImageDisplay() {  
  277.                 preview_div.innerHTML = "";  
  278.                 var div = document.createElement("div");  
  279.                 div.style.width = rect.width + 'px';  
  280.                 div.style.height = rect.height + 'px';  
  281.                 div.style.marginLeft = rect.left + 'px';  
  282.                 div.style.marginTop = rect.top + 'px';  
  283.                 div.style.filter = div_sFilter;  
  284.                 preview_div.appendChild(div);  
  285.             //圖檔加載計數  
  286.             var tally = 0;  
  287.             var timer = window.setInterval(function () {  
  288.                 if (img.offsetHeight != MAXHEIGHT) {  
  289.                     window.clearInterval(timer);  
  290.                     setImageDisplay()  
  291.                     tally++;  
  292.                 //如果超過兩秒鐘圖檔還不能加載,就停止目前的輪詢  
  293.                 if (tally > 20) {  
  294.             this.timers.push(timer);  
  295.     //按比例縮放圖檔  
  296.     clacImgZoomParam: function (maxWidth, maxHeight, width, height) {  
  297.         var param = { width: width, height: height };  
  298.         if (width > maxWidth || height > maxHeight) {  
  299.             var rateWidth = width / maxWidth;  
  300.             var rateHeight = height / maxHeight;  
  301.             if (rateWidth > rateHeight) {  
  302.                 param.width = maxWidth;  
  303.                 param.height = Math.round(height / rateWidth);  
  304.                 param.width = Math.round(width / rateHeight);  
  305.                 param.height = maxHeight;  
  306.         param.left = Math.round((maxWidth - param.width) / 2);  
  307.         param.top = Math.round((maxHeight - param.height) / 2);  
  308.         return param;  
  309.     //建立預覽元素  
  310.     createPreviewElement: function (/*上傳時的檔案名*/file, /*預覽時的樣式*/style) {  
  311.         style = style || { width: '100px', height: '100px', border: '1px solid #ebebeb' };  
  312.         var img = document.createElement("div");  
  313.         img.title = file;  
  314.         img.style.overflow = "hidden";  
  315.         for (var s in style) {  
  316.             img.style[s] = style[s];  
  317.         var text = document.createElement("div");  
  318.         text.style.width = style.width;  
  319.         text.style.overflow = "hidden";  
  320.         text.style.textOverflow = "ellipsis";  
  321.         text.style.whiteSpace = "nowrap";  
  322.         text.innerHTML = file;  
  323.         var top = 0 - window.parseInt(style.width) - 15;  
  324.         var right = 0 - window.parseInt(style.width) + 14;  
  325.         var close = document.createElement("img");  
  326.         close.setAttribute("filepath", file);  
  327.         close.src = this.closeImg.before;  
  328.         close.style.position = "relative";  
  329.         close.style.top = top + "px";  
  330.         close.style.right = right + "px";  
  331.         close.style.cursor = "pointer";  
  332.         var main = document.createElement("div");  
  333.         main.appendChild(img);  
  334.         main.appendChild(text);  
  335.         main.appendChild(close);  
  336.         return main;  
  337.     //擷取預覽區域  
  338.     getPerviewRegion: function (elem) {  
  339.         var perview = $(this.getElementObject(elem));  
  340.         if (!perview.find("ul").length) {  
  341.             var ul = document.createElement("ul");  
  342.             ul.style.listStyleType = "none";  
  343.             ul.style.margin = "0px";  
  344.             ul.style.padding = "0px";  
  345.             var div = document.createElement("div");  
  346.             div.style.clear = "both";  
  347.             perview.append(ul).append(div);  
  348.             return ul;  
  349.             return perview.children("ul").get(0);  

看看這個插件中的那個createIframe方法,對它做一點解釋

https://code.csdn.net/snippets/74821 https://code.csdn.net/snippets/74821/fork
  1. //建立iframe  
  2.         this.iframe.style.display = "none";  

大家應該都看到了這個方法中有一個html變量,它儲存的其實就是文章開頭的那段html。在這段html中我添加了一個function名為getDCMT的函數,這是為了擷取名為dynamic_creation_upload_iframe的iframe所在的document對象。在form中我也去掉了檔案上傳的input控件,這是因為我将動态建立input控件到這個form中,并且我會将這段html使用js的方式把它添加到一個動态建立的iframe中。為什麼要這樣呢?呵呵

我想聰明的你一定會明白的!

好了插件做好了,我們如何來使用呢?

首先 将那段插件js代碼儲存為notRefreshFilesUpload.js的檔案,友善在頁面上的引用,然後建構包含如下結構的page

https://code.csdn.net/snippets/74821 https://code.csdn.net/snippets/74821/fork
  1. <!DOCTYPE html>  
  2. <html xmlns="http://www.w3.org/1999/xhtml">  
  3. <head>  
  4. <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>  
  5.     <title>files upload</title>  
  6.     <script src="Scripts/jquery-1.7.1.min.js"></script>  
  7.     <script src="Scripts/notRefreshFilesUpload.js"></script>  
  8.     <script>  
  9.         $(function () {  
  10.             var btn = $("#Button1");  
  11.             btn.uploadFile({  
  12.                 url: "WebForm1.aspx",  
  13.                 fileSuffixs: ["jpg", "png", "gif"],  
  14.                 buttonFeature: true,  
  15.                 errorText: "{0}",  
  16.                 maximumFilesUpload: 5,//最大檔案上傳數  
  17.                 onComplete: function (msg) {  
  18.                     $("#testdiv").html(msg);  
  19.                 },  
  20.                 perviewImageElementId: "fileList", //設定預覽圖檔的元素id  
  21.                 perviewImgStyle: { width: '100px', height: '100px', border: '1px solid #ebebeb' }//設定預覽圖檔的樣式  
  22.             var upload = btn.data("uploadFileData");  
  23.             $("#files").click(function () {  
  24.                 upload.submitUpload();  
  25.     </script>  
  26. </head>  
  27.     <body>  
  28.         <div style="width: 400px; height: 300px; float:left">  
  29.             <input id="Button1" type="button" value="選擇檔案" />  
  30.             <input id="files" type="button" value="上傳" />  
  31.             <div id="fileList" style="margin-top: 10px; padding-top:10px; border-top:1px solid #C0C0C0;font-size: 13px; width:400px">  
  32.             </div>  
  33.         </div>  
  34.         <div id="testdiv"></div>  
  35.     </body>  
  36. </html>  

上面的代碼中已經包含了圖檔預覽的功能,使用非常簡單我就不多言了,隻需要給定用于顯示圖檔的元素id即可,一般用div作為圖檔預覽的元素。

這是服務端的方法,在這裡我使用了aspx作為服務端的接收方式,當然你可以換成其他任何語言或形式作為服務端的處理方案(可以是php、jsp、mvc等等)

[csharp] 

https://code.csdn.net/snippets/74821 https://code.csdn.net/snippets/74821/fork
  1. public partial class WebForm1 : System.Web.UI.Page  
  2. {  
  3.     protected void Page_Load(object sender, EventArgs e)  
  4.     {  
  5.         List<string> filenames = new List<string>();  
  6.         HttpFileCollection files = Request.Files;  
  7.         for (int i = 0; i < files.Count; i++)  
  8.         {  
  9.             filenames.Add(files[i].FileName);  
  10.         Response.Write(string.Join("___", filenames));  
  11.         Response.Flush();  
  12.         Response.End();  

服務端代碼為多檔案上傳處理的方式,呵呵 也就是說這個插件也是支援多檔案上傳的,在服務端的代碼中我僅僅傳回了上傳檔案的名稱作為對用戶端的響應,當然你可以傳回任何你希望的形式,你隻需要在用戶端用js做相應處理即可(即在插件的complete這個回調中處理響應,它其中的msg回調參數将把服務端的響應結果回傳給你)。

好了 一個相容各種浏覽器,并且支援圖檔預覽和無重新整理異步上傳的純js插件就搞定了。上個圖看看效果