diff --git a/app/admin/view/api/upload.js b/app/admin/view/api/upload.js index 215c54b42..ebca802ab 100644 --- a/app/admin/view/api/upload.js +++ b/app/admin/view/api/upload.js @@ -29,17 +29,18 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) { /*! 初始化上传组件 */ this.adapter = new Adapter(this.option, layui.upload.render({ - url: '{:url("admin/api.upload/file")}', auto: false, elem: elem, accept: 'file', multiple: this.option.mult, exts: this.option.exts.join('|'), acceptMime: this.option.mimes.join(','), choose: function (object) { - object.files = object.pushFile(); - layui.each(object.files, function (idx, file) { + url: '{:url("admin/api.upload/file")}', auto: false, elem: elem, accept: 'file', multiple: this.option.mult, exts: this.option.exts.join('|'), acceptMime: this.option.mimes.join(','), choose: function (obj) { + obj.items = [], obj.files = obj.pushFile(); + layui.each(obj.files, function (idx, file) { + obj.items.push(file); file.quality = that.option.quality; file.maxWidth = that.option.maxWidth; file.maxHeight = that.option.maxHeight; }); - that.adapter.event('upload.choose', object.files); - that.adapter.upload(object.files, done); - layui.each(object.files, function (idx) { - delete object.files[idx]; + that.adapter.event('upload.choose', obj.items); + that.adapter.upload(obj.items, done); + layui.each(obj.files, function (idx) { + delete obj.files[idx]; }); } })); @@ -74,14 +75,15 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) { // 禁传异常状态文件 if (typeof file.xstate === 'number' && file.xstate === -1) return; // 图片限宽限高压缩 - if (/^image\/*$/.test(file.type) && file.maxWidth > 0 || file.maxHeight > 0 || file.quality !== 1) { - FileToBase64(file).then(function (base64) { - ImageToThumb(base64, file).then(function (base64) { - files[index] = Base64ToFile(base64, file.name); - files[index].notify = file.notify; - that.hash(files[index]).then(function (file) { - that.event('upload.hash', file).request(file, done); - }); + if (/^image\//.test(file.type) && (file.maxWidth > 0 || file.maxHeight > 0 || file.quality !== 1)) { + require(['compressor'], function (Compressor) { + new Compressor(file, { + quality: file.quality, maxWidth: file.maxWidth, maxHeight: file.maxHeight, success(blob) { + files[index] = blob, blob.index = file.index, files[index].notify = file.notify; + that.hash(files[index]).then(function (file) { + that.event('upload.hash', file).request(file, done); + }); + } }); }); } else { @@ -132,7 +134,7 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) { uploader.form.append('authorization', ret.data['authorization']); uploader.form.append('Content-Disposition', 'inline;filename=' + encodeURIComponent(file.name)); } - uploader.form.append('file', file), jQuery.ajax({ + uploader.form.append('file', file, file.name), jQuery.ajax({ url: uploader.url, data: uploader.form, type: 'post', xhr: function (xhr) { xhr = new XMLHttpRequest(); return xhr.upload.addEventListener('progress', function (event) { @@ -263,61 +265,6 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) { return UploadAdapter; - /** - * Base64 转 File 对象 - * @param {String} base64 Base64内容 - * @param {String} filename 新文件名称 - * @return {File} - */ - function Base64ToFile(base64, filename) { - var arr = base64.split(','); - var mime = arr[0].match(/:(.*?);/)[1], suffix = mime.split('/')[1]; - var bstr = atob(arr[1]), n = bstr.length, u8arr = new Uint8Array(n); - while (n--) u8arr[n] = bstr.charCodeAt(n); - return new File([u8arr], filename + '.' + suffix, {type: mime}); - } - - /** - * File 对象转 Base64 - * @param {File} file 文件对象 - * @return {Promise} - */ - function FileToBase64(file) { - var defer = jQuery.Deferred(), reader = new FileReader(); - return (reader.onload = function () { - defer.resolve(this.result); - }), reader.readAsDataURL(file), defer.promise(); - } - - /** - * 图片压缩处理 - * @param {String} url 图片链接 - * @param {Object} option 压缩参数 - * @return {Promise} - */ - function ImageToThumb(url, option) { - var defer = jQuery.Deferred(), image = new Image(); - image.src = url, image.onload = function () { - var canvas = document.createElement('canvas'), context = canvas.getContext('2d'); - option.maxWidth = option.maxWidth || this.width, option.maxHeight = option.maxHeight || this.height; - var originWidth = this.width, originHeight = this.height, targetWidth = originWidth, targetHeight = originHeight; - if (originWidth > option.maxWidth || originHeight > option.maxHeight) { - if (originWidth / option.maxWidth > option.maxWidth / option.maxHeight) { - targetWidth = option.maxWidth; - targetHeight = Math.round(option.maxWidth * (originHeight / originWidth)); - } else { - targetHeight = option.maxHeight; - targetWidth = Math.round(option.maxHeight * (originWidth / originHeight)); - } - } - canvas.width = targetWidth, canvas.height = targetHeight; - context.clearRect(0, 0, targetWidth, targetHeight); - context.drawImage(this, 0, 0, targetWidth, targetHeight); - defer.resolve(canvas.toDataURL('image/jpeg', option.quality || 0.9)); - }; - return defer.promise(); - } - /** * 上传状态提示扩展插件 * @param {File} file 文件对象 diff --git a/public/static/admin.js b/public/static/admin.js index 74a9ff75c..cefb1e354 100644 --- a/public/static/admin.js +++ b/public/static/admin.js @@ -70,6 +70,7 @@ require.config({ 'filesaver': ['plugs/jquery/filesaver.min'], 'websocket': ['plugs/socket/websocket'], 'pcasunzips': ['plugs/jquery/pcasunzips'], + 'compressor': ['plugs/jquery/compressor.min'], 'sortablejs': ['plugs/sortable/sortable.min'], 'vue.sortable': ['plugs/sortable/vue.draggable.min'], 'jquery.ztree': ['plugs/ztree/ztree.all.min'], @@ -583,7 +584,11 @@ $(function () { return this.each(function () { if ($(this).data('inited')) return true; else $(this).data('inited', true); var $in = $(this), $bt = $('').data('input', this); - $bt.attr('data-size', $in.data('size') || 0).attr('data-type', $in.data('type') || 'mp4').find('span').on('click', function (event) { + $bt.attr('data-size', $in.data('size') || 0).attr('data-type', $in.data('type') || 'mp4') + .attr('data-max-width', $in.data('max-width')) + .attr('data-max-height', $in.data('max-height')) + + .find('span').on('click', function (event) { event.stopPropagation(); if ($(this).index() === 0) $bt.attr('style', ''), $in.val(''); else $in.val() && $.previewImage(encodeURI($in.val())); }), $in.on('change', function () { @@ -604,7 +609,10 @@ $(function () { event.stopPropagation(), $in.val() && $.previewImage(encodeURI($in.val())); }).on('click', 'i.layui-icon-close', function (event) { event.stopPropagation(), $bt.attr('style', ''), $in.val(''); - }).find('[data-file]').data('input', this).attr('data-size', $in.data('size') || 0).attr('data-type', $in.data('type') || 'png,jpg,gif,jpeg'); + }).find('[data-file]').data('input', this).attr({ + 'data-size': $in.data('size') || 0, 'data-type': $in.data('type') || 'gif,png,jpg,jpeg', + 'data-max-width': $in.data('max-width') || 0, 'data-max-height': $in.data('max-height') || 0 + }); }); }; @@ -614,7 +622,10 @@ $(function () { if ($(this).data('inited')) return true; else $(this).data('inited', true); var $bt = $('
'); var ims = this.value ? this.value.split('|') : [], $in = $(this).after($bt); - $bt.find('[data-file]').attr('data-size', $in.data('size') || 0).attr('data-type', $in.data('type') || 'gif,png,jpg,jpeg').on('push', function (evt, src) { + $bt.find('[data-file]').attr({ + 'data-size': $in.data('size') || 0, 'data-type': $in.data('type') || 'gif,png,jpg,jpeg', + 'data-max-width': $in.data('max-width') || 0, 'data-max-height': $in.data('max-height') || 0 + }).on('push', function (evt, src) { ims.push(src), $in.val(ims.join('|')), showImageContainer([src]); }) && (ims.length > 0 && showImageContainer(ims)); diff --git a/public/static/plugs/jquery/compressor.min.js b/public/static/plugs/jquery/compressor.min.js new file mode 100644 index 000000000..f18770a42 --- /dev/null +++ b/public/static/plugs/jquery/compressor.min.js @@ -0,0 +1,10 @@ +/*! + * Compressor.js v1.1.1 + * https://fengyuanchen.github.io/compressorjs + * + * Copyright 2018-present Chen Fengyuan + * Released under the MIT license + * + * Date: 2021-10-05T02:32:40.212Z + */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define(t):(e="undefined"!=typeof globalThis?globalThis:e||self).Compressor=t()}(this,function(){"use strict";function t(t,e){var r,i=Object.keys(t);return Object.getOwnPropertySymbols&&(r=Object.getOwnPropertySymbols(t),e&&(r=r.filter(function(e){return Object.getOwnPropertyDescriptor(t,e).enumerable})),i.push.apply(i,r)),i}function a(i){for(var e=1;ef.convertSize&&0<=f.convertTypes.indexOf(f.mimeType)&&(f.mimeType="image/jpeg"),"image/jpeg"===f.mimeType&&(B="#fff"),m.fillStyle=B,m.fillRect(0,0,e,l),f.beforeDraw&&f.beforeDraw.call(this,m,d),this.aborted||(m.save(),m.translate(e/2,l/2),m.rotate(n*Math.PI/180),m.scale(s,c),m.drawImage.apply(m,[u].concat(a)),m.restore(),f.drew&&f.drew.call(this,m,d),this.aborted||(m=function(e){t.aborted||t.done({naturalWidth:r,naturalHeight:i,result:e})},d.toBlob?d.toBlob(m,f.mimeType,f.quality):m(T(d.toDataURL(f.mimeType,f.quality)))))}},{key:"done",value:function(e){var t=e.naturalWidth,r=e.naturalHeight,i=e.result,a=this.file,n=this.image,e=this.options;A&&!e.checkOrientation&&A.revokeObjectURL(n.src),!i||e.strict&&i.size>a.size&&e.mimeType===a.type&&!(e.width>t||e.height>r||e.minWidth>t||e.minHeight>r||e.maxWidth