/** * 图片裁剪 * @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{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;292px;overflow:hidden}.crop-wrapper{border:2px solid #888;280px;height:260px;overflow:hidden;position:relative;margin:4px;cursor:pointer;float:left}.crop-preview{144px;height:144px;border-color:#ccc;border-style:solid;border-58px 68px;position:absolute;z-index:1;opacity:.75}.crop-preview-box{140px;height:140px;border:2px solid #e0e0e0}.crop-tool-bar{252px;padding:4px 20px;height:20px;position:relative;overflow:hidden}.bar-zoom-box{position:relative;200px;display:inline-block}.crop-grip{position:absolute;z-index:100;left:100%;top:0;cursor:pointer;11px;height:18px;display:inline-block;background-position:-287px 0}.right-box{float:right;144px;padding:8px;height:280px}.right-box .image-preview{height:166px;144px;margin:8px auto;overflow:hidden;text-align:center;}.image-preview img{border:2px #eee solid;max-height:100%;max-140px;}.right-box .btn-box{height:50px;text-align:center}.right-box .btn-box a{display:inline-block;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{18px;height:18px;display:inline-block}.bar-zoom{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);