1.下載webuploader;下載地址:https://github.com/fex-team/webuploader/releases/download/0.1.5/webuploader-0.1.5.zip
2.創建web空項目,解壓后將如下文件引入項目中:

3.將以上文件引入相應頁面中,並引入jQuery:
<!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"> <title>WebUploader演示</title> <link href="Styles/webuploader.css" rel="stylesheet" type="text/css" /> <link href="Styles/style.css" rel="stylesheet" type="text/css" /> <script src="Scripts/jquery.js" type="text/javascript"></script> <script src="Scripts/webuploader.js" type="text/javascript"></script> <script src="Scripts/upload.js" type="text/javascript"></script> </head> <body> <div id="wrapper"> <div id="container"> <!--頭部,相冊選擇和格式選擇--> <div id="uploader"> <div class="queueList"> <div id="dndArea" class="placeholder"> <div id="filePicker"></div> </div> </div> <div class="statusBar" style="display:none;"> <div class="progress"> <span class="text">0%</span> <span class="percentage"></span> </div><div class="info"></div> <div class="btns"> <div id="filePicker2"></div><div class="uploadBtn">開始上傳</div> </div> </div> </div> </div> </div> </body> </html>
4.upload.js(實例化、上傳、合並)代碼如下:
(function ($) { // 當domReady的時候開始初始化 $(function () { var $wrap = $('#uploader'), // 圖片容器 $queue = $('<ul class="filelist"></ul>') .appendTo($wrap.find('.queueList')), // 狀態欄,包括進度和控制按鈕 $statusBar = $wrap.find('.statusBar'), // 文件總體選擇信息。 $info = $statusBar.find('.info'), // 上傳按鈕 $upload = $wrap.find('.uploadBtn'), // 沒選擇文件之前的內容。 $placeHolder = $wrap.find('.placeholder'), $progress = $statusBar.find('.progress').hide(), // 添加的文件數量 fileCount = 0, // 添加的文件總大小 fileSize = 0, // 優化retina, 在retina下這個值是2 ratio = window.devicePixelRatio || 1, // 縮略圖大小 thumbnailWidth = 110 * ratio, thumbnailHeight = 110 * ratio, // 可能有pedding, ready, uploading, confirm, done. state = 'pedding', // 所有文件的進度信息,key為file id percentages = {}, // 判斷瀏覽器是否支持圖片的base64 isSupportBase64 = (function () { var data = new Image(); var support = true; data.onload = data.onerror = function () { if (this.width != 1 || this.height != 1) { support = false; } } data.src = ""; return support; })(), // 檢測是否已經安裝flash,檢測flash的版本 flashVersion = (function () { var version; try { version = navigator.plugins['Shockwave Flash']; version = version.description; } catch (ex) { try { version = new ActiveXObject('ShockwaveFlash.ShockwaveFlash') .GetVariable('$version'); } catch (ex2) { version = '0.0'; } } version = version.match(/\d+/g); return parseFloat(version[0] + '.' + version[1], 10); })(), supportTransition = (function () { var s = document.createElement('p').style, r = 'transition' in s || 'WebkitTransition' in s || 'MozTransition' in s || 'msTransition' in s || 'OTransition' in s; s = null; return r; })(), // WebUploader實例 uploader, GUID = WebUploader.Base.guid(); //當前頁面是生成的GUID作為標示 if (!WebUploader.Uploader.support('flash') && WebUploader.browser.ie) { // flash 安裝了但是版本過低。 if (flashVersion) { (function (container) { window['expressinstallcallback'] = function (state) { switch (state) { case 'Download.Cancelled': alert('您取消了更新!') break; case 'Download.Failed': alert('安裝失敗') break; default: alert('安裝已成功,請刷新!'); break; } delete window['expressinstallcallback']; }; var swf = 'Scripts/expressInstall.swf'; // insert flash object var html = '<object type="application/' + 'x-shockwave-flash" data="' + swf + '" '; if (WebUploader.browser.ie) { html += 'classid="clsid:d27cdb6e-ae6d-11cf-96b8-444553540000" '; } html += 'width="100%" height="100%" style="outline:0">' + '<param name="movie" value="' + swf + '" />' + '<param name="wmode" value="transparent" />' + '<param name="allowscriptaccess" value="always" />' + '</object>'; container.html(html); })($wrap); // 壓根就沒有安轉。 } else { $wrap.html('<a href="http://www.adobe.com/go/getflashplayer" target="_blank" border="0"><img alt="get flash player" src="http://www.adobe.com/macromedia/style_guide/images/160x41_Get_Flash_Player.jpg" /></a>'); } return; } else if (!WebUploader.Uploader.support()) { alert('Web Uploader 不支持您的瀏覽器!'); return; } // 實例化 uploader = WebUploader.create({ pick: { id: '#filePicker', label: '點擊選擇文件' }, formData: { uid: 123 }, dnd: '#dndArea', paste: '#uploader', swf: 'Scripts/Uploader.swf', chunked: true, //分片處理大文件 chunkSize: 2 * 1024 * 1024, server: 'fileupload.ashx', disableGlobalDnd: true, threads: 1, //上傳並發數 //由於Http的無狀態特征,在往服務器發送數據過程傳遞一個進入當前頁面是生成的GUID作為標示 formData: { guid: GUID }, fileNumLimit: 2, //上傳文件的數量限制 compress: false, //圖片在上傳前不進行壓縮 fileSizeLimit: 6*1024* 1024 * 1024, // 所有文件總大小限制 6G fileSingleSizeLimit: 5*1024* 1024 * 1024 // 單個文件大小限制 5 G }); // 拖拽時不接受 js, txt 文件。 uploader.on('dndAccept', function (items) { var denied = false, len = items.length, i = 0, // 修改js類型 unAllowed = 'text/plain;application/javascript '; for (; i < len; i++) { // 如果在列表里面 if (~unAllowed.indexOf(items[i].type)) { denied = true; break; } } return !denied; }); uploader.on('dialogOpen', function () { console.log('here'); }); // uploader.on('filesQueued', function() { // uploader.sort(function( a, b ) { // if ( a.name < b.name ) // return -1; // if ( a.name > b.name ) // return 1; // return 0; // }); // }); // 添加“添加文件”的按鈕, uploader.addButton({ id: '#filePicker2', label: '繼續添加' }); uploader.on('ready', function () { window.uploader = uploader; }); // 當有文件添加進來時執行,負責view的創建 function addFile(file) { var $li = $('<li id="' + file.id + '">' + '<p class="title">' + file.name + '</p>' + '<p class="imgWrap"></p>' + '<p class="progress"><span></span></p>' + '</li>'), $btns = $('<div class="file-panel">' + '<span class="cancel">刪除</span>' + '<span class="rotateRight">向右旋轉</span>' + '<span class="rotateLeft">向左旋轉</span></div>').appendTo($li), $prgress = $li.find('p.progress span'), $wrap = $li.find('p.imgWrap'), $info = $('<p class="error"></p>'), showError = function (code) { switch (code) { case 'exceed_size': text = '文件大小超出'; break; case 'interrupt': text = '上傳暫停'; break; default: text = '上傳失敗,請重試'; break; } $info.text(text).appendTo($li); }; if (file.getStatus() === 'invalid') { showError(file.statusText); } else { // @todo lazyload $wrap.text('預覽中'); uploader.makeThumb(file, function (error, src) { var img; if (error) { $wrap.text('不能預覽'); return; } if (isSupportBase64) { img = $('<img src="' + src + '">'); $wrap.empty().append(img); } else { $.ajax('preview.ashx', { method: 'POST', data: src, dataType: 'json' }).done(function (response) { if (response.result) { img = $('<img src="' + response.result + '">'); $wrap.empty().append(img); } else { $wrap.text("預覽出錯"); } }); } }, thumbnailWidth, thumbnailHeight); percentages[file.id] = [file.size, 0]; file.rotation = 0; } file.on('statuschange', function (cur, prev) { if (prev === 'progress') { $prgress.hide().width(0); } else if (prev === 'queued') { $li.off('mouseenter mouseleave'); $btns.remove(); } // 成功 if (cur === 'error' || cur === 'invalid') { console.log(file.statusText); showError(file.statusText); percentages[file.id][1] = 1; } else if (cur === 'interrupt') { showError('interrupt'); } else if (cur === 'queued') { $info.remove(); $prgress.css('display', 'block'); percentages[file.id][1] = 0; } else if (cur === 'progress') { $info.remove(); $prgress.css('display', 'block'); } else if (cur === 'complete') { $prgress.hide().width(0); $li.append('<span class="success"></span>'); } $li.removeClass('state-' + prev).addClass('state-' + cur); }); $li.on('mouseenter', function () { $btns.stop().animate({ height: 30 }); }); $li.on('mouseleave', function () { $btns.stop().animate({ height: 0 }); }); $btns.on('click', 'span', function () { var index = $(this).index(), deg; switch (index) { case 0: uploader.removeFile(file); return; case 1: file.rotation += 90; break; case 2: file.rotation -= 90; break; } if (supportTransition) { deg = 'rotate(' + file.rotation + 'deg)'; $wrap.css({ '-webkit-transform': deg, '-mos-transform': deg, '-o-transform': deg, 'transform': deg }); } else { $wrap.css('filter', 'progid:DXImageTransform.Microsoft.BasicImage(rotation=' + (~ ~((file.rotation / 90) % 4 + 4) % 4) + ')'); // use jquery animate to rotation // $({ // rotation: rotation // }).animate({ // rotation: file.rotation // }, { // easing: 'linear', // step: function( now ) { // now = now * Math.PI / 180; // var cos = Math.cos( now ), // sin = Math.sin( now ); // $wrap.css( 'filter', "progid:DXImageTransform.Microsoft.Matrix(M11=" + cos + ",M12=" + (-sin) + ",M21=" + sin + ",M22=" + cos + ",SizingMethod='auto expand')"); // } // }); } }); $li.appendTo($queue); } // 負責view的銷毀 function removeFile(file) { var $li = $('#' + file.id); delete percentages[file.id]; updateTotalProgress(); $li.off().find('.file-panel').off().end().remove(); } function updateTotalProgress() { var loaded = 0, total = 0, spans = $progress.children(), percent; $.each(percentages, function (k, v) { total += v[0]; loaded += v[0] * v[1]; }); percent = total ? loaded / total : 0; spans.eq(0).text(Math.round(percent * 100) + '%'); spans.eq(1).css('width', Math.round(percent * 100) + '%'); updateStatus(); } function updateStatus() { var text = '', stats; if (state === 'ready') { text = '選中' + fileCount + '張圖片,共' + WebUploader.formatSize(fileSize) + '。'; } else if (state === 'confirm') { stats = uploader.getStats(); if (stats.uploadFailNum) { text = '已成功上傳' + stats.successNum + '張照片至XX相冊,' + stats.uploadFailNum + '張照片上傳失敗,<a class="retry" href="#">重新上傳</a>失敗圖片或<a class="ignore" href="#">忽略</a>' } } else { stats = uploader.getStats(); text = '共' + fileCount + '張(' + WebUploader.formatSize(fileSize) + '),已上傳' + stats.successNum + '張'; if (stats.uploadFailNum) { text += ',失敗' + stats.uploadFailNum + '張'; } } $info.html(text); } function setState(val) { var file, stats; if (val === state) { return; } $upload.removeClass('state-' + state); $upload.addClass('state-' + val); state = val; switch (state) { case 'pedding': $placeHolder.removeClass('element-invisible'); $queue.hide(); $statusBar.addClass('element-invisible'); uploader.refresh(); break; case 'ready': $placeHolder.addClass('element-invisible'); $('#filePicker2').removeClass('element-invisible'); $queue.show(); $statusBar.removeClass('element-invisible'); uploader.refresh(); break; case 'uploading': $('#filePicker2').addClass('element-invisible'); $progress.show(); $upload.text('暫停上傳'); break; case 'paused': $progress.show(); $upload.text('繼續上傳'); break; case 'confirm': $progress.hide(); $('#filePicker2').removeClass('element-invisible'); $upload.text('開始上傳'); stats = uploader.getStats(); if (stats.successNum && !stats.uploadFailNum) { setState('finish'); return; } break; case 'finish': stats = uploader.getStats(); if (stats.successNum) { alert('上傳完成'); } else { // 沒有成功的圖片,重設 state = 'done'; location.reload(); } break; } updateStatus(); } uploader.onUploadProgress = function (file, percentage) { var $li = $('#' + file.id), $percent = $li.find('.progress span'); $percent.css('width', percentage * 100 + '%'); percentages[file.id][1] = percentage; updateTotalProgress(); }; uploader.onFileQueued = function (file) { fileCount++; fileSize += file.size; if (fileCount === 1) { $placeHolder.addClass('element-invisible'); $statusBar.show(); } addFile(file); setState('ready'); updateTotalProgress(); }; uploader.onFileDequeued = function (file) { fileCount--; fileSize -= file.size; if (!fileCount) { setState('pedding'); } removeFile(file); updateTotalProgress(); }; //all算是一個總監聽器 uploader.on('all', function (type, arg1, arg2) { console.log("all監聽:", type, arg1, arg2); var stats; switch (type) { case 'uploadFinished': setState('confirm'); break; case 'startUpload': setState('uploading'); break; case 'stopUpload': setState('paused'); break; } }); // 文件上傳成功,合並文件。 uploader.on('uploadSuccess', function (file, response) { if (response.chunked) { $.post("MergeFiles.ashx", { guid: GUID, fileExt: response.f_ext }, function (data) { data = $.parseJSON(data); if (data.hasError) { alert('文件合並失敗!'); } else { alert(decodeURIComponent(data.savePath)); } }); } }); uploader.onError = function (code) { alert('Eroor: ' + code); }; $upload.on('click', function () { if ($(this).hasClass('disabled')) { return false; } if (state === 'ready') { uploader.upload(); } else if (state === 'paused') { uploader.upload(); } else if (state === 'uploading') { uploader.stop(); } }); $info.on('click', '.retry', function () { uploader.retry(); }); $info.on('click', '.ignore', function () { alert('todo'); }); $upload.addClass('state-' + state); updateTotalProgress(); }); })(jQuery);
5.一般處理程序代碼如下:
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Text; using System.IO; namespace WebUploadTest { /// <summary> /// Summary description for Handler1 /// </summary> public class Handler1 : IHttpHandler { #region 文件分片 public void ProcessRequest(HttpContext context) { context.Response.ContentType = "text/plain"; //如果進行了分片 if (context.Request.Form.AllKeys.Any(m => m == "chunk")) { //取得chunk和chunks int chunk = Convert.ToInt32(context.Request.Form["chunk"]);//當前分片在上傳分片中的順序(從0開始) int chunks = Convert.ToInt32(context.Request.Form["chunks"]);//總分片數 //根據GUID創建用該GUID命名的臨時文件夾 string folder = context.Server.MapPath("~/1/" + context.Request["guid"]+"/"); string path = folder + chunk; //建立臨時傳輸文件夾 if (!Directory.Exists(Path.GetDirectoryName(folder))) { Directory.CreateDirectory(folder); } FileStream addFile = new FileStream(path, FileMode.Append, FileAccess.Write); BinaryWriter AddWriter = new BinaryWriter(addFile); //獲得上傳的分片數據流 HttpPostedFile file = context.Request.Files[0]; Stream stream = file.InputStream; BinaryReader TempReader = new BinaryReader(stream); //將上傳的分片追加到臨時文件末尾 AddWriter.Write(TempReader.ReadBytes((int)stream.Length)); //關閉BinaryReader文件閱讀器 TempReader.Close(); stream.Close(); AddWriter.Close(); addFile.Close(); TempReader.Dispose(); stream.Dispose(); AddWriter.Dispose(); addFile.Dispose(); context.Response.Write("{\"chunked\" : true, \"hasError\" : false, \"f_ext\" : \"" + Path.GetExtension(file.FileName) + "\"}"); } else//沒有分片直接保存 { context.Request.Files[0].SaveAs(context.Server.MapPath("~/1/" + DateTime.Now.ToFileTime() + Path.GetExtension(context.Request.Files[0].FileName))); context.Response.Write("{\"chunked\" : false, \"hasError\" : false}"); } } #endregion public bool IsReusable { get { return false; } } } }
6.upload.js中實例化代碼詳解:
// 實例化 uploader = WebUploader.create({ pick: { id: '#filePicker', label: '點擊選擇文件' }, formData: { uid: 123 }, dnd: '#dndArea', paste: '#uploader', swf: 'Scripts/Uploader.swf', //必須要引入的文件 chunked: true, //分片處理大文件 chunkSize: 2 * 1024 * 1024, //每一片的大小 server: 'fileupload.ashx', //一般處理程序 disableGlobalDnd: true, threads: 1, //上傳並發數 //由於Http的無狀態特征,在往服務器發送數據過程傳遞一個進入當前頁面是生成的GUID作為標示 formData: { guid: GUID }, fileNumLimit: 2, //上傳文件的總數量限制 compress: false, //圖片在上傳前不進行壓縮 fileSizeLimit: 6*1024* 1024 * 1024, // 上傳文件的總大小限制 6G fileSingleSizeLimit: 5*1024* 1024 * 1024 // 上傳文件的單個文件大小限制 5 G });
7.官網參考地址:http://fex.baidu.com/webuploader/getting-started.html
8.原文參考地址:http://www.cnblogs.com/hijimi/p/4988027.html
