From 4f70edc8c1d78dce186bd8c2550d97ba8be1eb65 Mon Sep 17 00:00:00 2001 From: Anton Lavrenov Date: Tue, 11 Sep 2018 16:15:42 +0300 Subject: [PATCH] Centered resize with ALT key for `Konva.Transformer` --- CHANGELOG.md | 4 + konva.js | 70 +++++++---- konva.min.js | 4 +- src/shapes/Transformer.js | 53 +++++---- test/unit/shapes/Transformer-test.js | 171 +++++++++++++++++++++++++++ 5 files changed, 251 insertions(+), 51 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4d3a9ca5..acbd303d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,6 +9,10 @@ This project adheres to [Semantic Versioning](http://semver.org/). * Tween support for gradient properties +### Added + +* Centered resize with ALT key for `Konva.Transformer` + ## [2.3.0][2018-08-30] ### Added diff --git a/konva.js b/konva.js index 853a5171..4f3d9a43 100644 --- a/konva.js +++ b/konva.js @@ -2,7 +2,7 @@ * Konva JavaScript Framework v2.3.0 * http://konvajs.github.io/ * Licensed under the MIT - * Date: Mon Sep 10 2018 + * Date: Tue Sep 11 2018 * * Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS) * Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva) @@ -19565,7 +19565,7 @@ this.____init(config); }; - var RESIZERS_NAMES = [ + var ANCHORS_NAMES = [ 'top-left', 'top-center', 'top-right', @@ -19721,7 +19721,7 @@ _createElements: function() { this._createBack(); - RESIZERS_NAMES.forEach( + ANCHORS_NAMES.forEach( function(name) { this._createAnchor(name); }.bind(this) @@ -19758,25 +19758,8 @@ var layer = this.getLayer(); var tr = this.getParent(); - // TODO: I guess there are some ways to simplify that calculations - // the basic idea is to find "angle" of handler var rad = Konva.getAngle(tr.rotation()); - // var cdx = tr.getWidth() / 2; - // var cdy = tr.getHeight() / 2; - - // var parentPos = tr.getAbsolutePosition(tr.getParent()); - // var center = { - // x: parentPos.x + (cdx * Math.cos(rad) + cdy * Math.sin(-rad)), - // y: parentPos.y + (cdy * Math.cos(rad) + cdx * Math.sin(rad)) - // }; - - // var pos = this.getAbsolutePosition(tr.getParent()); - - // var dx = -pos.x + center.x; - // var dy = -pos.y + center.y; - - // var angle = -Math.atan2(-dy, dx) - Math.PI / 2; var scale = tr.getNode().getAbsoluteScale(); // If scale.y < 0 xor scale.x < 0 we need to flip (not rotate). var isMirrored = scale.y * scale.x < 0; @@ -20014,6 +19997,43 @@ this.getParent() ); + // we need to refactor that centeredScaling ASAP! + if (e.altKey) { + var topLeft = this.findOne('.top-left'); + var bottomRight = this.findOne('.bottom-right'); + var topOffsetX = topLeft.x(); + var topOffsetY = topLeft.y(); + + var bottomOffsetX = this.getWidth() - bottomRight.x(); + var bottomOffsetY = this.getHeight() - bottomRight.y(); + + bottomRight.move({ + x: -topOffsetX, + y: -topOffsetY + }); + + topLeft.move({ + x: bottomOffsetX, + y: bottomOffsetY + }); + + absPos = topLeft.getAbsolutePosition(this.getParent()); + + // if (topOffsetX) { + // } + // var pos = this.findOne('.top-left').position(); + // if (pos.x === 0 && pos.y === 0) { + // this.findOne('.top-left').setAttrs({ + // x: this.getWidth() - this.findOne('.bottom-right').x(), + // y: this.getWidth() - this.findOne('.bottom-right').y() + // }); + // absPos = this.findOne('.top-left').getAbsolutePosition( + // this.getParent() + // ); + // } else { + // } + } + x = absPos.x; y = absPos.y; var width = @@ -20233,12 +20253,12 @@ } if (val instanceof Array) { val.forEach(function(name) { - if (RESIZERS_NAMES.indexOf(name) === -1) { + if (ANCHORS_NAMES.indexOf(name) === -1) { Konva.Util.warn( - 'Unknown resizer name: ' + + 'Unknown anchor name: ' + name + '. Available names are: ' + - RESIZERS_NAMES.join(', ') + ANCHORS_NAMES.join(', ') ); } }); @@ -20263,7 +20283,7 @@ Konva.Factory.addGetterSetter( Konva.Transformer, 'enabledAnchors', - RESIZERS_NAMES, + ANCHORS_NAMES, validateResizers ); @@ -20487,7 +20507,7 @@ Konva.Factory.addGetterSetter(Konva.Transformer, 'borderDash'); /** - * get/set should we keep ration of resize? + * get/set should we keep ratio while resize anchors at corners * @name keepRatio * @method * @memberof Konva.Transformer.prototype diff --git a/konva.min.js b/konva.min.js index da614527..5de42aee 100644 --- a/konva.min.js +++ b/konva.min.js @@ -2,7 +2,7 @@ * Konva JavaScript Framework v2.3.0 * http://konvajs.github.io/ * Licensed under the MIT - * Date: Mon Sep 10 2018 + * Date: Tue Sep 11 2018 * * Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS) * Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva) @@ -23,4 +23,4 @@ * @example * node.cache(); * node.filters([Konva.Filters.Sepia]); - */Konva.Filters.Sepia=function(t){var e,i,n,a,r,o,s,h,l,c=t.data,d=t.width,u=t.height,g=4*d;do{for(e=(u-1)*g,i=d;s=.393*(a=c[n=e+4*(i-1)])+.769*(r=c[n+1])+.189*(o=c[n+2]),h=.349*a+.686*r+.168*o,l=.272*a+.534*r+.131*o,c[n]=255this.duration?this.yoyo?(this._time=this.duration,this.reverse()):this.finish():t<0?this.yoyo?(this._time=0,this.play()):this.reset():(this._time=t,this.update())},getTime:function(){return this._time},setPosition:function(t){this.prevPos=this._pos,this.propFunc(t),this._pos=t},getPosition:function(t){return void 0===t&&(t=this._time),this.func(t,this.begin,this._change,this.duration)},play:function(){this.state=2,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onPlay")},reverse:function(){this.state=3,this._time=this.duration-this._time,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onReverse")},seek:function(t){this.pause(),this._time=t,this.update(),this.fire("onSeek")},reset:function(){this.pause(),this._time=0,this.update(),this.fire("onReset")},finish:function(){this.pause(),this._time=this.duration,this.update(),this.fire("onFinish")},update:function(){this.setPosition(this.getPosition(this._time))},onEnterFrame:function(){var t=this.getTimer()-this._startTime;2===this.state?this.setTime(t):3===this.state&&this.setTime(this.duration-t)},pause:function(){this.state=1,this.fire("onPause")},getTimer:function(){return(new Date).getTime()}},Konva.Tween=function(t){var e,i,n=this,a=t.node,r=a._id,o=t.easing||Konva.Easings.Linear,s=!!t.yoyo;e=void 0===t.duration?.3:0===t.duration?.001:t.duration,this.node=a,this._id=c++;var h=a.getLayer()||(a instanceof Konva.Stage?a.getLayers():null);for(i in h||Konva.Util.error("Tween constructor have `node` that is not in a layer. Please add node into layer first."),this.anim=new Konva.Animation(function(){n.tween.onEnterFrame()},h),this.tween=new d(i,function(t){n._tweenFunc(t)},o,0,1,1e3*e,s),this._addListeners(),Konva.Tween.attrs[r]||(Konva.Tween.attrs[r]={}),Konva.Tween.attrs[r][this._id]||(Konva.Tween.attrs[r][this._id]={}),Konva.Tween.tweens[r]||(Konva.Tween.tweens[r]={}),t)void 0===l[i]&&this._addAttr(i,t[i]);this.reset(),this.onFinish=t.onFinish,this.onReset=t.onReset},Konva.Tween.attrs={},Konva.Tween.tweens={},Konva.Tween.prototype={_addAttr:function(t,e){var i,n,a,r,o,s,h,l,c=this.node,d=c._id;if((a=Konva.Tween.tweens[d][t])&&delete Konva.Tween.attrs[d][a][t],i=c.getAttr(t),Konva.Util._isArray(e))if(n=[],o=Math.max(e.length,i.length),"points"===t&&e.length!==i.length&&(e.length>i.length?(h=i,i=Konva.Util._prepareArrayForTween(i,e,c.closed())):(s=e,e=Konva.Util._prepareArrayForTween(e,i,c.closed()))),0===t.indexOf("fill"))for(r=0;r>>1,F=_.slice(0,w+1),T=this._getTextWidth(F)+y;T<=l?(S=w+1,x=F+(v?"…":""),C=T):b=w}if(!x)break;if(f){var P,A=_[x.length];0<(P=(" "===A||"-"===A)&&C<=l?x.length:Math.max(x.lastIndexOf(" "),x.lastIndexOf("-"))+1)&&(S=P,x=x.slice(0,S),C=this._getTextWidth(x))}if(this._addTextLine(x),i=Math.max(i,C),d+=n,!g||s&&cthis.dataArray[i].pathLength;)t-=this.dataArray[i].pathLength,++i;if(i===n)return{x:(e=this.dataArray[i-1].points.slice(-2))[0],y:e[1]};if(t<.01)return{x:(e=this.dataArray[i].points.slice(0,2))[0],y:e[1]};var a=this.dataArray[i],r=a.points;switch(a.command){case"L":return Konva.Path.getPointOnLine(t,a.start.x,a.start.y,r[0],r[1]);case"C":return Konva.Path.getPointOnCubicBezier(t/a.pathLength,a.start.x,a.start.y,r[0],r[1],r[2],r[3],r[4],r[5]);case"Q":return Konva.Path.getPointOnQuadraticBezier(t/a.pathLength,a.start.x,a.start.y,r[0],r[1],r[2],r[3]);case"A":var o=r[0],s=r[1],h=r[2],l=r[3],c=r[4],d=r[5],u=r[6];return c+=d*t/a.pathLength,Konva.Path.getPointOnEllipticalArc(o,s,h,l,c,u)}return null}},Konva.Util.extend(Konva.Path,Konva.Shape),Konva.Path.getLineLength=function(t,e,i,n){return Math.sqrt((i-t)*(i-t)+(n-e)*(n-e))},Konva.Path.getPointOnLine=function(t,e,i,n,a,r,o){void 0===r&&(r=e),void 0===o&&(o=i);var s=(a-i)/(n-e+1e-8),h=Math.sqrt(t*t/(1+s*s));ne?v=Konva.Path.getPointOnLine(e,f.x,f.y,p.points[0],p.points[1],f.x,f.y):p=void 0;break;case"A":var o=p.points[4],s=p.points[5],h=p.points[4]+s;0===_?_=o+1e-8:ip.pathLength?1e-8:e/p.pathLength:ithis.duration?this.yoyo?(this._time=this.duration,this.reverse()):this.finish():t<0?this.yoyo?(this._time=0,this.play()):this.reset():(this._time=t,this.update())},getTime:function(){return this._time},setPosition:function(t){this.prevPos=this._pos,this.propFunc(t),this._pos=t},getPosition:function(t){return void 0===t&&(t=this._time),this.func(t,this.begin,this._change,this.duration)},play:function(){this.state=2,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onPlay")},reverse:function(){this.state=3,this._time=this.duration-this._time,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onReverse")},seek:function(t){this.pause(),this._time=t,this.update(),this.fire("onSeek")},reset:function(){this.pause(),this._time=0,this.update(),this.fire("onReset")},finish:function(){this.pause(),this._time=this.duration,this.update(),this.fire("onFinish")},update:function(){this.setPosition(this.getPosition(this._time))},onEnterFrame:function(){var t=this.getTimer()-this._startTime;2===this.state?this.setTime(t):3===this.state&&this.setTime(this.duration-t)},pause:function(){this.state=1,this.fire("onPause")},getTimer:function(){return(new Date).getTime()}},Konva.Tween=function(t){var e,i,n=this,a=t.node,r=a._id,o=t.easing||Konva.Easings.Linear,s=!!t.yoyo;e=void 0===t.duration?.3:0===t.duration?.001:t.duration,this.node=a,this._id=c++;var h=a.getLayer()||(a instanceof Konva.Stage?a.getLayers():null);for(i in h||Konva.Util.error("Tween constructor have `node` that is not in a layer. Please add node into layer first."),this.anim=new Konva.Animation(function(){n.tween.onEnterFrame()},h),this.tween=new d(i,function(t){n._tweenFunc(t)},o,0,1,1e3*e,s),this._addListeners(),Konva.Tween.attrs[r]||(Konva.Tween.attrs[r]={}),Konva.Tween.attrs[r][this._id]||(Konva.Tween.attrs[r][this._id]={}),Konva.Tween.tweens[r]||(Konva.Tween.tweens[r]={}),t)void 0===l[i]&&this._addAttr(i,t[i]);this.reset(),this.onFinish=t.onFinish,this.onReset=t.onReset},Konva.Tween.attrs={},Konva.Tween.tweens={},Konva.Tween.prototype={_addAttr:function(t,e){var i,n,a,r,o,s,h,l,c=this.node,d=c._id;if((a=Konva.Tween.tweens[d][t])&&delete Konva.Tween.attrs[d][a][t],i=c.getAttr(t),Konva.Util._isArray(e))if(n=[],o=Math.max(e.length,i.length),"points"===t&&e.length!==i.length&&(e.length>i.length?(h=i,i=Konva.Util._prepareArrayForTween(i,e,c.closed())):(s=e,e=Konva.Util._prepareArrayForTween(e,i,c.closed()))),0===t.indexOf("fill"))for(r=0;r>>1,F=_.slice(0,w+1),T=this._getTextWidth(F)+y;T<=l?(S=w+1,x=F+(v?"…":""),C=T):b=w}if(!x)break;if(f){var P,A=_[x.length];0<(P=(" "===A||"-"===A)&&C<=l?x.length:Math.max(x.lastIndexOf(" "),x.lastIndexOf("-"))+1)&&(S=P,x=x.slice(0,S),C=this._getTextWidth(x))}if(this._addTextLine(x),i=Math.max(i,C),d+=n,!g||s&&cthis.dataArray[i].pathLength;)t-=this.dataArray[i].pathLength,++i;if(i===n)return{x:(e=this.dataArray[i-1].points.slice(-2))[0],y:e[1]};if(t<.01)return{x:(e=this.dataArray[i].points.slice(0,2))[0],y:e[1]};var a=this.dataArray[i],r=a.points;switch(a.command){case"L":return Konva.Path.getPointOnLine(t,a.start.x,a.start.y,r[0],r[1]);case"C":return Konva.Path.getPointOnCubicBezier(t/a.pathLength,a.start.x,a.start.y,r[0],r[1],r[2],r[3],r[4],r[5]);case"Q":return Konva.Path.getPointOnQuadraticBezier(t/a.pathLength,a.start.x,a.start.y,r[0],r[1],r[2],r[3]);case"A":var o=r[0],s=r[1],h=r[2],l=r[3],c=r[4],d=r[5],u=r[6];return c+=d*t/a.pathLength,Konva.Path.getPointOnEllipticalArc(o,s,h,l,c,u)}return null}},Konva.Util.extend(Konva.Path,Konva.Shape),Konva.Path.getLineLength=function(t,e,i,n){return Math.sqrt((i-t)*(i-t)+(n-e)*(n-e))},Konva.Path.getPointOnLine=function(t,e,i,n,a,r,o){void 0===r&&(r=e),void 0===o&&(o=i);var s=(a-i)/(n-e+1e-8),h=Math.sqrt(t*t/(1+s*s));ne?v=Konva.Path.getPointOnLine(e,f.x,f.y,p.points[0],p.points[1],f.x,f.y):p=void 0;break;case"A":var o=p.points[4],s=p.points[5],h=p.points[4]+s;0===_?_=o+1e-8:ip.pathLength?1e-8:e/p.pathLength:i { + rect.setAttrs({ + x: 0, + y: 0, + width: 100, + height: 100, + scaleX: 1, + scaleY: 1 + }); + tr.update(); + + layer.draw(); + + stage.simulateMouseDown(test.startPos); + + var target = stage.getIntersection(test.startPos); + var top = stage.content.getBoundingClientRect().top; + tr._handleMouseMove({ + target: target, + clientX: test.endPos.x, + clientY: test.endPos.y + top, + altKey: true + }); + + // here is duplicate, because transformer is listening window events + tr._handleMouseUp({ + clientX: test.endPos.x, + clientY: test.endPos.y + top + }); + stage.simulateMouseUp({ + x: test.endPos.x, + y: test.endPos.y + }); + layer.draw(); + + assert.equal( + rect.width() * rect.scaleX(), + test.expectedWidth, + test.name + ' width check' + ); + assert.equal( + rect.height() * rect.scaleY(), + test.expectedHeight, + test.name + ' height check' + ); + }); + }); });