123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560 |
- var cropper = {
- CUT_START: null,
- cutX: 0, //画布x轴起点
- cutY: 0, //画布y轴起点0
- touchRelative: [{
- x: 0,
- y: 0
- }], //手指或鼠标和图片中心的相对位置
- flagCutTouch: false, //是否是拖动裁剪框
- hypotenuseLength: 0, //双指触摸时斜边长度
- flagEndTouch: false, //是否结束触摸
- canvasWidth: 0,
- canvasHeight: 0,
- imgWidth: 0, //图片宽度
- imgHeight: 0, //图片高度
- scale: 1, //图片缩放比
- angle: 0, //图片旋转角度
- imgTop: 0, //图片上边距
- imgLeft: 0, //图片左边距
- //是否限制移动范围(剪裁框只能在图片内,为true不可触摸转动图片)
- limitMove: true,
- minHeight: 0,
- maxHeight: 0,
- minWidth: 0,
- maxWidth: 0,
- windowHeight: 0,
- windowWidth: 0,
- init: true
- }
- function bool(str) {
- return str === 'true' || str == true ? true : false
- }
- function touchstart(e, ins) {
- //var instance = e.instance;
- // var state = instance.getState();
- var touch = e.touches || e.changedTouches;
- cropper.flagEndTouch = false;
- if (touch.length == 1) {
- cropper.touchRelative[0] = {
- x: touch[0].pageX - cropper.imgLeft,
- y: touch[0].pageY - cropper.imgTop
- };
- } else {
- var width = Math.abs(touch[0].pageX - touch[1].pageX);
- var height = Math.abs(touch[0].pageY - touch[1].pageY);
- cropper.touchRelative = [{
- x: touch[0].pageX - cropper.imgLeft,
- y: touch[0].pageY - cropper.imgTop
- },
- {
- x: touch[1].pageX - cropper.imgLeft,
- y: touch[1].pageY - cropper.imgTop
- }
- ];
- cropper.hypotenuseLength = Math.sqrt(Math.pow(width, 2) + Math.pow(height, 2));
- }
- }
- function moveDuring(ins) {
- if (!ins) return;
- ins.callMethod('moveDuring')
- }
- function moveStop(ins) {
- if (!ins) return;
- ins.callMethod('moveStop')
- };
- function setCutCenter(ins) {
- var cutY = (cropper.windowHeight - cropper.canvasHeight) * 0.5;
- var cutX = (cropper.windowWidth - cropper.canvasWidth) * 0.5;
- //顺序不能变
- cropper.imgTop = cropper.imgTop - cropper.cutY + cutY;
- cropper.cutY = cutY; //截取的框上边距
- cropper.imgLeft = cropper.imgLeft - cropper.cutX + cutX;
- cropper.cutX = cutX; //截取的框左边距
- styleUpdate(ins)
- cutDetectionPosition(ins)
- imgTransform(ins)
- updateData(ins)
- }
- function touchmove(e, ins) {
- var touch = e.touches || e.changedTouches;
- if (cropper.flagEndTouch) return;
- moveDuring(ins);
- if (e.touches.length == 1) {
- var left = touch[0].pageX - cropper.touchRelative[0].x,
- top = touch[0].pageY - cropper.touchRelative[0].y;
- cropper.imgLeft = left;
- cropper.imgTop = top;
- imgTransform(ins);
- imgMarginDetectionPosition(ins);
- } else {
- var res = e.instance.getDataset();
- var minScale = +res.minscale;
- var maxScale = +res.maxscale;
- var disableRotate = bool(res.disablerotate)
- var width = Math.abs(touch[0].pageX - touch[1].pageX),
- height = Math.abs(touch[0].pageY - touch[1].pageY),
- hypotenuse = Math.sqrt(Math.pow(width, 2) + Math.pow(height, 2)),
- scale = cropper.scale * (hypotenuse / cropper.hypotenuseLength),
- current_deg = 0;
- scale = scale <= minScale ? minScale : scale;
- scale = scale >= maxScale ? maxScale : scale;
- cropper.scale = scale;
- imgMarginDetectionScale(ins, true);
- var touchRelative = [{
- x: touch[0].pageX - cropper.imgLeft,
- y: touch[0].pageY - cropper.imgTop
- },
- {
- x: touch[1].pageX - cropper.imgLeft,
- y: touch[1].pageY - cropper.imgTop
- }
- ];
- if (!disableRotate) {
- var first_atan = (180 / Math.PI) * Math.atan2(touchRelative[0].y, touchRelative[0].x);
- var first_atan_old = (180 / Math.PI) * Math.atan2(cropper.touchRelative[0].y, cropper.touchRelative[0].x);
- var second_atan = (180 / Math.PI) * Math.atan2(touchRelative[1].y, touchRelative[1].x);
- var second_atan_old = (180 / Math.PI) * Math.atan2(cropper.touchRelative[1].y, cropper.touchRelative[1].x);
- var first_deg = first_atan - first_atan_old,
- second_deg = second_atan - second_atan_old;
- if (first_deg != 0) {
- current_deg = first_deg;
- } else if (second_deg != 0) {
- current_deg = second_deg;
- }
- }
- cropper.touchRelative = touchRelative;
- cropper.hypotenuseLength = Math.sqrt(Math.pow(width, 2) + Math.pow(height, 2));
- //更新视图
- cropper.angle = cropper.angle + current_deg;
- imgTransform(ins);
- }
- }
- function touchend(e, ins) {
- cropper.flagEndTouch = true;
- moveStop(ins);
- updateData(ins)
- }
- function cutTouchStart(e, ins) {
- var touch = e.touches || e.changedTouches;
- var currentX = touch[0].pageX;
- var currentY = touch[0].pageY;
- /*
- * (右下-1 右上-2 左上-3 左下-4)
- * left_x [3,4]
- * top_y [2,3]
- * right_x [1,2]
- * bottom_y [1,4]
- */
- var left_x1 = cropper.cutX - 30;
- var left_x2 = cropper.cutX + 30;
- var top_y1 = cropper.cutY - 30;
- var top_y2 = cropper.cutY + 30;
- var right_x1 = cropper.cutX + cropper.canvasWidth - 30;
- var right_x2 = cropper.cutX + cropper.canvasWidth + 30;
- var bottom_y1 = cropper.cutY + cropper.canvasHeight - 30;
- var bottom_y2 = cropper.cutY + cropper.canvasHeight + 30;
- if (currentX > right_x1 && currentX < right_x2 && currentY > bottom_y1 && currentY < bottom_y2) {
- moveDuring();
- cropper.flagCutTouch = true;
- cropper.flagEndTouch = true;
- cropper.CUT_START = {
- width: cropper.canvasWidth,
- height: cropper.canvasHeight,
- x: currentX,
- y: currentY,
- corner: 1
- };
- } else if (currentX > right_x1 && currentX < right_x2 && currentY > top_y1 && currentY < top_y2) {
- moveDuring();
- cropper.flagCutTouch = true;
- cropper.flagEndTouch = true;
- cropper.CUT_START = {
- width: cropper.canvasWidth,
- height: cropper.canvasHeight,
- x: currentX,
- y: currentY,
- cutY: cropper.cutY,
- cutX: cropper.cutX,
- corner: 2
- };
- } else if (currentX > left_x1 && currentX < left_x2 && currentY > top_y1 && currentY < top_y2) {
- moveDuring();
- cropper.flagCutTouch = true;
- cropper.flagEndTouch = true;
- cropper.CUT_START = {
- width: cropper.canvasWidth,
- height: cropper.canvasHeight,
- cutY: cropper.cutY,
- cutX: cropper.cutX,
- x: currentX,
- y: currentY,
- corner: 3
- };
- } else if (currentX > left_x1 && currentX < left_x2 && currentY > bottom_y1 && currentY < bottom_y2) {
- moveDuring();
- cropper.flagCutTouch = true;
- cropper.flagEndTouch = true;
- cropper.CUT_START = {
- width: cropper.canvasWidth,
- height: cropper.canvasHeight,
- cutY: cropper.cutY,
- cutX: cropper.cutX,
- x: currentX,
- y: currentY,
- corner: 4
- };
- }
- }
- function cutTouchMove(e, ins) {
- if (!cropper.CUT_START || cropper.CUT_START === 'null') return;
- if (cropper.flagCutTouch) {
- var touch = e.touches || e.changedTouches;
- var res = e.instance.getDataset();
- var lockRatio = bool(res.lockratio);
- var lockWidth = bool(res.lockwidth);
- var lockHeight = bool(res.lockheight);
- if (lockRatio && (lockWidth || lockHeight)) return;
- var width = cropper.canvasWidth,
- height = cropper.canvasHeight,
- cutY = cropper.cutY,
- cutX = cropper.cutX;
- var size_correct = function() {
- width = width <= cropper.maxWidth ? (width >= cropper.minWidth ? width : cropper.minWidth) : cropper.maxWidth;
- height = height <= cropper.maxHeight ? (height >= cropper.minHeight ? height : cropper.minHeight) : cropper.maxHeight;
- }
- var size_inspect = function() {
- if ((width > cropper.maxWidth || width < cropper.minWidth || height > cropper.maxHeight || height < cropper.minHeight) &&
- lockRatio) {
- size_correct();
- return false;
- } else {
- size_correct();
- return true;
- }
- };
- height = cropper.CUT_START.height + (cropper.CUT_START.corner > 1 && cropper.CUT_START.corner < 4 ? 1 : -1) * (
- cropper.CUT_START.y - touch[0].pageY);
- switch (cropper.CUT_START.corner) {
- case 1:
- width = cropper.CUT_START.width - cropper.CUT_START.x + touch[0].pageX;
- if (lockRatio) {
- height = width / (cropper.canvasWidth / cropper.canvasHeight);
- }
- if (!size_inspect()) return;
- break;
- case 2:
- width = cropper.CUT_START.width - cropper.CUT_START.x + touch[0].pageX;
- if (lockRatio) {
- height = width / (cropper.canvasWidth / cropper.canvasHeight);
- }
- if (!size_inspect()) return;
- cutY = cropper.CUT_START.cutY - (height - cropper.CUT_START.height);
- break;
- case 3:
- width = cropper.CUT_START.width + cropper.CUT_START.x - touch[0].pageX;
- if (lockRatio) {
- height = width / (cropper.canvasWidth / cropper.canvasHeight);
- }
- if (!size_inspect()) return;
- cutY = cropper.CUT_START.cutY - (height - cropper.CUT_START.height);
- cutX = cropper.CUT_START.cutX - (width - cropper.CUT_START.width);
- break;
- case 4:
- width = cropper.CUT_START.width + cropper.CUT_START.x - touch[0].pageX;
- if (lockRatio) {
- height = width / (cropper.canvasWidth / cropper.canvasHeight);
- }
- if (!size_inspect()) return;
- cutX = cropper.CUT_START.cutX - (width - cropper.CUT_START.width);
- break;
- default:
- break;
- }
- if (!lockWidth && !lockHeight) {
- cropper.canvasWidth = width;
- cropper.cutX = cutX;
- cropper.canvasHeight = height;
- cropper.cutY = cutY;
- canvasHeight(ins);
- canvasWidth(ins);
- } else if (!lockWidth) {
- cropper.canvasWidth = width;
- cropper.cutX = cutX;
- canvasWidth(ins);
- } else if (!lockHeight) {
- cropper.canvasHeight = height;
- cropper.cutY = cutY;
- canvasHeight(ins);
- }
- styleUpdate(ins)
- imgMarginDetectionScale(ins);
- }
- }
- //检测剪裁框位置是否在允许的范围内(屏幕内)
- function cutDetectionPosition(ins) {
- var windowHeight = cropper.windowHeight,
- windowWidth = cropper.windowWidth;
- var cutDetectionPositionTop = function() {
- //检测上边距是否在范围内
- if (cropper.cutY < 0) {
- cropper.cutY = 0;
- }
- if (cropper.cutY > windowHeight - cropper.canvasHeight) {
- cropper.cutY = windowHeight - cropper.canvasHeight;
- }
- }
- var cutDetectionPositionLeft = function() {
- //检测左边距是否在范围内
- if (cropper.cutX < 0) {
- cropper.cutX = 0;
- }
- if (cropper.cutX > windowWidth - cropper.canvasWidth) {
- cropper.cutX = windowWidth - cropper.canvasWidth;
- }
- }
- //裁剪框坐标处理(如果只写一个参数则另一个默认为0,都不写默认居中)
- if (cropper.cutY == null && cropper.cutX == null) {
- var cutY = (windowHeight - cropper.canvasHeight) * 0.5;
- var cutX = (windowWidth - cropper.canvasWidth) * 0.5;
- cropper.cutY = cutY; //截取的框上边距
- cropper.cutX = cutX; //截取的框左边距
- } else if (cropper.cutY != null && cropper.cutX != null) {
- cutDetectionPositionTop();
- cutDetectionPositionLeft();
- } else if (cropper.cutY != null && cropper.cutX == null) {
- cutDetectionPositionTop();
- cropper.cutX = (windowWidth - cropper.canvasWidth) / 2;
- } else if (cropper.cutY == null && cropper.cutX != null) {
- cutDetectionPositionLeft();
- cropper.cutY = (windowHeight - cropper.canvasHeight) / 2;
- }
- styleUpdate(ins)
- }
- /**
- * 图片边缘检测-缩放
- */
- function imgMarginDetectionScale(ins, delay) {
- if (!cropper.limitMove) return;
- var scale = cropper.scale;
- var imgWidth = cropper.imgWidth;
- var imgHeight = cropper.imgHeight;
- if ((cropper.angle / 90) % 2) {
- imgWidth = cropper.imgHeight;
- imgHeight = cropper.imgWidth;
- }
- if (imgWidth * scale < cropper.canvasWidth) {
- scale = cropper.canvasWidth / imgWidth;
- }
- if (imgHeight * scale < cropper.canvasHeight) {
- scale = Math.max(scale, cropper.canvasHeight / imgHeight);
- }
- imgMarginDetectionPosition(ins, scale, delay);
- }
- /**
- * 图片边缘检测-位置
- */
- function imgMarginDetectionPosition(ins, scale, delay) {
- if (!cropper.limitMove) return;
- var left = cropper.imgLeft;
- var top = cropper.imgTop;
- scale = scale || cropper.scale;
- var imgWidth = cropper.imgWidth;
- var imgHeight = cropper.imgHeight;
- if ((cropper.angle / 90) % 2) {
- imgWidth = cropper.imgHeight;
- imgHeight = cropper.imgWidth;
- }
- left = cropper.cutX + (imgWidth * scale) / 2 >= left ? left : cropper.cutX + (imgWidth * scale) / 2;
- left = cropper.cutX + cropper.canvasWidth - (imgWidth * scale) / 2 <= left ? left : cropper.cutX + cropper.canvasWidth -
- (imgWidth * scale) / 2;
- top = cropper.cutY + (imgHeight * scale) / 2 >= top ? top : cropper.cutY + (imgHeight * scale) / 2;
- top = cropper.cutY + cropper.canvasHeight - (imgHeight * scale) / 2 <= top ? top : cropper.cutY + cropper.canvasHeight -
- (imgHeight * scale) / 2;
- cropper.imgLeft = left;
- cropper.imgTop = top;
- cropper.scale = scale;
- styleUpdate(ins)
- if (!delay || delay === 'null') {
- imgTransform(ins);
- }
- }
- function cutTouchEnd(e, ins) {
- moveStop(ins);
- cropper.flagCutTouch = false;
- updateData(ins)
- }
- //改变截取框大小
- function computeCutSize(ins) {
- if (cropper.canvasWidth > cropper.windowWidth) {
- cropper.canvasWidth = cropper.windowWidth;
- // canvasWidth(ins)
- } else if (cropper.canvasWidth + cropper.cutX > cropper.windowWidth) {
- cropper.cutX = cropper.windowWidth - cropper.cutX;
- }
- if (cropper.canvasHeight > cropper.windowHeight) {
- cropper.canvasHeight = cropper.windowHeight;
- // canvasHeight(ins)
- } else if (cropper.canvasHeight + cropper.cutY > cropper.windowHeight) {
- cropper.cutY = cropper.windowHeight - cropper.cutY;
- }
- // styleUpdate(ins)
- }
- function styleUpdate(ins) {
- if (!ins) return;
- ins.selectComponent('.tui-cropper-box').setStyle({
- 'width': cropper.canvasWidth + 'px',
- 'height': cropper.canvasHeight + 'px'
- })
- ins.selectComponent('.tui-content-middle').setStyle({
- 'height': cropper.canvasHeight + 'px'
- })
- ins.selectComponent('.tui-content-top').setStyle({
- 'height': cropper.cutY + 'px'
- })
- ins.selectComponent('.tui-wxs-bg').setStyle({
- 'width': cropper.cutX + 'px'
- })
- }
- function imgTransform(ins) {
- var owner = ins.selectComponent('.tui-cropper-image')
- if (!owner) return
- var x = cropper.imgLeft - cropper.imgWidth / 2;
- var y = cropper.imgTop - cropper.imgHeight / 2;
- owner.setStyle({
- 'transform': 'translate3d(' + x + 'px,' + y + 'px,0) scale(' + cropper.scale + ') rotate(' + cropper.angle + 'deg)'
- })
- }
- function imageReset(ins) {
- cropper.scale = 1;
- cropper.angle = 0;
- imgTransform(ins);
- }
- //监听截取框宽高变化
- function canvasWidth(ins) {
- if (cropper.canvasWidth < cropper.minWidth) {
- cropper.canvasWidth = cropper.minWidth;
- }
- if (!ins) return;
- computeCutSize(ins);
- }
- function canvasHeight(ins) {
- if (cropper.canvasHeight < cropper.minHeight) {
- cropper.canvasHeight = cropper.minHeight;
- }
- if (!ins) return;
- computeCutSize(ins);
- }
- function updateData(ins) {
- if (!ins) return;
- ins.callMethod('change', {
- cutX: cropper.cutX,
- cutY: cropper.cutY,
- canvasWidth: cropper.canvasWidth,
- canvasHeight: cropper.canvasHeight,
- imgWidth: cropper.imgWidth,
- imgHeight: cropper.imgHeight,
- scale: cropper.scale,
- angle: cropper.angle,
- imgTop: cropper.imgTop,
- imgLeft: cropper.imgLeft
- })
- }
- function propsChange(prop, oldProp, ownerInstance, ins) {
- if (prop && prop !== 'null') {
- var params = prop.split(',')
- var type = +params[0]
- var dataset = ins.getDataset();
- if (cropper.init || type == 4) {
- cropper.maxHeight = +dataset.maxheight;
- cropper.minHeight = +dataset.minheight;
- cropper.maxWidth = +dataset.maxwidth;
- cropper.minWidth = +dataset.minwidth;
- cropper.canvasWidth = +dataset.width;
- cropper.canvasHeight = +dataset.height;
- cropper.imgTop = dataset.windowheight / 2;
- cropper.imgLeft = dataset.windowwidth / 2;
- cropper.imgWidth = +dataset.imgwidth;
- cropper.imgHeight = +dataset.imgheight;
- cropper.windowHeight = +dataset.windowheight;
- cropper.windowWidth = +dataset.windowwidth;
- cropper.init = false
- } else if (type == 2 || type == 3) {
- cropper.imgWidth = +dataset.imgwidth;
- cropper.imgHeight = +dataset.imgheight;
- }
- cropper.limitMove = bool(dataset.limitmove);
- cropper.angle = +dataset.angle;
- if (type == 3) {
- imgTransform(ownerInstance);
- }
- switch (type) {
- case 1:
- setCutCenter(ownerInstance);
- //设置裁剪框大小>设置图片尺寸>绘制canvas
- computeCutSize(ownerInstance);
- //检查裁剪框是否在范围内
- cutDetectionPosition(ownerInstance);
- break;
- case 2:
- setCutCenter(ownerInstance);
- break;
- case 3:
- imgMarginDetectionScale(ownerInstance)
- break;
- case 4:
- imageReset(ownerInstance);
- break;
- case 5:
- setCutCenter(ownerInstance);
- break;
- default:
- break;
- }
- }
- }
- module.exports = {
- touchstart: touchstart,
- touchmove: touchmove,
- touchend: touchend,
- cutTouchStart: cutTouchStart,
- cutTouchMove: cutTouchMove,
- cutTouchEnd: cutTouchEnd,
- propsChange: propsChange
- }
|