upload.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591
  1. (function( $ ){
  2. // 当domReady的时候开始初始化
  3. $(function() {
  4. var $wrap = $('#uploader'),
  5. // 图片容器
  6. $queue = $( '<ul class="filelist"></ul>' ).appendTo( $wrap.find( '.queueList' ) ),
  7. // 状态栏,包括进度和控制按钮
  8. $statusBar = $wrap.find( '.statusBar' ),
  9. // 文件总体选择信息。
  10. $info = $statusBar.find( '.info' ),
  11. // 上传按钮
  12. $upload = $wrap.find( '.uploadBtn' ),
  13. // 没选择文件之前的内容。
  14. $placeHolder = $wrap.find( '.placeholder' ),
  15. $progress = $statusBar.find( '.progress' ).hide(),
  16. // 添加的文件数量
  17. fileCount = 0,
  18. // 添加的文件总大小
  19. fileSize = 0,
  20. // 优化retina, 在retina下这个值是2
  21. ratio = window.devicePixelRatio || 1,
  22. // 缩略图大小
  23. thumbnailWidth = 110 * ratio,
  24. thumbnailHeight = 110 * ratio,
  25. // 可能有pedding, ready, uploading, confirm, done.
  26. state = 'pedding',
  27. // 所有文件的进度信息,key为file id
  28. percentages = {},
  29. // 判断浏览器是否支持图片的base64
  30. isSupportBase64 = ( function() {
  31. var data = new Image();
  32. var support = true;
  33. data.onload = data.onerror = function() {
  34. if( this.width != 1 || this.height != 1 ) {
  35. support = false;
  36. }
  37. }
  38. data.src = "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///ywAAAAAAQABAAACAUwAOw==";
  39. return support;
  40. } )(),
  41. // 检测是否已经安装flash,检测flash的版本
  42. flashVersion = ( function() {
  43. var version;
  44. try {
  45. version = navigator.plugins[ 'Shockwave Flash' ];
  46. version = version.description;
  47. } catch ( ex ) {
  48. try {
  49. version = new ActiveXObject('ShockwaveFlash.ShockwaveFlash')
  50. .GetVariable('$version');
  51. } catch ( ex2 ) {
  52. version = '0.0';
  53. }
  54. }
  55. version = version.match( /\d+/g );
  56. return parseFloat( version[ 0 ] + '.' + version[ 1 ], 10 );
  57. } )(),
  58. supportTransition = (function(){
  59. var s = document.createElement('p').style,
  60. r = 'transition' in s ||
  61. 'WebkitTransition' in s ||
  62. 'MozTransition' in s ||
  63. 'msTransition' in s ||
  64. 'OTransition' in s;
  65. s = null;
  66. return r;
  67. })(),
  68. // WebUploader实例
  69. uploader;
  70. if ( !WebUploader.Uploader.support('flash') && WebUploader.browser.ie ) {
  71. // flash 安装了但是版本过低。
  72. if (flashVersion) {
  73. (function(container) {
  74. window['expressinstallcallback'] = function( state ) {
  75. switch(state) {
  76. case 'Download.Cancelled':
  77. alert('您取消了更新!')
  78. break;
  79. case 'Download.Failed':
  80. alert('安装失败')
  81. break;
  82. default:
  83. alert('安装已成功,请刷新!');
  84. break;
  85. }
  86. delete window['expressinstallcallback'];
  87. };
  88. var swf = './expressInstall.swf';
  89. // insert flash object
  90. var html = '<object type="application/' +
  91. 'x-shockwave-flash" data="' + swf + '" ';
  92. if (WebUploader.browser.ie) {
  93. html += 'classid="clsid:d27cdb6e-ae6d-11cf-96b8-444553540000" ';
  94. }
  95. html += 'width="100%" height="100%" style="outline:0">' +
  96. '<param name="movie" value="' + swf + '" />' +
  97. '<param name="wmode" value="transparent" />' +
  98. '<param name="allowscriptaccess" value="always" />' +
  99. '</object>';
  100. container.html(html);
  101. })($wrap);
  102. // 压根就没有安转。
  103. } else {
  104. $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>');
  105. }
  106. return;
  107. } else if (!WebUploader.Uploader.support()) {
  108. alert( 'Web Uploader 不支持您的浏览器!');
  109. return;
  110. }
  111. // 实例化
  112. uploader = WebUploader.create({
  113. pick: {
  114. id: '#filePicker',
  115. label: '点击选择图片'
  116. },
  117. formData: {
  118. //uid: 123
  119. },
  120. dnd: '#uploader .queueList',
  121. paste: '#uploader',
  122. swf: $swfurl,
  123. chunked: false,
  124. chunkSize: 512 * 1024,
  125. server: uploadurl,
  126. // runtimeOrder: 'flash',
  127. accept: {
  128. title: '',
  129. extensions: $extensions,
  130. mimeTypes: ''
  131. },
  132. // 禁掉全局的拖拽功能。这样不会出现图片拖进页面的时候,把图片打开。
  133. disableGlobalDnd: true,
  134. fileNumLimit: $numlimit,
  135. fileSizeLimit: 200 * 1024 * 1024, // 200 M
  136. fileSingleSizeLimit: parseFloat($sizelimt) * 1024 *1024
  137. });
  138. uploader.on('uploadSuccess', function (file, response) {
  139. var imginfo = response._raw; //上传图片的路径
  140. imginfo = JSON.parse(imginfo);
  141. $.ajaxSettings.async = false;
  142. $.get('index.php?m=attachment&c=attachments&a=swfupload_json&aid='+imginfo.aid+'&src='+imginfo.filepath+'&filename='+imginfo.filename,
  143. function(res){
  144. //console.log(res)
  145. });
  146. $.ajaxSettings.async = true;
  147. $('#att-status').append('|'+ imginfo.filepath);
  148. $('#att-name').append('|'+ imginfo.filename);
  149. });
  150. uploader.on( 'uploadBeforeSend', function( block, data ) {
  151. var file = block.file;
  152. // 修改data可以控制发送哪些携带数据。
  153. data.module = $module;
  154. data.catid = $catid;
  155. data.authkey = $authkey;
  156. data.watermark_enable = $('#watermark_enable').attr('checked')? 1 : 0;
  157. });
  158. // 拖拽时不接受 js, txt 文件。
  159. uploader.on( 'dndAccept', function( items ) {
  160. var denied = false,
  161. len = items.length,
  162. i = 0,
  163. // 修改js类型
  164. unAllowed = 'text/plain;application/javascript ';
  165. for ( ; i < len; i++ ) {
  166. // 如果在列表里面
  167. if ( ~unAllowed.indexOf( items[ i ].type ) ) {
  168. denied = true;
  169. break;
  170. }
  171. }
  172. return !denied;
  173. });
  174. uploader.on('dialogOpen', function() {
  175. console.log('here');
  176. });
  177. // uploader.on('filesQueued', function() {
  178. // uploader.sort(function( a, b ) {
  179. // if ( a.name < b.name )
  180. // return -1;
  181. // if ( a.name > b.name )
  182. // return 1;
  183. // return 0;
  184. // });
  185. // });
  186. // 添加“添加文件”的按钮,
  187. uploader.addButton({
  188. id: '#filePicker2',
  189. label: '继续添加'
  190. });
  191. uploader.on('ready', function() {
  192. window.uploader = uploader;
  193. });
  194. // 当有文件添加进来时执行,负责view的创建
  195. function addFile( file ) {
  196. var $li = $( '<li id="' + file.id + '">' +
  197. '<p class="title">' + file.name + '</p>' +
  198. '<p class="imgWrap"></p>'+
  199. '<p class="progress"><span></span></p>' +
  200. '</li>' ),
  201. $btns = $('<div class="file-panel">' +
  202. '<span class="cancel">删除</span>' +
  203. '<span class="rotateRight">向右旋转</span>' +
  204. '<span class="rotateLeft">向左旋转</span></div>').appendTo( $li ),
  205. $prgress = $li.find('p.progress span'),
  206. $wrap = $li.find( 'p.imgWrap' ),
  207. $info = $('<p class="error"></p>'),
  208. showError = function( code ) {
  209. switch( code ) {
  210. case 'exceed_size':
  211. text = '文件大小超出';
  212. break;
  213. case 'interrupt':
  214. text = '上传暂停';
  215. break;
  216. default:
  217. text = '上传失败,请重试';
  218. break;
  219. }
  220. $info.text( text ).appendTo( $li );
  221. };
  222. if ( file.getStatus() === 'invalid' ) {
  223. showError( file.statusText );
  224. } else {
  225. // @todo lazyload
  226. $wrap.text( '预览中' );
  227. uploader.makeThumb( file, function( error, src ) {
  228. var img;
  229. if ( error ) {
  230. $wrap.text( '不能预览' );
  231. return;
  232. }
  233. if( isSupportBase64 ) {
  234. img = $('<img src="'+src+'">');
  235. $wrap.empty().append( img );
  236. } else {
  237. $.ajax(uploadurl, {
  238. method: 'POST',
  239. data: src,
  240. dataType:'json'
  241. }).done(function( response ) {
  242. if (response.result) {
  243. img = $('<img src="'+response.result+'">');
  244. $wrap.empty().append( img );
  245. } else {
  246. $wrap.text("预览出错");
  247. }
  248. });
  249. }
  250. }, thumbnailWidth, thumbnailHeight );
  251. percentages[ file.id ] = [ file.size, 0 ];
  252. file.rotation = 0;
  253. }
  254. file.on('statuschange', function( cur, prev ) {
  255. if ( prev === 'progress' ) {
  256. $prgress.hide().width(0);
  257. } else if ( prev === 'queued' ) {
  258. $li.off( 'mouseenter mouseleave' );
  259. $btns.remove();
  260. }
  261. // 成功
  262. if ( cur === 'error' || cur === 'invalid' ) {
  263. console.log( file.statusText );
  264. showError( file.statusText );
  265. percentages[ file.id ][ 1 ] = 1;
  266. } else if ( cur === 'interrupt' ) {
  267. showError( 'interrupt' );
  268. } else if ( cur === 'queued' ) {
  269. $info.remove();
  270. $prgress.css('display', 'block');
  271. percentages[ file.id ][ 1 ] = 0;
  272. } else if ( cur === 'progress' ) {
  273. $info.remove();
  274. $prgress.css('display', 'block');
  275. } else if ( cur === 'complete' ) {
  276. $prgress.hide().width(0);
  277. $li.append( '<span class="success"></span>' );
  278. }
  279. $li.removeClass( 'state-' + prev ).addClass( 'state-' + cur );
  280. });
  281. $li.on( 'mouseenter', function() {
  282. $btns.stop().animate({height: 30});
  283. });
  284. $li.on( 'mouseleave', function() {
  285. $btns.stop().animate({height: 0});
  286. });
  287. $btns.on( 'click', 'span', function() {
  288. var index = $(this).index(),
  289. deg;
  290. switch ( index ) {
  291. case 0:
  292. uploader.removeFile( file );
  293. return;
  294. case 1:
  295. file.rotation += 90;
  296. break;
  297. case 2:
  298. file.rotation -= 90;
  299. break;
  300. }
  301. if ( supportTransition ) {
  302. deg = 'rotate(' + file.rotation + 'deg)';
  303. $wrap.css({
  304. '-webkit-transform': deg,
  305. '-mos-transform': deg,
  306. '-o-transform': deg,
  307. 'transform': deg
  308. });
  309. } else {
  310. $wrap.css( 'filter', 'progid:DXImageTransform.Microsoft.BasicImage(rotation='+ (~~((file.rotation/90)%4 + 4)%4) +')');
  311. // use jquery animate to rotation
  312. // $({
  313. // rotation: rotation
  314. // }).animate({
  315. // rotation: file.rotation
  316. // }, {
  317. // easing: 'linear',
  318. // step: function( now ) {
  319. // now = now * Math.PI / 180;
  320. // var cos = Math.cos( now ),
  321. // sin = Math.sin( now );
  322. // $wrap.css( 'filter', "progid:DXImageTransform.Microsoft.Matrix(M11=" + cos + ",M12=" + (-sin) + ",M21=" + sin + ",M22=" + cos + ",SizingMethod='auto expand')");
  323. // }
  324. // });
  325. }
  326. });
  327. $li.appendTo( $queue );
  328. }
  329. // 负责view的销毁
  330. function removeFile( file ) {
  331. var $li = $('#'+file.id);
  332. delete percentages[ file.id ];
  333. updateTotalProgress();
  334. $li.off().find('.file-panel').off().end().remove();
  335. }
  336. function updateTotalProgress() {
  337. var loaded = 0,
  338. total = 0,
  339. spans = $progress.children(),
  340. percent;
  341. $.each( percentages, function( k, v ) {
  342. total += v[ 0 ];
  343. loaded += v[ 0 ] * v[ 1 ];
  344. } );
  345. percent = total ? loaded / total : 0;
  346. spans.eq( 0 ).text( Math.round( percent * 100 ) + '%' );
  347. spans.eq( 1 ).css( 'width', Math.round( percent * 100 ) + '%' );
  348. updateStatus();
  349. }
  350. function updateStatus() {
  351. var text = '', stats;
  352. if ( state === 'ready' ) {
  353. text = '选中' + fileCount + '张图片,共' +
  354. WebUploader.formatSize( fileSize ) + '。';
  355. } else if ( state === 'confirm' ) {
  356. stats = uploader.getStats();
  357. if ( stats.uploadFailNum ) {
  358. text = '已成功上传' + stats.successNum+ '张照片至XX相册,'+
  359. stats.uploadFailNum + '张照片上传失败,<a class="retry" href="#">重新上传</a>失败图片或<a class="ignore" href="#">忽略</a>'
  360. }
  361. } else {
  362. stats = uploader.getStats();
  363. text = '共' + fileCount + '张(' +
  364. WebUploader.formatSize( fileSize ) +
  365. '),已上传' + stats.successNum + '张';
  366. if ( stats.uploadFailNum ) {
  367. text += ',失败' + stats.uploadFailNum + '张';
  368. }
  369. }
  370. $info.html( text );
  371. }
  372. function setState( val ) {
  373. var file, stats;
  374. if ( val === state ) {
  375. return;
  376. }
  377. $upload.removeClass( 'state-' + state );
  378. $upload.addClass( 'state-' + val );
  379. state = val;
  380. switch ( state ) {
  381. case 'pedding':
  382. $placeHolder.removeClass( 'element-invisible' );
  383. $queue.hide();
  384. $statusBar.addClass( 'element-invisible' );
  385. uploader.refresh();
  386. break;
  387. case 'ready':
  388. $placeHolder.addClass( 'element-invisible' );
  389. $( '#filePicker2' ).removeClass( 'element-invisible');
  390. $queue.show();
  391. $statusBar.removeClass('element-invisible');
  392. uploader.refresh();
  393. break;
  394. case 'uploading':
  395. $( '#filePicker2' ).addClass( 'element-invisible' );
  396. $progress.show();
  397. $upload.text( '暂停上传' );
  398. break;
  399. case 'paused':
  400. $progress.show();
  401. $upload.text( '继续上传' );
  402. break;
  403. case 'confirm':
  404. $progress.hide();
  405. $( '#filePicker2' ).removeClass( 'element-invisible' );
  406. $upload.text( '开始上传' );
  407. stats = uploader.getStats();
  408. if ( stats.successNum && !stats.uploadFailNum ) {
  409. setState( 'finish' );
  410. return;
  411. }
  412. break;
  413. case 'finish':
  414. stats = uploader.getStats();
  415. if ( stats.successNum ) {
  416. alert( '上传成功' );
  417. } else {
  418. // 没有成功的图片,重设
  419. state = 'done';
  420. location.reload();
  421. }
  422. break;
  423. }
  424. updateStatus();
  425. }
  426. uploader.onUploadProgress = function( file, percentage ) {
  427. var $li = $('#'+file.id),
  428. $percent = $li.find('.progress span');
  429. $percent.css( 'width', percentage * 100 + '%' );
  430. percentages[ file.id ][ 1 ] = percentage;
  431. updateTotalProgress();
  432. };
  433. uploader.onFileQueued = function( file ) {
  434. fileCount++;
  435. fileSize += file.size;
  436. if ( fileCount === 1 ) {
  437. $placeHolder.addClass( 'element-invisible' );
  438. $statusBar.show();
  439. }
  440. addFile( file );
  441. setState( 'ready' );
  442. updateTotalProgress();
  443. };
  444. uploader.onFileDequeued = function( file ) {
  445. fileCount--;
  446. fileSize -= file.size;
  447. if ( !fileCount ) {
  448. setState( 'pedding' );
  449. }
  450. removeFile( file );
  451. updateTotalProgress();
  452. };
  453. uploader.on( 'all', function( type ) {
  454. var stats;
  455. switch( type ) {
  456. case 'uploadFinished':
  457. setState( 'confirm' );
  458. break;
  459. case 'startUpload':
  460. setState( 'uploading' );
  461. break;
  462. case 'stopUpload':
  463. setState( 'paused' );
  464. break;
  465. }
  466. });
  467. uploader.onError = function( code ) {
  468. if(code == "F_EXCEED_SIZE"){
  469. alert("上传出错,文件大小不得大于"+ $sizelimt)
  470. }
  471. if(code == "Q_EXCEED_NUM_LIMIT"){
  472. alert("上传出错,文件数量多于"+ $numlimit + "个")
  473. }
  474. if(code == "Q_EXCEED_SIZE_LIMIT"){
  475. alert("上传出错,文件大小总和超限")
  476. }
  477. if(code == "Q_TYPE_DENIED"){
  478. alert("上传出错,文件类型错误")
  479. }
  480. };
  481. $upload.on('click', function() {
  482. if ( $(this).hasClass( 'disabled' ) ) {
  483. return false;
  484. }
  485. if ( state === 'ready' ) {
  486. uploader.upload();
  487. } else if ( state === 'paused' ) {
  488. uploader.upload();
  489. } else if ( state === 'uploading' ) {
  490. uploader.stop();
  491. }
  492. });
  493. $info.on( 'click', '.retry', function() {
  494. uploader.retry();
  495. } );
  496. $info.on( 'click', '.ignore', function() {
  497. alert( 'todo' );
  498. } );
  499. $upload.addClass( 'state-' + state );
  500. updateTotalProgress();
  501. });
  502. })( jQuery );