2011-03-03 60 views
16

我想在ASP.NET MVC中實現Gmail風格的拖放文件上傳。文件API文件上傳 - 讀取ASP.NET MVC中的XMLHttpRequest

我一直在關注這篇文章:http://robertnyman.com/html5/fileapi-upload/fileapi-upload.html並希望將上傳的文件發佈到MVC控制器操作。

要做到這一點,我修改了樣品中的鏈接的JavaScript腳本,指向我的控制器操作:

xhr.open("post", "/home/UploadFiles", true); 

這裏是我的控制器操作:

[HttpPost] 
public virtual string UploadFiles(object obj) 
{ 
    var length = Request.ContentLength; 
    var bytes = new byte[length]; 
    Request.InputStream.Read(bytes, 0, length); 
    // var bytes has byte content here. what do do next? 

    return "Files uploaded!"; 
} 

我設置一個斷點,當我上傳一個文件時,斷點會被打 - 這很好。但是,如何從上傳的(javascript)XMLHttpRequest對象中提取數據?我不認爲它在HttpRequest中 - 它是參數嗎?如果是這樣,我應該期待什麼類型&如何提取字節數組並從中提取上傳的文件信息?

(我使用的瀏覽器 - 我知道這並不在IE瀏覽器)

任何建議,將不勝感激!

回答

38

想通了。下面是C#代碼:

[HttpPost] 
    public virtual string UploadFiles(object obj) 
    { 
     var length = Request.ContentLength; 
     var bytes = new byte[length]; 
     Request.InputStream.Read(bytes, 0, length); 
     // bytes has byte content here. what do do next? 

     var fileName = Request.Headers["X-File-Name"]; 
     var fileSize = Request.Headers["X-File-Size"]; 
     var fileType = Request.Headers["X-File-Type"]; 

     var saveToFileLoc = string.Format("{0}\\{1}", 
             Server.MapPath("/Files"), 
             fileName); 

     // save the file. 
     var fileStream = new FileStream(saveToFileLoc, FileMode.Create, FileAccess.ReadWrite); 
     fileStream.Write(bytes, 0, length); 
     fileStream.Close(); 

     return string.Format("{0} bytes uploaded", bytes.Length); 
    } 

而這裏的Javascript代碼:

<script type="text/javascript"> 
(function() 
{ 
    var filesUpload = document.getElementById("files-upload"); 
    var dropArea = document.getElementById("drop-area"); 
    var fileList = document.getElementById("file-list"); 

    function uploadFile(file) 
    { 
     var li = document.createElement("li"); 
     var progressBarContainer = document.createElement("div"); 
     var progressBar = document.createElement("div"); 

     progressBarContainer.className = "progress-bar-container"; 
     progressBar.className = "progress-bar"; 
     progressBarContainer.appendChild(progressBar); 
     li.appendChild(progressBarContainer); 

     // Uploading - for Firefox, Google Chrome and Safari 
     var xhr = new XMLHttpRequest(); 

     // Update progress bar 
     xhr.upload.addEventListener("progress", function (evt) 
     { 
      if (evt.lengthComputable) 
      { 
       progressBar.style.width = (evt.loaded/evt.total) * 100 + "%"; 
      } 
     }, false); 

     // File uploaded 
     xhr.addEventListener("load", function() 
     { 
      progressBarContainer.className += " uploaded"; 
      progressBar.innerHTML = "Uploaded!"; 
     }, false); 

     xhr.open("post", "/home/UploadFile", true); 

     // Set appropriate headers 
     xhr.setRequestHeader("Content-Type", "multipart/form-data"); 
     xhr.setRequestHeader("X-File-Name", file.fileName); 
     xhr.setRequestHeader("X-File-Size", file.fileSize); 
     xhr.setRequestHeader("X-File-Type", file.type); 

     // Send the file 
     xhr.send(file); 

     // Present file info and append it to the list of files 
     var div = document.createElement("div"); 
     li.appendChild(div); 
     var fileInfo = "<div><strong>Name:</strong> " + file.name + "</div>"; 
     fileInfo += "<div><strong>Size:</strong> " + parseInt(file.size/1024, 10) + " kb</div>"; 
     fileInfo += "<div><strong>Type:</strong> " + file.type + "</div>"; 
     div.innerHTML = fileInfo; 

     // insert at beginning of list. 
     fileList.insertBefore(li, fileList.firstChild); 

     // or insert at end of list. 
     //fileList.appendChild(li); 
    } 

    function traverseFiles(files) 
    { 
     if (typeof files !== "undefined") 
     { 
      for (var i = 0, l = files.length; i < l; i++) 
      { 
       uploadFile(files[i]); 
      } 
     } 
     else 
     { 
      fileList.innerHTML = "No support for the File API in this web browser"; 
     } 
    } 

    filesUpload.addEventListener("change", function() 
    { 
     traverseFiles(this.files); 
    }, false); 

    dropArea.addEventListener("dragleave", function (evt) 
    { 
     var target = evt.target; 

     if (target && target === dropArea) 
     { 
      this.className = ""; 
     } 
     evt.preventDefault(); 
     evt.stopPropagation(); 
    }, false); 

    dropArea.addEventListener("dragenter", function (evt) 
    { 
     this.className = "over"; 
     evt.preventDefault(); 
     evt.stopPropagation(); 
    }, false); 

    dropArea.addEventListener("dragover", function (evt) 
    { 
     evt.preventDefault(); 
     evt.stopPropagation(); 
    }, false); 

    dropArea.addEventListener("drop", function (evt) 
    { 
     //document.getElementById("file-list").innerHTML = ""; 

     traverseFiles(evt.dataTransfer.files); 
     this.className = ""; 
     evt.preventDefault(); 
     evt.stopPropagation(); 
    }, false); 
})(); 
+0

我假設保存請求byte []將是一個XML文檔,而不是文件內容本身。結果比我預想的要簡單得多。 – 2011-03-03 15:03:11

+2

+1用於保存我的傍晚 – Valentin 2011-03-14 15:51:33

+0

不錯的代碼。這是工作,WooHoo ~~~ – HorseKing 2012-01-23 19:37:59

0

這是偉大的。稍微改進:

xhr.setRequestHeader("X-File-Name", file.name); 
xhr.setRequestHeader("X-File-Size", file.size);