/*
* LUploader圖片壓縮上傳插件
*
* 做者:黃磊
*
* 報告漏洞,意見或建議, 請聯繫郵箱:xfhxbb@yeah.net
*
* 建立於:2016年3月16日
*
* Copyright 2016
*
* 得到使用本類庫的許可, 您必須保留著做權聲明信息。
*
*/
(function() {
window.LUploader = function(el, params) {
var _self = this;
_self.trigger=el;
_self.params = {
accept: 'image/*',
multiple: false,
maxsize: 102400,
imgObj: {},
showsize: false,
quality:0.1,
url: ''
}
for (var param in params) {
_self.params[param] = params[param];
}
_self.init();
};
LUploader.prototype.init = function() {
var _self = this;
_self.trigger.setAttribute('accept', _self.params.accept);
_self.params.multiple && _self.trigger.setAttribute('multiple', '');
var btn = document.querySelector('#' + _self.trigger.getAttribute('data-LUploader'));
btn.addEventListener('click', function() {
_self.trigger.click();
});
_self.trigger.addEventListener('change', function() {
if (!this.files.length) return;
var files = Array.prototype.slice.call(this.files);
files.forEach(function(file, i) {
if (!/\/(?:jpeg|png|gif)/i.test(file.type)) return;
var reader = new FileReader();
_self.params.imgObj.size = file.size / 1024 > 1024 ? (~~(10 * file.size / 1024 / 1024)) / 10 + "MB" : ~~(file.size / 1024) + "KB";
var li = document.createElement("li");
li.innerHTML = '<div class="LUploader-progress"><span></span><input type="hidden" value="" /></div>';
if (_self.params.showsize) {
var div_size = document.createElement('div');
div_size.className = 'LUploader-size';
div_size.textContent = _self.params.imgObj.size;
li.appendChild(div_size);
}
var LUploaderList = _self.trigger.parentElement.querySelector('.LUploader-list');
if (!_self.params.multiple) { //假如是單個上傳
if (_self.old_li) {
LUploaderList.removeChild(_self.old_li);
} else {
_self.old_li = li;
}
}
LUploaderList.appendChild(li);
LUploaderList.parentElement.nextElementSibling.style['display'] = 'none';
reader.onload = function() {
var params = dataSet(_self.trigger);
var url = _self.params.url;
var result = this.result;
var img = new Image();
_self.params.imgObj.src = img.src = result;
li.style['background-image'] = 'url(' + result + ')';
if (result.length <= _self.params.maxsize) {
img = null;
_self.upload(url, params, result, file.type, li);
return;
}
if (img.complete) {
callback();
} else {
img.onload = callback;
}
function callback() {
var data = _self.compress(img);
_self.upload(url, params, data, file.type, li);
img = null;
}
};
reader.readAsDataURL(file);
});
});
};
LUploader.prototype.compress = function(img) {
var canvas = document.createElement("canvas");
var ctx = canvas.getContext('2d');
var moreCanvas = document.createElement("canvas");
var morectx = moreCanvas.getContext("2d");
var maxsize = 100 * 1024;
var width = img.width;
var height = img.height;
var ratio;
if ((ratio = width * height / 4000000) > 1) {
ratio = Math.sqrt(ratio);
width /= ratio;
height /= ratio;
} else {
ratio = 1;
}
canvas.width = width;
canvas.height = height;
ctx.fillStyle = "#fff";
ctx.fillRect(0, 0, canvas.width, canvas.height);
var count;
if ((count = width * height / 1000000) > 1) {
count = ~~(Math.sqrt(count) + 1);
var nw = ~~(width / count);
var nh = ~~(height / count);
moreCanvas.width = nw;
moreCanvas.height = nh;
for (var i = 0; i < count; i++) {
for (var j = 0; j < count; j++) {
morectx.drawImage(img, i * nw * ratio, j * nh * ratio, nw * ratio, nh * ratio, 0, 0, nw, nh);
ctx.drawImage(moreCanvas, i * nw, j * nh, nw, nh);
}
}
} else {
ctx.drawImage(img, 0, 0, width, height);
}
var ndata = canvas.toDataURL('image/jpeg', this.params.quality);
moreCanvas.width = moreCanvas.height = canvas.width = canvas.height = 0;
return ndata;
};
LUploader.prototype.upload = function(url, obj, basestr, type, li) {
var text = window.atob(basestr.split(",")[1]);
var buffer = new Uint8Array(text.length);
var pecent = 0;
for (var i = 0; i < text.length; i++) {
buffer[i] = text.charCodeAt(i);
}
var span = li.querySelector('.LUploader-progress').querySelector('span');
var hidden_input = li.querySelector('.LUploader-progress').querySelector('input');
var xhr = new XMLHttpRequest();
xhr.open('post', url);
xhr.onload = function(e) {
if ((xhr.status >= 200 && xhr.status < 300) || xhr.status === 0) {
var data = JSON.parse(xhr.responseText);
var result = data['status'];
var text = result == 0 ? '上傳成功' : '上傳失敗';
span.style['width'] = '100%';
span.innerHTML = text;
hidden_input.value = data['path'];
}
else {
span.innerHTML = '上傳失敗';
}
}
xhr.upload.addEventListener('progress', function(e) {
pecent = ~~(100 * e.loaded / e.total);
span.style['width'] = pecent + '%';
span.innerHTML = (pecent == 100 ? 99 : pecent) + '%';
}, false);
var data = {};
for (var key in obj) {
if (key !== 'luploader') {
if (obj[key] == 'basestr') {
data[key] = basestr;
} else {
data[key] = obj[key];
}
}
};
data = serializeObject(data);
xhr.setRequestHeader('Content-type', 'application/x-www-form-urlencoded; charset=UTF-8');
xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
xhr.send(data);
}
function isArray(arr) {
if (Object.prototype.toString.apply(arr) === '[object Array]') return true;
else return false;
};
function serializeObject(obj) {
if (typeof obj === 'string') return obj;
var resultArray = [];
var separator = '&';
for (var prop in obj) {
if (obj.hasOwnProperty(prop)) {
if (isArray(obj[prop])) {
var toPush = [];
for (var i = 0; i < obj[prop].length; i++) {
toPush.push(encodeURIComponent(prop) + '=' + encodeURIComponent(obj[prop][i]));
}
if (toPush.length > 0) resultArray.push(toPush.join(separator));
} else {
resultArray.push(encodeURIComponent(prop) + '=' + encodeURIComponent(obj[prop]));
}
}
}
return resultArray.join(separator);
};
function dataSet(el) {
var dataset = {};
for (var i = 0; i < el.attributes.length; i++) {
var attr = el.attributes[i];
if (attr.name.indexOf('data-') >= 0) {
dataset[toCamelCase(attr.name.split('data-')[1])] = attr.value;
}
}
return dataset;
}
function toCamelCase(string) {
return string.toLowerCase().replace(/-(.)/g, function(match, group1) {
return group1.toUpperCase();
});
};
})();
<!doctype html>
<html>
<head>
<meta charset="utf-8" />
<meta http-equiv="Access-Control-Allow-Origin" content="*">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no, minimal-ui">
<meta content="yes" name="apple-mobile-web-app-capable">
<meta content="black" name="apple-mobile-web-app-status-bar-style">
<meta content="telephone=no" name="format-detection">
<meta content="email=no" name="format-detection">
<title>移動端圖片壓縮上傳</title>
<link rel="stylesheet" href="css/LUploader.css">
<style>
* {
margin: 0;
padding: 0;
}
h1{
background-color: #19b5ee;
color: #fff;
font-size: 25px;
text-align: center;
padding: 10px;
}
li {
list-style-type: none;
}
input {
outline: none;
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
}
</style>
</head>
<body>
<h1>LUploader移動端圖片壓縮上傳</h1>
<div class="LUploader" id="demo1">
<div class="LUploader-container">
<input data-LUploader='demo1' data-form-file='basestr' data-upload-type='front' type="file" />
<ul class="LUploader-list"></ul>
</div>
<div>
<div class="icon icon-camera font20"></div>
<p>單擊上傳</p>
</div>
</div>
<script src="js/LUploader.js?r=<?php echo time();?>"></script>
<script>
[].slice.call(document.querySelectorAll('input[data-LUploader]')).forEach(function(el) {
new LUploader(el, {
url: './upload.php',//post請求地址
multiple: false,//是否一次上傳多個文件 默認false
maxsize: 102400,//忽略壓縮操做的文件體積上限 默認100kb
accept: 'image/*',//可上傳的圖片類型
quality: 0.5,//壓縮比 默認0.1 範圍0.1-1.0 越小壓縮率越大
showsize:true//是否顯示原始文件大小 默認false
});
});
</script>
</body>
</html>