Jquery插件-Html5图片上传并裁剪
/**
* 图片裁剪
* @author yanglizhe
* 2015/11/16
*/
(function($){ /**
* Drag
*/
var Drag={obj:null,init:function(elementHeader,element){elementHeader.onmousedown=Drag.start;elementHeader.obj=element;if(isNaN(parseInt(element.style.left))){element.style.left="0px"}if(isNaN(parseInt(element.style.top))){element.style.top="0px"}element.onDragStart=new Function();element.onDragEnd=new Function();element.onDrag=new Function()},start:function(event){var element=Drag.obj=this.obj;event=Drag.fixE(event);if(event.which!=1){return true}element.onDragStart();element.lastMouseX=event.clientX;element.lastMouseY=event.clientY;document.onmouseup=Drag.end;document.onmousemove=Drag.drag;return false},drag:function(event){event=Drag.fixE(event);if(event.which==0){return Drag.end()}var element=Drag.obj;var _clientX=event.clientY;var _clientY=event.clientX;if(element.lastMouseX==_clientY&&element.lastMouseY==_clientX){return false}var _lastX=parseInt(element.style.top);var _lastY=parseInt(element.style.left);var newX,newY;newX=_lastY+_clientY-element.lastMouseX;newY=_lastX+_clientX-element.lastMouseY;element.style.left=newX+"px";element.style.top=newY+"px";element.lastMouseX=_clientY;element.lastMouseY=_clientX;element.onDrag(newX,newY);return false},end:function(event){event=Drag.fixE(event);document.onmousemove=null;document.onmouseup=null;var _onDragEndFuc=Drag.obj.onDragEnd();Drag.obj=null;return _onDragEndFuc},fixE:function(ig_){if(typeof ig_=="undefined"){ig_=window.event}if(typeof ig_.layerX=="undefined"){ig_.layerX=ig_.offsetX}if(typeof ig_.layerY=="undefined"){ig_.layerY=ig_.offsetY}if(typeof ig_.which=="undefined"){ig_.which=ig_.button}return ig_}}; /**
* 创建样式表
* @param String cssStr
*/
function createStyle(cssStr){
if( document.createStyleSheet){
var style = document.createStyleSheet();
style.cssText = cssStr;
}
else{
var style = document.createElement("style");
style.type = "text/css";
style.textContent = cssStr;
document.getElementsByTagName("head").item(0).appendChild(style);
}
} var MAX_SIZE = 200; function ImageCrop(config){
this.zoomMin = 0.1; //最小比例
this.zoomMax = 10; //最大比例
this.wrapWidth = 280;
this.wrapHeight = 260;
this.gripMinX = 0;
this.gripMaxX = 189;
this.originalWidth = 0;
this.originalHeight = 0;
this.originalLeft = 0;
this.origianlTop = 0;
this._construct(config);
} ImageCrop.prototype = {
/**
* 构建方法
* @param config
* @private
*/
_construct:function(config){
this.eventTarget = config.target;
this.cropWidth = config.width || 140;
this.cropHeight = config.height || 140;
this.ratio = 1;
if(this.cropWidth > MAX_SIZE || this.cropHeight > MAX_SIZE){
if(this.cropWidth > this.cropHeight){
this.ratio = MAX_SIZE/this.cropWidth;
this.cropWidth = MAX_SIZE;
this.cropHeight *= this.ratio;
}
else{
this.ratio = MAX_SIZE / this.cropHeight;
this.cropHeight = MAX_SIZE;
this.cropWidth *= this.ratio;
}
}
this.offsetX = (this.wrapWidth - this.cropWidth ) / 2;
this.offsetY = (this.wrapHeight - this.cropHeight) / 2;
this.destWidth = config.destWidth || this.cropWidth;
this.destHeight = config.destHeight || this.cropHeight;
this.onConfirm = config.onConfirm || function(){};
var parent = config.container || $(document.body);
this.initHTML(parent);
this.bindEvent(); },
initHTML:function(parent){
var html = '<div class="crop-container" style="display:block;">'+
' <div class="left-box">'+
' <div class="crop-wrapper">'+
' <div class="crop-preview"><div class="crop-preview-box"></div></div>'+
' <img class="crop-target" style="position:relative;" src="" />'+
' </div>'+
' <div class="crop-tool-bar">'+
' <span><a href="javascript:;" class="zoom-tool btn-zoom zoom-out" alt="图片缩小"></a></span>'+
' <span class="bar-zoom-box">'+
' <span class="zoom-tool bar-zoom"></span>'+
' <a href="javascript:;" class= "zoom-tool crop-grip"></a>'+
' </span>'+
' <span><a href="javascript:;" class="zoom-tool btn-zoom zoom-in" alt="图片放大"></a></span>'+
' </div>'+
' </div>'+
' <div class="right-box">'+
' <div class="image-preview"><img src="" /></div>'+
' <div class="btn-box"><a class="btn-upload-image" href="javascript:;">图片上传</a></div>'+
' <div class="btn-box"><a class="btn-confirm-crop" href="javascript:;">确认裁剪</a></div>'+
' </div>'+
'</div>';
this.container = $(html); this.find(".crop-preview").css({
width : (this.cropWidth + 4) +"px",
height: (this.cropHeight + 4) + "px",
"border-width": (this.offsetY -2) +"px " + (this.offsetX - 2) +"px"
}); this.find(".crop-preview-box").css({
width : (this.cropWidth) +"px",
height: (this.cropHeight) + "px"
}); parent.append(this.container);
},
bindEvent:function(){
this.cropTarget = this.find(".crop-target");
this.cropGrip = this.find(".crop-grip");
this.wrapper = this.find(".crop-wrapper");
var _this = this;
this.eventTarget.bind("click", function(){
_this.show();
}); this.bindCropGrip();
this.bindUpload();
this.find(".btn-zoom").bind("click", function(){
if($(this).hasClass("zoom-out")){
_this.scale(false);
}
else if($(this).hasClass("zoom-in")){
_this.scale(true);
}
}); var canvas = document.createElement("canvas");
canvas.width = this.destWidth;
canvas.height = this.destHeight; var context = canvas.getContext("2d");
this.find(".btn-confirm-crop").bind("click", function(){
var target = _this.cropTarget.get(0);
var top = parseInt(target.style.top);
var left = parseInt(target.style.left);
var offsetLeft = left - _this.originalLeft;
var offsetTop = top - _this.origianlTop;
var scale = _this.zoom;
var _left = (_this.zoom * _this.originalWidth - _this.cropWidth) / 2;
var _top = (_this.zoom * _this.originalHeight - _this.cropHeight) / 2;
var x = (-_left + offsetLeft) / scale;
var y = (-_top + offsetTop) /scale;
context.clearRect(0, 0, canvas.width, canvas.height);
context.save();
context.scale(scale, scale);
context.drawImage(_this.cropTarget.get(0), x, y);
context.restore();
var base64Url = canvas.toDataURL("image/png");
_this.find(".image-preview").find("img").attr("src", base64Url);
_this.onConfirm(base64Url)
}); },
bindUpload:function(){
var _this = this; this.fileInput = $('<input type="file" accept="image/*" style="position: absolute;left: -999999px;" />');
this.fileInput.bind("change", function(e){
if($.trim($(this).val())==""){
return ;
}
var files = e.target.files || e.dataTransfer.files;
var file = files[0]; if(!/^image\//.test(file.type)){
alert("请上传图片文件");
_this.fileInput.val("");
return false;
} //对路径进行压缩
var reader = new FileReader();
reader.onload = function(e) {
var image = new Image();
image.onload = function(){
_this.cropTarget.attr("src", this.src);
_this.initImage(this.width, this.height); };
image.src = e.target.result;
}
reader.readAsDataURL(file);
}); var _this = this;
this.find(".btn-upload-image").bind("click", function(){
_this.fileInput.trigger("click");
});
},
initImage:function(width, height){ var _this = this;
var top = -(height - this.wrapHeight) / 2;
var left = -(width - this.wrapWidth) / 2;
this.originalLeft = left;
this.origianlTop = top;
this.originalWidth = width;
this.originalHeight = height;
this.cropTarget.get(0).style.transform = "";
this.cropTarget.css({
top:top,
left:left
});
this.zoomMin = 1;
this.gripPos = 5;
this.zoom = 1;
if (width > this.cropWidth) {
this.zoomMin = this.cropWidth / width;
}
this.zoomMax = this.zoomMin > 0.25 ? 8.0 : 4.0 / Math.sqrt(this.zoomMin);
this.cropGrip.css({
"left":(this.gripMinX + (this.gripPos / 10 * (this.gripMaxX - this.gripMinX))) + "px"
}); var target = this.cropTarget.get(0);
Drag.init(this.wrapper.get(0), target);
target.onDrag = function(){
_this.rePosition();
};
},
rePosition:function(){
var target = this.cropTarget.get(0);
var width = this.originalWidth;
var height = this.originalHeight;
var offsetLeft = parseInt(this.cropTarget.css("left"));
var offsetTop = parseInt(this.cropTarget.css("top"));
var _width = width * this.zoom;
var _height = height * this.zoom;
var _x = (_width - width ) / 2;
var _y = (_height - height) / 2; var maxX = Math.max(this.offsetX, this.offsetX + this.cropWidth - _width) + _x;
var minX = Math.min(this.offsetX + this.cropWidth - _width, this.offsetX) + _x; if (offsetLeft > maxX) target.style.left = maxX + 'px';
else if (offsetLeft < minX) target.style.left = minX + 'px'; var maxY = Math.max(this.offsetY, this.offsetY + this.cropHeight - _height) + _y;
var minY = Math.min(this.offsetY + this.cropHeight - _height, this.offsetY) + _y; if (offsetTop > maxY) target.style.top = maxY + 'px';
else if (offsetTop < minY) target.style.top = minY + 'px';
},
scale:function(flag){
if (flag) {
this.zoom = this.zoom * 1.5;
} else {
this.zoom = this.zoom / 1.5;
}
if (this.zoom < this.zoomMin) this.zoom = this.zoomMin;
if (this.zoom > this.zoomMax) this.zoom = this.zoomMax;
this.imageResize(this.zoom);
this.gripPos = 5 * (Math.log(this.zoom * this.zoomMax) / Math.log(this.zoomMax));
this.cropGrip.css({
left:(this.gripMinX + (this.gripPos / 10 * (this.gripMaxX - this.gripMinX))) + "px"
});
this.rePosition();
},
bindCropGrip:function(){
var _this = this; var cropGrip = this.cropGrip.get(0);
Drag.init(cropGrip, cropGrip);
cropGrip.onDrag = function (clientX, clientY) {
var offsetX = clientX;
if (clientX < _this.gripMinX) {
cropGrip.style.left = _this.gripMinX + "px";
offsetX = _this.gripMinX;
}
if (clientX > _this.gripMaxX) {
cropGrip.style.left = _this.gripMaxX + "px";
offsetX = _this.gripMaxX;
}
cropGrip.style.top = 0;
var x = (offsetX - _this.gripMinX) * 10 / (_this.gripMaxX - _this.gripMinX);
var zoom = Math.pow(_this.zoomMax, x / 5) / _this.zoomMax;
if (zoom < _this.zoomMin) zoom = _this.zoomMin;
if (zoom > _this.zoomMax) zoom = _this.zoomMax;
_this.imageResize(zoom);
_this.rePosition();
}
},
imageResize:function(zoom){
this.zoom = zoom;
this.cropTarget.get(0).style.transform = "scale("+zoom+")";
},
show:function(){
this.container.show();
},
hide:function(){
this.container.hide();
},
find:function(selector){
return $(selector, this.container);
}
}; $.fn.imageCrop = function(config){
return this.each(function(){
new ImageCrop($.extend({
target:$(this)
},config||{}));
});
}; createStyle('.crop-container{width:480px;height:300px;overflow:hidden;margin:10px auto;border:1px solid #e0e0e0;border-radius:5px;padding:10px}.zoom-tool{background-image:url();background-repeat:no-repeat}.left-box{float:left;height:320px;width:292px;overflow:hidden}.crop-wrapper{border:2px solid #888;width:280px;height:260px;overflow:hidden;position:relative;margin:4px;cursor:pointer;float:left}.crop-preview{width:144px;height:144px;border-color:#ccc;border-style:solid;border-width:58px 68px;position:absolute;z-index:1;opacity:.75}.crop-preview-box{width:140px;height:140px;border:2px solid #e0e0e0}.crop-tool-bar{width:252px;padding:4px 20px;height:20px;position:relative;overflow:hidden}.bar-zoom-box{position:relative;width:200px;display:inline-block}.crop-grip{position:absolute;z-index:100;left:100%;top:0;cursor:pointer;width:11px;height:18px;display:inline-block;background-position:-287px 0}.right-box{float:right;width:144px;padding:8px;height:280px}.right-box .image-preview{height:166px;width:144px;margin:8px auto;overflow:hidden;text-align:center;}.image-preview img{border:2px #eee solid;max-height:100%;max-width:140px;}.right-box .btn-box{height:50px;text-align:center}.right-box .btn-box a{display:inline-block;width:100px;padding:8px;text-align:center;border-radius:8px;text-decoration:none;border:1px solid #efefef;color:#FFF;font-size:14px;background:#80b6e7}.right-box .btn-box a:hover{background:#5498db}.btn-zoom{width:18px;height:18px;display:inline-block}.bar-zoom{width:200px;height:18px;display:inline-block}.zoom-out{background-position:-224px 0}.zoom-out:hover{background-position:-203px 0}.zoom-in{background-position:-266px 0}.zoom-in:hover{background-position:-245px 0}'); })(jQuery);
Jquery插件-Html5图片上传并裁剪的更多相关文章
- JQuery插件:图片上传本地预览插件,改进案例一则。
/* *名称:图片上传本地预览插件 v1.1 *作者:周祥 *时间:2013年11月26日 *介绍:基于JQUERY扩展,图片上传预览插件 目前兼容浏览器(IE 谷歌 火狐) 不支持safari *插 ...
- jquery插件fileupload图片上传(前端如何处理)
1.页面首先引入jquery,版本不要低于1.6 <script src="../js/jquery.min.js"></script>2.其次页面引入对应 ...
- 基于cropper.js的图片上传和裁剪
项目中要求图片上传并裁剪的功能,之前也有接触过很多图片裁剪插件,效果体验不是很好,今天推荐一款好用的插件-cropper,超级好用,裁剪功能丰富,满足了各种需求. 功能: 1:点击选择图片,弹出文件夹 ...
- 【转】JQuery插件ajaxFileUpload 异步上传文件(PHP版)
前几天想在手机端做个异步上传图片的功能,平时用的比较多的JQuery图片上传插件是Uploadify这个插件,效果很不错,但是由于手机不支持flash,所以不得不再找一个文件上传插件来用了.后来发现a ...
- HTML5 图片上传预览
<!DOCTYPE html> <html lang="zh-cn"> <head> <meta charset="utf-8& ...
- JQuery插件ajaxFileUpload 异步上传文件(PHP版)
太久没写博客了,真的是太忙了.善于总结,进步才会更快啊.不多说,直接进入主题. 前几天想在手机端做个异步上传图片的功能,平时用的比较多的JQuery图片上传插件是Uploadify这个插件,效果很不错 ...
- jQuery插件AjaxFileUpload文件上传实现Javascript多文件上传功能
Ajax file upload plugin是一个功能强大的文件上传jQuery插件,可自定义链接.或其它元素庖代传统的file表单上传结果,可实现Ajax动态提示文件上传 过程,同时支撑多文 ...
- Uploadify & jQuery.imgAreaSelect 插件实现图片上传裁剪
在网站中需要一个图片上传裁剪的功能,借鉴这篇文章 Ajax+PHP+jQuery图片截图上传 的指点,找到了jquery.imgAreaSelect这个不错插件,能对图片进行自定义区域选择并给出坐标, ...
- 网站开发常用jQuery插件总结(15)上传插件blueimp
在介绍这个插件之前,先吐槽一下.这个插件功能很强大.带有的功能有:上传(单个文件或批量文件),自动上传或点击按钮上传,上传前缩略图显示,判断文件格式,上传前的文件操作,上传时进度条显示等功能.如果你用 ...
随机推荐
- Eclipse中使用Maven创建Servlet3.0 Web 项目
摘要 Apache Maven是一个优秀的项目构建和管理工具,许多开源项目都使用Maven进行构建.由于最近工作中要用到Maven,于是这里记录下在Eclipse中使用Maven插件创建一个基于Ser ...
- PHP mongoDB 操作
<?php /** * PHP操作MongoDB学习笔记 */ //************************* //** 连接MongoDB数据库 **// //************ ...
- 基类中定义的虚函数在派生类中重新定义时,其函数原型,包括返回类型、函数名、参数个数、参数类型及参数的先后顺序,都必须与基类中的原型完全相同 but------> 可以返回派生类对象的引用或指针
您查询的关键词是:c++primer习题15.25 以下是该网页在北京时间 2016年07月15日 02:57:08 的快照: 如果打开速度慢,可以尝试快速版:如果想更新或删除快照,可以投诉快照. ...
- laravel框架——保存用户登陆信息(session)
public function inlog(Request $request) { //获取表单提交的数据 $input = $request->all(); //根本获取的数据去数据库中查询 ...
- 随机数(random)
在测试你的程序是否超时时,可以随机生成一组大数据,进行一下测试. 当然如果你考场上一道题直接读不懂不会做的时候,可以random一下,拼一下RP嘛.2333. #include<cstdio&g ...
- 怎样在Swift中使用CocoaPods-b
最近关于CocoaPods有很多的议论.你可能从别的开发者那里听到过,或者在Github的目录中看到过.如果你之前从来没有用过,你可能会问,"CocoaPods到底是什么?" 它不 ...
- Intra Chroma Prediction
帧内预测依赖于当前宏块的相邻宏块,如果任何一个相邻宏块不可用,那么会直接影响到当前宏块的预测方式. 那么宏块怎么才谓之可用? 满足以下几个条件的相邻宏块为不可用: 相邻宏块超出边界,即(x<0 ...
- Chapter 7 Backup and Recovery 备份和恢复:
Chapter 7 Backup and Recovery 备份和恢复: Table of Contents 7.1 Backup and Recovery Types 7.2 Database Ba ...
- PLSQL死循环
begin loop insert into for1 values('a'); commit; end loop; end;
- 两个bootstrap插件bootstrap-select和bootstrap-paginator
基于bootstrap的选择器 http://silviomoreto.github.io/bootstrap-select/ <label for="androids" c ...