From 1b065a55a0fd1d24b95e4bc65fbd08c22fd96d4c Mon Sep 17 00:00:00 2001 From: Anton Lavrenov Date: Tue, 19 Feb 2019 08:36:16 -0500 Subject: [PATCH] ixed automatic updates for Konva.Transformer --- CHANGELOG.md | 25 +- konva.js | 155 +++++----- konva.min.js | 4 +- src/Node.ts | 2 + src/Shape.ts | 1 - src/shapes/Arc.ts | 16 +- src/shapes/Circle.ts | 9 +- src/shapes/Ellipse.ts | 5 +- src/shapes/Line.ts | 1 + src/shapes/Path.ts | 1 + src/shapes/RegularPolygon.ts | 9 +- src/shapes/Ring.ts | 9 +- src/shapes/Sprite.ts | 1 - src/shapes/Star.ts | 9 +- src/shapes/Text.ts | 7 + src/shapes/TextPath.ts | 1 + src/shapes/Transformer.ts | 83 +++--- src/shapes/Wedge.ts | 9 +- test/unit/shapes/Transformer-test.js | 431 +++++++++++++++++++++++++++ 19 files changed, 600 insertions(+), 178 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index de5cb9fe..c312ec50 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,19 @@ This project adheres to [Semantic Versioning](http://semver.org/). ## [new version][unreleased] +### Possibly breaking + +That changes are private and internal specific. They should not break most of `Konva` apps. + +* `Konva.Util.addMethods` +* `Konva.Util._removeLastLetter` +* `Konva.Util._getImage` +* `Konv.Util._getRGBAString` +* `Konv.Util._merge` +* Removed polyfill for `requestAnimationFrame`. +* `id` and `name` properties defaults are empty strings, not `undefined` +* internal `_cache` property was updated to use es2015 `Map` instead of `{}`. + ### Added * Show a warning when a stage has too many layers * Show a warning on duplicate ids @@ -16,28 +29,20 @@ This project adheres to [Semantic Versioning](http://semver.org/). * You can configure what mouse buttons can be used for drag&drop. To enable right button you can use `Konva.dragButtons = [0, 1]`. ### Changed -* Fixes inconsistent `layer.setSize()` method. Now it has same arguments as any container. * Full rewrite to Typescript with tons of refactoring and small optimizations. The public API should be 100% the same * Fixed `patternImage` and `radialGradient` for `Konva.Text` * `Konva.Util._isObject` is renamed to `Konva.Util._isPlainObject`. * A bit changed behavior of `removeId` (private method), now it doesn't clear node ref, if object is changed. * simplified `batchDraw` method (it doesn't use `Konva.Animation`) now. -* `id` and `name` properties defaults are empty strings, not `undefined` * Performance improvements for shapes will image patterns, linear and radial fills -### Removed -* `Konva.Util.addMethods` -* `Konva.Util._removeLastLetter` -* `Konva.Util._getImage` -* `Konv.Util._getRGBAString` -* `Konv.Util._merge` -* Removed polyfill for `requestAnimationFrame`. - ### Fixed * Better mouse support on mobile devices (yes, that is possible to connect mouse to mobile) * Better implementation of `mouseover` event for stage * Fixed underline drawing for text with `lineHeight !== 1` +* Fixed some caching behavior when a node has `globalCompositeOperation`. +* Fixed automatic updates for `Konva.Transformer` ## [2.6.0][2018-12-14] diff --git a/konva.js b/konva.js index cad2bf45..fef048f2 100644 --- a/konva.js +++ b/konva.js @@ -8,7 +8,7 @@ * Konva JavaScript Framework v3.0.0-0 * http://konvajs.github.io/ * Licensed under the MIT - * Date: Mon Feb 18 2019 + * Date: Tue Feb 19 2019 * * Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS) * Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva) @@ -4337,6 +4337,7 @@ return Node; }()); Node.prototype.nodeType = 'Node'; + Node.prototype._attrsAffectingSize = []; /** * get/set zIndex relative to the node's siblings who share the same parent. * Please remember that zIndex is not absolute (like in CSS). It is relative to parent element only. @@ -7134,7 +7135,6 @@ // why do we need buffer canvas? // it give better result when a shape has // stroke with fill and with some opacity - // TODO: try to use it without stage (use global buffer canvas) Shape.prototype._useBufferCanvas = function (caching) { return ((!caching || this.hasShadow()) && this.perfectDrawEnabled() && @@ -9143,9 +9143,7 @@ var Arc = /** @class */ (function (_super) { __extends(Arc, _super); function Arc() { - var _this = _super !== null && _super.apply(this, arguments) || this; - _this._centroid = true; - return _this; + return _super !== null && _super.apply(this, arguments) || this; } Arc.prototype._sceneFunc = function (context) { var angle = getAngle(this.angle()), clockwise = this.clockwise(); @@ -9162,22 +9160,16 @@ return this.outerRadius() * 2; }; Arc.prototype.setWidth = function (width) { - // TODO: remove this line? - Node.prototype['setWidth'].call(this, width); - if (this.outerRadius() !== width / 2) { - this.outerRadius(width / 2); - } + this.outerRadius(width / 2); }; Arc.prototype.setHeight = function (height) { - // TODO: remove this line? - Node.prototype['setHeight'].call(this, height); - if (this.outerRadius() !== height / 2) { - this.outerRadius(height / 2); - } + this.outerRadius(height / 2); }; return Arc; }(Shape)); + Arc.prototype._centroid = true; Arc.prototype.className = 'Arc'; + Arc.prototype._attrsAffectingSize = ['innerRadius', 'outerRadius']; // add getters setters Factory.addGetterSetter(Arc, 'innerRadius', 0, Validators.getNumberValidator()); /** @@ -9452,6 +9444,7 @@ return Line; }(Shape)); Line.prototype.className = 'Line'; + Line.prototype._attrsAffectingSize = ['points', 'bezier', 'tension']; // add getters setters Factory.addGetterSetter(Line, 'closed', false); /** @@ -9817,10 +9810,7 @@ var Circle = /** @class */ (function (_super) { __extends(Circle, _super); function Circle() { - var _this = _super !== null && _super.apply(this, arguments) || this; - _this.className = 'Circle'; - _this._centroid = true; - return _this; + return _super !== null && _super.apply(this, arguments) || this; } Circle.prototype._sceneFunc = function (context) { context.beginPath(); @@ -9846,9 +9836,9 @@ }; return Circle; }(Shape)); + Circle.prototype._centroid = true; Circle.prototype.className = 'Circle'; - // add getters setters - Factory.addGetterSetter(Circle, 'radius', 0, Validators.getNumberValidator()); + Circle.prototype._attrsAffectingSize = ['radius']; /** * get/set radius * @name Konva.Arrow#radius @@ -9862,6 +9852,7 @@ * // set radius * circle.radius(10); */ + Factory.addGetterSetter(Circle, 'radius', 0, Validators.getNumberValidator()); Collection.mapMethods(Circle); /** @@ -9974,19 +9965,16 @@ return this.radiusY() * 2; }; Ellipse.prototype.setWidth = function (width) { - // TODO: remove this line? - Node.prototype['setWidth'].call(this, width); this.radiusX(width / 2); }; Ellipse.prototype.setHeight = function (height) { - // TODO: remove this line? - Node.prototype['setHeight'].call(this, height); this.radiusY(height / 2); }; return Ellipse; }(Shape)); Ellipse.prototype.className = 'Ellipse'; Ellipse.prototype._centroid = true; + Ellipse.prototype._attrsAffectingSize = ['radiusX', 'radiusY']; // add getters setters Factory.addComponentsGetterSetter(Ellipse, 'radius', ['x', 'y']); /** @@ -11361,6 +11349,7 @@ return Path; }(Shape)); Path.prototype.className = 'Path'; + Path.prototype._attrsAffectingSize = ['data']; /** * get/set SVG path data string. This method * also automatically parses the data string @@ -11628,19 +11617,16 @@ return this.radius() * 2; }; RegularPolygon.prototype.setWidth = function (width) { - if (this.radius() !== width / 2) { - this.radius(width / 2); - } + this.radius(width / 2); }; RegularPolygon.prototype.setHeight = function (height) { - if (this.radius() !== height / 2) { - this.radius(height / 2); - } + this.radius(height / 2); }; return RegularPolygon; }(Shape)); RegularPolygon.prototype.className = 'RegularPolygon'; RegularPolygon.prototype._centroid = true; + RegularPolygon.prototype._attrsAffectingSize = ['radius']; /** * get/set radius * @method @@ -11781,19 +11767,16 @@ return this.outerRadius() * 2; }; Ring.prototype.setWidth = function (width) { - if (this.outerRadius() !== width / 2) { - this.outerRadius(width / 2); - } + this.outerRadius(width / 2); }; Ring.prototype.setHeight = function (height) { - if (this.outerRadius() !== height / 2) { - this.outerRadius(height / 2); - } + this.outerRadius(height / 2); }; return Ring; }(Shape)); Ring.prototype.className = 'Ring'; Ring.prototype._centroid = true; + Ring.prototype._attrsAffectingSize = ['innerRadius', 'outerRadius']; /** * get/set innerRadius * @method @@ -12309,19 +12292,16 @@ return this.outerRadius() * 2; }; Star.prototype.setWidth = function (width) { - if (this.outerRadius() !== width / 2) { - this.outerRadius(width / 2); - } + this.outerRadius(width / 2); }; Star.prototype.setHeight = function (height) { - if (this.outerRadius() !== height / 2) { - this.outerRadius(height / 2); - } + this.outerRadius(height / 2); }; return Star; }(Shape)); Star.prototype.className = 'Star'; Star.prototype._centroid = true; + Star.prototype._attrsAffectingSize = ['innerRadius', 'outerRadius']; /** * get/set number of points * @name Konva.Ring#numPoints @@ -12823,6 +12803,13 @@ Text.prototype._fillFunc = _fillFunc$1; Text.prototype._strokeFunc = _strokeFunc$1; Text.prototype.className = TEXT_UPPER; + Text.prototype._attrsAffectingSize = [ + 'text', + 'fontSize', + 'padding', + 'wrap', + 'lineHeight' + ]; /** * get/set width of text area, which includes padding. * @name Konva.Text#width @@ -13519,6 +13506,7 @@ TextPath.prototype._fillFuncHit = _fillFunc$2; TextPath.prototype._strokeFuncHit = _strokeFunc$2; TextPath.prototype.className = 'TextPath'; + TextPath.prototype._attrsAffectingSize = ['text', 'fontSize', 'data']; /** * get/set SVG path data string. This method * also automatically parses the data string @@ -13700,32 +13688,20 @@ 'ignoreStrokeChange' ].join(' '); var NODE_RECT = 'nodeRect'; - // TODO: check circles and text here!!!! - // change text? change radius? change arc? var TRANSFORM_CHANGE_STR$1 = [ - 'xChange.resizer', - 'yChange.resizer', - 'widthChange.resizer', - 'heightChange.resizer', - 'scaleXChange.resizer', - 'scaleYChange.resizer', - 'skewXChange.resizer', - 'skewYChange.resizer', - 'rotationChange.resizer', - 'offsetXChange.resizer', - 'offsetYChange.resizer', - 'transformsEnabledChange.resizer' - ].join(' '); - var REDRAW_CHANGE_STR = [ - 'widthChange.resizer', - 'heightChange.resizer', - 'scaleXChange.resizer', - 'scaleYChange.resizer', - 'skewXChange.resizer', - 'skewYChange.resizer', - 'rotationChange.resizer', - 'offsetXChange.resizer', - 'offsetYChange.resizer' + 'xChange.tr', + 'yChange.tr', + 'widthChange.tr', + 'heightChange.tr', + 'scaleXChange.tr', + 'scaleYChange.tr', + 'skewXChange.tr', + 'skewYChange.tr', + 'rotationChange.tr', + 'offsetXChange.tr', + 'offsetYChange.tr', + 'transformsEnabledChange.tr', + 'strokeWidthChange.tr' ].join(' '); var ANGLES = { 'top-left': -45, @@ -13864,17 +13840,39 @@ return this; }; Transformer.prototype.setNode = function (node) { + var _this = this; if (this._node) { this.detach(); } this._node = node; this._resetTransformCache(); - node.on(TRANSFORM_CHANGE_STR$1, this._resetTransformCache.bind(this)); - node.on(REDRAW_CHANGE_STR, function () { - if (!this._transforming) { - this.update(); + var additionalEvents = node._attrsAffectingSize + .map(function (prop) { return prop + 'Change.tr'; }) + .join(' '); + var upChange = function () { + _this._resetTransformCache(); + if (!_this._transforming) { + _this.update(); } - }.bind(this)); + }; + node.on(additionalEvents, upChange); + node.on(TRANSFORM_CHANGE_STR$1, upChange); + // node.on( + // additionalEvents, + // function() { + // if (!this._transforming) { + // this.update(); + // } + // }.bind(this) + // ); + // node.on( + // REDRAW_CHANGE_STR, + // function() { + // if (!this._transforming) { + // this.update(); + // } + // }.bind(this) + // ); // we may need it if we set not in initial props // so elements are not defined yet var elementsCreated = !!this.findOne('.top-left'); @@ -13897,7 +13895,7 @@ */ Transformer.prototype.detach = function () { if (this.getNode()) { - this.getNode().off('.resizer'); + this.getNode().off('.tr'); this._node = undefined; } this._resetTransformCache(); @@ -14813,19 +14811,16 @@ return this.radius() * 2; }; Wedge.prototype.setWidth = function (width) { - if (this.radius() !== width / 2) { - this.radius(width / 2); - } + this.radius(width / 2); }; Wedge.prototype.setHeight = function (height) { - if (this.radius() !== height / 2) { - this.radius(height / 2); - } + this.radius(height / 2); }; return Wedge; }(Shape)); Wedge.prototype.className = 'Wedge'; Wedge.prototype._centroid = true; + Wedge.prototype._attrsAffectingSize = ['radius']; /** * get/set radius * @name Konva.Wedge#radius diff --git a/konva.min.js b/konva.min.js index 70b67420..831ad3c7 100644 --- a/konva.min.js +++ b/konva.min.js @@ -3,10 +3,10 @@ * Konva JavaScript Framework v3.0.0-0 * http://konvajs.github.io/ * Licensed under the MIT - * Date: Mon Feb 18 2019 + * Date: Tue Feb 19 2019 * * Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS) * Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva) * * @license - */var e=Math.PI/180,n={},o={},a="undefined"!=typeof window&&("[object Window]"==={}.toString.call(window)||"[object global]"==={}.toString.call(window)),t=/comment/.test(function(){}.toString()),h=function(t,e){e&&(n[e]||(n[e]=[]),n[e].push(t))},l=function(t,e){if(t){var i=n[t];if(i){for(var r=0;r>16&255,g:e>>8&255,b:255&e}},getRandomColor:function(){for(var t=(16777215*Math.random()<<0).toString(16);t.length<6;)t="0"+t;return"#"+t},get:function(t,e){return void 0===t?e:t},getRGB:function(t){var e;return t in y?{r:(e=y[t])[0],g:e[1],b:e[2]}:"#"===t[0]?this._hexToRgb(t.substring(1)):"rgb("===t.substr(0,4)?(e=m.exec(t.replace(/ /g,"")),{r:parseInt(e[1],10),g:parseInt(e[2],10),b:parseInt(e[3],10)}):{r:0,g:0,b:0}},colorToRGBA:function(t){return t=t||"black",O._namedColorToRBA(t)||O._hex3ColorToRGBA(t)||O._hex6ColorToRGBA(t)||O._rgbColorToRGBA(t)||O._rgbaColorToRGBA(t)},_namedColorToRBA:function(t){var e=y[t.toLowerCase()];return e?{r:e[0],g:e[1],b:e[2],a:1}:null},_rgbColorToRGBA:function(t){if(0===t.indexOf("rgb(")){var e=(t=t.match(/rgb\(([^)]+)\)/)[1]).split(/ *, */).map(Number);return{r:e[0],g:e[1],b:e[2],a:1}}},_rgbaColorToRGBA:function(t){if(0===t.indexOf("rgba(")){var e=(t=t.match(/rgba\(([^)]+)\)/)[1]).split(/ *, */).map(Number);return{r:e[0],g:e[1],b:e[2],a:e[3]}}},_hex6ColorToRGBA:function(t){if("#"===t[0]&&7===t.length)return{r:parseInt(t.slice(1,3),16),g:parseInt(t.slice(3,5),16),b:parseInt(t.slice(5,7),16),a:1}},_hex3ColorToRGBA:function(t){if("#"===t[0]&&4===t.length)return{r:parseInt(t[1]+t[1],16),g:parseInt(t[2]+t[2],16),b:parseInt(t[3]+t[3],16),a:1}},haveIntersection:function(t,e){return!(e.x>t.x+t.width||e.x+e.widtht.y+t.height||e.y+e.heighte.length){var o=e;e=t,t=o}for(r=0;r=D().traceArrMax&&e.shift()},t.prototype.reset=function(){var t=this.getCanvas().getPixelRatio();this.setTransform(1*t,0,0,1*t,0,0)},t.prototype.getCanvas=function(){return this.canvas},t.prototype.clear=function(t){var e=this.getCanvas();t?this.clearRect(t.x||0,t.y||0,t.width||0,t.height||0):this.clearRect(0,0,e.getWidth()/e.pixelRatio,e.getHeight()/e.pixelRatio)},t.prototype._applyLineCap=function(t){var e=t.getLineCap();e&&this.setAttr("lineCap",e)},t.prototype._applyOpacity=function(t){var e=t.getAbsoluteOpacity();1!==e&&this.setAttr("globalAlpha",e)},t.prototype._applyLineJoin=function(t){var e=t.getLineJoin();e&&this.setAttr("lineJoin",e)},t.prototype.setAttr=function(t,e){this._context[t]=e},t.prototype.arc=function(t,e,i,r,n,a){this._context.arc(t,e,i,r,n,a)},t.prototype.arcTo=function(t,e,i,r,n,a){this._context.arc(t,e,i,r,n,a)},t.prototype.beginPath=function(){this._context.beginPath()},t.prototype.bezierCurveTo=function(t,e,i,r,n,a){this._context.bezierCurveTo(t,e,i,r,n,a)},t.prototype.clearRect=function(t,e,i,r){this._context.clearRect(t,e,i,r)},t.prototype.clip=function(){this._context.clip()},t.prototype.closePath=function(){this._context.closePath()},t.prototype.createImageData=function(t,e){var i=arguments;return 2===i.length?this._context.createImageData(t,e):1===i.length?this._context.createImageData(t):void 0},t.prototype.createLinearGradient=function(t,e,i,r){return this._context.createLinearGradient(t,e,i,r)},t.prototype.createPattern=function(t,e){return this._context.createPattern(t,e)},t.prototype.createRadialGradient=function(t,e,i,r,n,a){return this._context.createRadialGradient(t,e,i,r,n,a)},t.prototype.drawImage=function(t,e,i,r,n,a,o,s,h){var l=arguments,d=this._context;3===l.length?d.drawImage(t,e,i):5===l.length?d.drawImage(t,e,i,r,n):9===l.length&&d.drawImage(t,e,i,r,n,a,o,s,h)},t.prototype.isPointInPath=function(t,e){return this._context.isPointInPath(t,e)},t.prototype.fill=function(){this._context.fill()},t.prototype.fillRect=function(t,e,i,r){this._context.fillRect(t,e,i,r)},t.prototype.strokeRect=function(t,e,i,r){this._context.strokeRect(t,e,i,r)},t.prototype.fillText=function(t,e,i){this._context.fillText(t,e,i)},t.prototype.measureText=function(t){return this._context.measureText(t)},t.prototype.getImageData=function(t,e,i,r){return this._context.getImageData(t,e,i,r)},t.prototype.lineTo=function(t,e){this._context.lineTo(t,e)},t.prototype.moveTo=function(t,e){this._context.moveTo(t,e)},t.prototype.rect=function(t,e,i,r){this._context.rect(t,e,i,r)},t.prototype.putImageData=function(t,e,i){this._context.putImageData(t,e,i)},t.prototype.quadraticCurveTo=function(t,e,i,r){this._context.quadraticCurveTo(t,e,i,r)},t.prototype.restore=function(){this._context.restore()},t.prototype.rotate=function(t){this._context.rotate(t)},t.prototype.save=function(){this._context.save()},t.prototype.scale=function(t,e){this._context.scale(t,e)},t.prototype.setLineDash=function(t){this._context.setLineDash?this._context.setLineDash(t):"mozDash"in this._context?this._context.mozDash=t:"webkitLineDash"in this._context&&(this._context.webkitLineDash=t)},t.prototype.getLineDash=function(){return this._context.getLineDash()},t.prototype.setTransform=function(t,e,i,r,n,a){this._context.setTransform(t,e,i,r,n,a)},t.prototype.stroke=function(){this._context.stroke()},t.prototype.strokeText=function(t,e,i,r){this._context.strokeText(t,e,i,r)},t.prototype.transform=function(t,e,i,r,n,a){this._context.transform(t,e,i,r,n,a)},t.prototype.translate=function(t,e){this._context.translate(t,e)},t.prototype._enableTrace=function(){var t,r,n=this,e=k.length,a=O._simplifyArray,i=this.setAttr,o=function(t){var e,i=n[t];n[t]=function(){return r=a(Array.prototype.slice.call(arguments,0)),e=i.apply(n,arguments),n._trace({method:t,args:r}),e}};for(t=0;t=this.parent.children.length)&&O.warn("Unexpected value "+t+" for zIndex property. zIndex is just index of a node in children of its parent. Expected value is from 0 to "+(this.parent.children.length-1)+".");var e=this.index;return this.parent.children.splice(e,1),this.parent.children.splice(t,0,this),this.parent._setChildrenIndices(),this},s.prototype.getAbsoluteOpacity=function(){return this._getCache(z,this._getAbsoluteOpacity)},s.prototype._getAbsoluteOpacity=function(){var t=this.opacity(),e=this.getParent();return e&&!e._isUnderCache&&(t*=this.getParent().getAbsoluteOpacity()),t},s.prototype.moveTo=function(t){return this.getParent()!==t&&(this._remove(),t.add(this)),this},s.prototype.toObject=function(){var t,e,i,r={},n=this.getAttrs();for(t in r.attrs={},n)e=n[t],O.isObject(e)&&!O._isPlainObject(e)&&!O._isArray(e)||(i="function"==typeof this[t]&&this[t],delete n[t],(i?i.call(this):null)!==(n[t]=e)&&(r.attrs[t]=e));return r.className=this.getClassName(),O._prepareToStringify(r)},s.prototype.toJSON=function(){return JSON.stringify(this.toObject())},s.prototype.getParent=function(){return this.parent},s.prototype.findAncestors=function(t,e,i){var r=[];e&&this._isMatch(t)&&r.push(this);for(var n=this.parent;n;){if(n===i)return r;n._isMatch(t)&&r.push(n),n=n.parent}return r},s.prototype.isAncestorOf=function(t){return!1},s.prototype.findAncestor=function(t,e,i){return this.findAncestors(t,e,i)[0]},s.prototype._isMatch=function(t){if(!t)return!1;var e,i,r=t.replace(/ /g,"").split(","),n=r.length;for(e=0;ethis.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())},t.prototype.getTime=function(){return this._time},t.prototype.setPosition=function(t){this.prevPos=this._pos,this.propFunc(t),this._pos=t},t.prototype.getPosition=function(t){return void 0===t&&(t=this._time),this.func(t,this.begin,this._change,this.duration)},t.prototype.play=function(){this.state=2,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onPlay")},t.prototype.reverse=function(){this.state=3,this._time=this.duration-this._time,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onReverse")},t.prototype.seek=function(t){this.pause(),this._time=t,this.update(),this.fire("onSeek")},t.prototype.reset=function(){this.pause(),this._time=0,this.update(),this.fire("onReset")},t.prototype.finish=function(){this.pause(),this._time=this.duration,this.update(),this.fire("onFinish")},t.prototype.update=function(){this.setPosition(this.getPosition(this._time))},t.prototype.onEnterFrame=function(){var t=this.getTimer()-this._startTime;2===this.state?this.setTime(t):3===this.state&&this.setTime(this.duration-t)},t.prototype.pause=function(){this.state=1,this.fire("onPause")},t.prototype.getTimer=function(){return(new Date).getTime()},t}(),Yt=function(){function p(t){var e,i,r=this,n=t.node,a=n._id,o=t.easing||Xt.Linear,s=!!t.yoyo;e=void 0===t.duration?.3:0===t.duration?.001:t.duration,this.node=n,this._id=zt++;var h=n.getLayer()||(n instanceof D().Stage?n.getLayers():null);for(i in h||O.error("Tween constructor have `node` that is not in a layer. Please add node into layer first."),this.anim=new F(function(){r.tween.onEnterFrame()},h),this.tween=new Wt(i,function(t){r._tweenFunc(t)},o,0,1,1e3*e,s),this._addListeners(),p.attrs[a]||(p.attrs[a]={}),p.attrs[a][this._id]||(p.attrs[a][this._id]={}),p.tweens[a]||(p.tweens[a]={}),t)void 0===Vt[i]&&this._addAttr(i,t[i]);this.reset(),this.onFinish=t.onFinish,this.onReset=t.onReset}return p.prototype._addAttr=function(t,e){var i,r,n,a,o,s,h,l,d=this.node,c=d._id;if((n=p.tweens[c][t])&&delete p.attrs[c][n][t],i=d.getAttr(t),O._isArray(e))if(r=[],o=Math.max(e.length,i.length),"points"===t&&e.length!==i.length&&(e.length>i.length?(h=i,i=O._prepareArrayForTween(i,e,d.closed())):(s=e,e=O._prepareArrayForTween(e,i,d.closed()))),0===t.indexOf("fill"))for(a=0;athis.dataArray[i].pathLength;)t-=this.dataArray[i].pathLength,++i;if(i===r)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 n=this.dataArray[i],a=n.points;switch(n.command){case"L":return p.getPointOnLine(t,n.start.x,n.start.y,a[0],a[1]);case"C":return p.getPointOnCubicBezier(t/n.pathLength,n.start.x,n.start.y,a[0],a[1],a[2],a[3],a[4],a[5]);case"Q":return p.getPointOnQuadraticBezier(t/n.pathLength,n.start.x,n.start.y,a[0],a[1],a[2],a[3]);case"A":var o=a[0],s=a[1],h=a[2],l=a[3],d=a[4],c=a[5],u=a[6];return d+=c*t/n.pathLength,p.getPointOnEllipticalArc(o,s,h,l,d,u)}return null},p.getLineLength=function(t,e,i,r){return Math.sqrt((i-t)*(i-t)+(r-e)*(r-e))},p.getPointOnLine=function(t,e,i,r,n,a,o){void 0===a&&(a=e),void 0===o&&(o=i);var s=(n-i)/(r-e+1e-8),h=Math.sqrt(t*t/(1+s*s));r>>1,P=_.slice(0,k+1),T=this._getTextWidth(P)+v;T<=l?(S=k+1,w=P+(g?"…":""),C=T):x=k}if(!w)break;if(f){var M,G=_[w.length];0<(M=(" "===G||"-"===G)&&C<=l?w.length:Math.max(w.lastIndexOf(" "),w.lastIndexOf("-"))+1)&&(S=M,w=w.slice(0,S),C=this._getTextWidth(w))}if(w=w.trimRight(),this._addTextLine(w),i=Math.max(i,C),c+=r,!p||s&&de?g=ae.getPointOnLine(e,f.x,f.y,v.points[0],v.points[1],f.x,f.y):v=void 0;break;case"A":var o=v.points[4],s=v.points[5],h=v.points[4]+s;0===m?m=o+1e-8:iv.pathLength?1e-8:e/v.pathLength:i>V,0!==C?(C=255/C,P[s]=(l*B>>V)*C,P[s+1]=(d*B>>V)*C,P[s+2]=(c*B>>V)*C):P[s]=P[s+1]=P[s+2]=0,l-=p,d-=f,c-=g,u-=v,p-=F.r,f-=F.g,g-=F.b,v-=F.a,a=h+((a=i+e+1)>V,0>V)*C,P[a+1]=(d*B>>V)*C,P[a+2]=(c*B>>V)*C):P[a]=P[a+1]=P[a+2]=0,l-=p,d-=f,c-=g,u-=v,p-=F.r,f-=F.g,g-=F.b,v-=F.a,a=i+((a=r+L)>16&255,g:e>>8&255,b:255&e}},getRandomColor:function(){for(var t=(16777215*Math.random()<<0).toString(16);t.length<6;)t="0"+t;return"#"+t},get:function(t,e){return void 0===t?e:t},getRGB:function(t){var e;return t in y?{r:(e=y[t])[0],g:e[1],b:e[2]}:"#"===t[0]?this._hexToRgb(t.substring(1)):"rgb("===t.substr(0,4)?(e=m.exec(t.replace(/ /g,"")),{r:parseInt(e[1],10),g:parseInt(e[2],10),b:parseInt(e[3],10)}):{r:0,g:0,b:0}},colorToRGBA:function(t){return t=t||"black",O._namedColorToRBA(t)||O._hex3ColorToRGBA(t)||O._hex6ColorToRGBA(t)||O._rgbColorToRGBA(t)||O._rgbaColorToRGBA(t)},_namedColorToRBA:function(t){var e=y[t.toLowerCase()];return e?{r:e[0],g:e[1],b:e[2],a:1}:null},_rgbColorToRGBA:function(t){if(0===t.indexOf("rgb(")){var e=(t=t.match(/rgb\(([^)]+)\)/)[1]).split(/ *, */).map(Number);return{r:e[0],g:e[1],b:e[2],a:1}}},_rgbaColorToRGBA:function(t){if(0===t.indexOf("rgba(")){var e=(t=t.match(/rgba\(([^)]+)\)/)[1]).split(/ *, */).map(Number);return{r:e[0],g:e[1],b:e[2],a:e[3]}}},_hex6ColorToRGBA:function(t){if("#"===t[0]&&7===t.length)return{r:parseInt(t.slice(1,3),16),g:parseInt(t.slice(3,5),16),b:parseInt(t.slice(5,7),16),a:1}},_hex3ColorToRGBA:function(t){if("#"===t[0]&&4===t.length)return{r:parseInt(t[1]+t[1],16),g:parseInt(t[2]+t[2],16),b:parseInt(t[3]+t[3],16),a:1}},haveIntersection:function(t,e){return!(e.x>t.x+t.width||e.x+e.widtht.y+t.height||e.y+e.heighte.length){var o=e;e=t,t=o}for(r=0;r=D().traceArrMax&&e.shift()},t.prototype.reset=function(){var t=this.getCanvas().getPixelRatio();this.setTransform(1*t,0,0,1*t,0,0)},t.prototype.getCanvas=function(){return this.canvas},t.prototype.clear=function(t){var e=this.getCanvas();t?this.clearRect(t.x||0,t.y||0,t.width||0,t.height||0):this.clearRect(0,0,e.getWidth()/e.pixelRatio,e.getHeight()/e.pixelRatio)},t.prototype._applyLineCap=function(t){var e=t.getLineCap();e&&this.setAttr("lineCap",e)},t.prototype._applyOpacity=function(t){var e=t.getAbsoluteOpacity();1!==e&&this.setAttr("globalAlpha",e)},t.prototype._applyLineJoin=function(t){var e=t.getLineJoin();e&&this.setAttr("lineJoin",e)},t.prototype.setAttr=function(t,e){this._context[t]=e},t.prototype.arc=function(t,e,i,r,n,a){this._context.arc(t,e,i,r,n,a)},t.prototype.arcTo=function(t,e,i,r,n,a){this._context.arc(t,e,i,r,n,a)},t.prototype.beginPath=function(){this._context.beginPath()},t.prototype.bezierCurveTo=function(t,e,i,r,n,a){this._context.bezierCurveTo(t,e,i,r,n,a)},t.prototype.clearRect=function(t,e,i,r){this._context.clearRect(t,e,i,r)},t.prototype.clip=function(){this._context.clip()},t.prototype.closePath=function(){this._context.closePath()},t.prototype.createImageData=function(t,e){var i=arguments;return 2===i.length?this._context.createImageData(t,e):1===i.length?this._context.createImageData(t):void 0},t.prototype.createLinearGradient=function(t,e,i,r){return this._context.createLinearGradient(t,e,i,r)},t.prototype.createPattern=function(t,e){return this._context.createPattern(t,e)},t.prototype.createRadialGradient=function(t,e,i,r,n,a){return this._context.createRadialGradient(t,e,i,r,n,a)},t.prototype.drawImage=function(t,e,i,r,n,a,o,s,h){var l=arguments,d=this._context;3===l.length?d.drawImage(t,e,i):5===l.length?d.drawImage(t,e,i,r,n):9===l.length&&d.drawImage(t,e,i,r,n,a,o,s,h)},t.prototype.isPointInPath=function(t,e){return this._context.isPointInPath(t,e)},t.prototype.fill=function(){this._context.fill()},t.prototype.fillRect=function(t,e,i,r){this._context.fillRect(t,e,i,r)},t.prototype.strokeRect=function(t,e,i,r){this._context.strokeRect(t,e,i,r)},t.prototype.fillText=function(t,e,i){this._context.fillText(t,e,i)},t.prototype.measureText=function(t){return this._context.measureText(t)},t.prototype.getImageData=function(t,e,i,r){return this._context.getImageData(t,e,i,r)},t.prototype.lineTo=function(t,e){this._context.lineTo(t,e)},t.prototype.moveTo=function(t,e){this._context.moveTo(t,e)},t.prototype.rect=function(t,e,i,r){this._context.rect(t,e,i,r)},t.prototype.putImageData=function(t,e,i){this._context.putImageData(t,e,i)},t.prototype.quadraticCurveTo=function(t,e,i,r){this._context.quadraticCurveTo(t,e,i,r)},t.prototype.restore=function(){this._context.restore()},t.prototype.rotate=function(t){this._context.rotate(t)},t.prototype.save=function(){this._context.save()},t.prototype.scale=function(t,e){this._context.scale(t,e)},t.prototype.setLineDash=function(t){this._context.setLineDash?this._context.setLineDash(t):"mozDash"in this._context?this._context.mozDash=t:"webkitLineDash"in this._context&&(this._context.webkitLineDash=t)},t.prototype.getLineDash=function(){return this._context.getLineDash()},t.prototype.setTransform=function(t,e,i,r,n,a){this._context.setTransform(t,e,i,r,n,a)},t.prototype.stroke=function(){this._context.stroke()},t.prototype.strokeText=function(t,e,i,r){this._context.strokeText(t,e,i,r)},t.prototype.transform=function(t,e,i,r,n,a){this._context.transform(t,e,i,r,n,a)},t.prototype.translate=function(t,e){this._context.translate(t,e)},t.prototype._enableTrace=function(){var t,r,n=this,e=k.length,a=O._simplifyArray,i=this.setAttr,o=function(t){var e,i=n[t];n[t]=function(){return r=a(Array.prototype.slice.call(arguments,0)),e=i.apply(n,arguments),n._trace({method:t,args:r}),e}};for(t=0;t=this.parent.children.length)&&O.warn("Unexpected value "+t+" for zIndex property. zIndex is just index of a node in children of its parent. Expected value is from 0 to "+(this.parent.children.length-1)+".");var e=this.index;return this.parent.children.splice(e,1),this.parent.children.splice(t,0,this),this.parent._setChildrenIndices(),this},s.prototype.getAbsoluteOpacity=function(){return this._getCache(z,this._getAbsoluteOpacity)},s.prototype._getAbsoluteOpacity=function(){var t=this.opacity(),e=this.getParent();return e&&!e._isUnderCache&&(t*=this.getParent().getAbsoluteOpacity()),t},s.prototype.moveTo=function(t){return this.getParent()!==t&&(this._remove(),t.add(this)),this},s.prototype.toObject=function(){var t,e,i,r={},n=this.getAttrs();for(t in r.attrs={},n)e=n[t],O.isObject(e)&&!O._isPlainObject(e)&&!O._isArray(e)||(i="function"==typeof this[t]&&this[t],delete n[t],(i?i.call(this):null)!==(n[t]=e)&&(r.attrs[t]=e));return r.className=this.getClassName(),O._prepareToStringify(r)},s.prototype.toJSON=function(){return JSON.stringify(this.toObject())},s.prototype.getParent=function(){return this.parent},s.prototype.findAncestors=function(t,e,i){var r=[];e&&this._isMatch(t)&&r.push(this);for(var n=this.parent;n;){if(n===i)return r;n._isMatch(t)&&r.push(n),n=n.parent}return r},s.prototype.isAncestorOf=function(t){return!1},s.prototype.findAncestor=function(t,e,i){return this.findAncestors(t,e,i)[0]},s.prototype._isMatch=function(t){if(!t)return!1;var e,i,r=t.replace(/ /g,"").split(","),n=r.length;for(e=0;ethis.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())},t.prototype.getTime=function(){return this._time},t.prototype.setPosition=function(t){this.prevPos=this._pos,this.propFunc(t),this._pos=t},t.prototype.getPosition=function(t){return void 0===t&&(t=this._time),this.func(t,this.begin,this._change,this.duration)},t.prototype.play=function(){this.state=2,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onPlay")},t.prototype.reverse=function(){this.state=3,this._time=this.duration-this._time,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onReverse")},t.prototype.seek=function(t){this.pause(),this._time=t,this.update(),this.fire("onSeek")},t.prototype.reset=function(){this.pause(),this._time=0,this.update(),this.fire("onReset")},t.prototype.finish=function(){this.pause(),this._time=this.duration,this.update(),this.fire("onFinish")},t.prototype.update=function(){this.setPosition(this.getPosition(this._time))},t.prototype.onEnterFrame=function(){var t=this.getTimer()-this._startTime;2===this.state?this.setTime(t):3===this.state&&this.setTime(this.duration-t)},t.prototype.pause=function(){this.state=1,this.fire("onPause")},t.prototype.getTimer=function(){return(new Date).getTime()},t}(),Yt=function(){function p(t){var e,i,r=this,n=t.node,a=n._id,o=t.easing||Xt.Linear,s=!!t.yoyo;e=void 0===t.duration?.3:0===t.duration?.001:t.duration,this.node=n,this._id=zt++;var h=n.getLayer()||(n instanceof D().Stage?n.getLayers():null);for(i in h||O.error("Tween constructor have `node` that is not in a layer. Please add node into layer first."),this.anim=new F(function(){r.tween.onEnterFrame()},h),this.tween=new Wt(i,function(t){r._tweenFunc(t)},o,0,1,1e3*e,s),this._addListeners(),p.attrs[a]||(p.attrs[a]={}),p.attrs[a][this._id]||(p.attrs[a][this._id]={}),p.tweens[a]||(p.tweens[a]={}),t)void 0===Vt[i]&&this._addAttr(i,t[i]);this.reset(),this.onFinish=t.onFinish,this.onReset=t.onReset}return p.prototype._addAttr=function(t,e){var i,r,n,a,o,s,h,l,d=this.node,c=d._id;if((n=p.tweens[c][t])&&delete p.attrs[c][n][t],i=d.getAttr(t),O._isArray(e))if(r=[],o=Math.max(e.length,i.length),"points"===t&&e.length!==i.length&&(e.length>i.length?(h=i,i=O._prepareArrayForTween(i,e,d.closed())):(s=e,e=O._prepareArrayForTween(e,i,d.closed()))),0===t.indexOf("fill"))for(a=0;athis.dataArray[i].pathLength;)t-=this.dataArray[i].pathLength,++i;if(i===r)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 n=this.dataArray[i],a=n.points;switch(n.command){case"L":return p.getPointOnLine(t,n.start.x,n.start.y,a[0],a[1]);case"C":return p.getPointOnCubicBezier(t/n.pathLength,n.start.x,n.start.y,a[0],a[1],a[2],a[3],a[4],a[5]);case"Q":return p.getPointOnQuadraticBezier(t/n.pathLength,n.start.x,n.start.y,a[0],a[1],a[2],a[3]);case"A":var o=a[0],s=a[1],h=a[2],l=a[3],d=a[4],c=a[5],u=a[6];return d+=c*t/n.pathLength,p.getPointOnEllipticalArc(o,s,h,l,d,u)}return null},p.getLineLength=function(t,e,i,r){return Math.sqrt((i-t)*(i-t)+(r-e)*(r-e))},p.getPointOnLine=function(t,e,i,r,n,a,o){void 0===a&&(a=e),void 0===o&&(o=i);var s=(n-i)/(r-e+1e-8),h=Math.sqrt(t*t/(1+s*s));r>>1,P=_.slice(0,k+1),T=this._getTextWidth(P)+v;T<=l?(S=k+1,w=P+(g?"…":""),C=T):x=k}if(!w)break;if(f){var M,A=_[w.length];0<(M=(" "===A||"-"===A)&&C<=l?w.length:Math.max(w.lastIndexOf(" "),w.lastIndexOf("-"))+1)&&(S=M,w=w.slice(0,S),C=this._getTextWidth(w))}if(w=w.trimRight(),this._addTextLine(w),i=Math.max(i,C),c+=r,!p||s&&de?g=ae.getPointOnLine(e,f.x,f.y,v.points[0],v.points[1],f.x,f.y):v=void 0;break;case"A":var o=v.points[4],s=v.points[5],h=v.points[4]+s;0===m?m=o+1e-8:iv.pathLength?1e-8:e/v.pathLength:i>V,0!==C?(C=255/C,P[s]=(l*B>>V)*C,P[s+1]=(d*B>>V)*C,P[s+2]=(c*B>>V)*C):P[s]=P[s+1]=P[s+2]=0,l-=p,d-=f,c-=g,u-=v,p-=F.r,f-=F.g,g-=F.b,v-=F.a,a=h+((a=i+e+1)>V,0>V)*C,P[a+1]=(d*B>>V)*C,P[a+2]=(c*B>>V)*C):P[a]=P[a+1]=P[a+2]=0,l-=p,d-=f,c-=g,u-=v,p-=F.r,f-=F.g,g-=F.b,v-=F.a,a=i+((a=r+L) = new Map(); _lastPos = null; + _attrsAffectingSize: string[]; _filterUpToDate = false; _isUnderCache = false; @@ -2324,6 +2325,7 @@ export abstract class Node { } Node.prototype.nodeType = 'Node'; +Node.prototype._attrsAffectingSize = []; /** * get/set zIndex relative to the node's siblings who share the same parent. diff --git a/src/Shape.ts b/src/Shape.ts index f0ef1eb0..5285c57a 100644 --- a/src/Shape.ts +++ b/src/Shape.ts @@ -320,7 +320,6 @@ export class Shape extends Node { // why do we need buffer canvas? // it give better result when a shape has // stroke with fill and with some opacity - // TODO: try to use it without stage (use global buffer canvas) _useBufferCanvas(caching) { return ( (!caching || this.hasShadow()) && diff --git a/src/shapes/Arc.ts b/src/shapes/Arc.ts index 2c88ee9f..f01bae11 100644 --- a/src/shapes/Arc.ts +++ b/src/shapes/Arc.ts @@ -30,8 +30,6 @@ import { GetSet } from '../types'; * }); */ export class Arc extends Shape { - _centroid = true; - _sceneFunc(context) { var angle = getAngle(this.angle()), clockwise = this.clockwise(); @@ -49,18 +47,10 @@ export class Arc extends Shape { return this.outerRadius() * 2; } setWidth(width) { - // TODO: remove this line? - Node.prototype['setWidth'].call(this, width); - if (this.outerRadius() !== width / 2) { - this.outerRadius(width / 2); - } + this.outerRadius(width / 2); } setHeight(height) { - // TODO: remove this line? - Node.prototype['setHeight'].call(this, height); - if (this.outerRadius() !== height / 2) { - this.outerRadius(height / 2); - } + this.outerRadius(height / 2); } innerRadius: GetSet; @@ -69,7 +59,9 @@ export class Arc extends Shape { clockwise: GetSet; } +Arc.prototype._centroid = true; Arc.prototype.className = 'Arc'; +Arc.prototype._attrsAffectingSize = ['innerRadius', 'outerRadius']; // add getters setters Factory.addGetterSetter(Arc, 'innerRadius', 0, Validators.getNumberValidator()); diff --git a/src/shapes/Circle.ts b/src/shapes/Circle.ts index ed9b1136..22b62beb 100644 --- a/src/shapes/Circle.ts +++ b/src/shapes/Circle.ts @@ -23,9 +23,6 @@ import { GetSet } from '../types'; * }); */ export class Circle extends Shape { - className = 'Circle'; - _centroid = true; - _sceneFunc(context) { context.beginPath(); context.arc(0, 0, this.radius(), 0, Math.PI * 2, false); @@ -52,10 +49,9 @@ export class Circle extends Shape { radius: GetSet; } +Circle.prototype._centroid = true; Circle.prototype.className = 'Circle'; - -// add getters setters -Factory.addGetterSetter(Circle, 'radius', 0, Validators.getNumberValidator()); +Circle.prototype._attrsAffectingSize = ['radius']; /** * get/set radius @@ -70,5 +66,6 @@ Factory.addGetterSetter(Circle, 'radius', 0, Validators.getNumberValidator()); * // set radius * circle.radius(10); */ +Factory.addGetterSetter(Circle, 'radius', 0, Validators.getNumberValidator()); Collection.mapMethods(Circle); diff --git a/src/shapes/Ellipse.ts b/src/shapes/Ellipse.ts index 0fec4ad2..6857e458 100644 --- a/src/shapes/Ellipse.ts +++ b/src/shapes/Ellipse.ts @@ -44,13 +44,9 @@ export class Ellipse extends Shape { return this.radiusY() * 2; } setWidth(width) { - // TODO: remove this line? - Node.prototype['setWidth'].call(this, width); this.radiusX(width / 2); } setHeight(height) { - // TODO: remove this line? - Node.prototype['setHeight'].call(this, height); this.radiusY(height / 2); } @@ -61,6 +57,7 @@ export class Ellipse extends Shape { Ellipse.prototype.className = 'Ellipse'; Ellipse.prototype._centroid = true; +Ellipse.prototype._attrsAffectingSize = ['radiusX', 'radiusY']; // add getters setters Factory.addComponentsGetterSetter(Ellipse, 'radius', ['x', 'y']); diff --git a/src/shapes/Line.ts b/src/shapes/Line.ts index a8c34ea8..7549b062 100644 --- a/src/shapes/Line.ts +++ b/src/shapes/Line.ts @@ -207,6 +207,7 @@ export class Line extends Shape { } Line.prototype.className = 'Line'; +Line.prototype._attrsAffectingSize = ['points', 'bezier', 'tension']; // add getters setters Factory.addGetterSetter(Line, 'closed', false); diff --git a/src/shapes/Path.ts b/src/shapes/Path.ts index c76890f2..db6c5734 100644 --- a/src/shapes/Path.ts +++ b/src/shapes/Path.ts @@ -868,6 +868,7 @@ export class Path extends Shape { } Path.prototype.className = 'Path'; +Path.prototype._attrsAffectingSize = ['data']; /** * get/set SVG path data string. This method diff --git a/src/shapes/RegularPolygon.ts b/src/shapes/RegularPolygon.ts index 7041bcd7..2c43a071 100644 --- a/src/shapes/RegularPolygon.ts +++ b/src/shapes/RegularPolygon.ts @@ -50,14 +50,10 @@ export class RegularPolygon extends Shape { return this.radius() * 2; } setWidth(width) { - if (this.radius() !== width / 2) { - this.radius(width / 2); - } + this.radius(width / 2); } setHeight(height) { - if (this.radius() !== height / 2) { - this.radius(height / 2); - } + this.radius(height / 2); } radius: GetSet; @@ -66,6 +62,7 @@ export class RegularPolygon extends Shape { RegularPolygon.prototype.className = 'RegularPolygon'; RegularPolygon.prototype._centroid = true; +RegularPolygon.prototype._attrsAffectingSize = ['radius']; /** * get/set radius diff --git a/src/shapes/Ring.ts b/src/shapes/Ring.ts index 9053ae31..217cbfa1 100644 --- a/src/shapes/Ring.ts +++ b/src/shapes/Ring.ts @@ -40,14 +40,10 @@ export class Ring extends Shape { return this.outerRadius() * 2; } setWidth(width) { - if (this.outerRadius() !== width / 2) { - this.outerRadius(width / 2); - } + this.outerRadius(width / 2); } setHeight(height) { - if (this.outerRadius() !== height / 2) { - this.outerRadius(height / 2); - } + this.outerRadius(height / 2); } outerRadius: GetSet; @@ -56,6 +52,7 @@ export class Ring extends Shape { Ring.prototype.className = 'Ring'; Ring.prototype._centroid = true; +Ring.prototype._attrsAffectingSize = ['innerRadius', 'outerRadius']; /** * get/set innerRadius diff --git a/src/shapes/Sprite.ts b/src/shapes/Sprite.ts index 42d8c689..1525b8e5 100644 --- a/src/shapes/Sprite.ts +++ b/src/shapes/Sprite.ts @@ -209,7 +209,6 @@ export class Sprite extends Shape { frameIndex: GetSet; animation: GetSet; image: GetSet; - // TODO: write better type animations: GetSet; frameOffsets: GetSet; frameRate: GetSet; diff --git a/src/shapes/Star.ts b/src/shapes/Star.ts index 37a8f6d1..f17ad85b 100644 --- a/src/shapes/Star.ts +++ b/src/shapes/Star.ts @@ -54,14 +54,10 @@ export class Star extends Shape { return this.outerRadius() * 2; } setWidth(width) { - if (this.outerRadius() !== width / 2) { - this.outerRadius(width / 2); - } + this.outerRadius(width / 2); } setHeight(height) { - if (this.outerRadius() !== height / 2) { - this.outerRadius(height / 2); - } + this.outerRadius(height / 2); } outerRadius: GetSet; @@ -71,6 +67,7 @@ export class Star extends Shape { Star.prototype.className = 'Star'; Star.prototype._centroid = true; +Star.prototype._attrsAffectingSize = ['innerRadius', 'outerRadius']; /** * get/set number of points diff --git a/src/shapes/Text.ts b/src/shapes/Text.ts index b7292c80..6354ef3e 100644 --- a/src/shapes/Text.ts +++ b/src/shapes/Text.ts @@ -504,6 +504,13 @@ export class Text extends Shape { Text.prototype._fillFunc = _fillFunc; Text.prototype._strokeFunc = _strokeFunc; Text.prototype.className = TEXT_UPPER; +Text.prototype._attrsAffectingSize = [ + 'text', + 'fontSize', + 'padding', + 'wrap', + 'lineHeight' +]; /** * get/set width of text area, which includes padding. diff --git a/src/shapes/TextPath.ts b/src/shapes/TextPath.ts index e1bbd650..6d7a92b0 100644 --- a/src/shapes/TextPath.ts +++ b/src/shapes/TextPath.ts @@ -539,6 +539,7 @@ TextPath.prototype._strokeFunc = _strokeFunc; TextPath.prototype._fillFuncHit = _fillFunc; TextPath.prototype._strokeFuncHit = _strokeFunc; TextPath.prototype.className = 'TextPath'; +TextPath.prototype._attrsAffectingSize = ['text', 'fontSize', 'data']; /** * get/set SVG path data string. This method diff --git a/src/shapes/Transformer.ts b/src/shapes/Transformer.ts index 638d91ac..a23216bd 100644 --- a/src/shapes/Transformer.ts +++ b/src/shapes/Transformer.ts @@ -27,34 +27,20 @@ var ATTR_CHANGE_LIST = [ var NODE_RECT = 'nodeRect'; -// TODO: check circles and text here!!!! -// change text? change radius? change arc? - var TRANSFORM_CHANGE_STR = [ - 'xChange.resizer', - 'yChange.resizer', - 'widthChange.resizer', - 'heightChange.resizer', - 'scaleXChange.resizer', - 'scaleYChange.resizer', - 'skewXChange.resizer', - 'skewYChange.resizer', - 'rotationChange.resizer', - 'offsetXChange.resizer', - 'offsetYChange.resizer', - 'transformsEnabledChange.resizer' -].join(' '); - -var REDRAW_CHANGE_STR = [ - 'widthChange.resizer', - 'heightChange.resizer', - 'scaleXChange.resizer', - 'scaleYChange.resizer', - 'skewXChange.resizer', - 'skewYChange.resizer', - 'rotationChange.resizer', - 'offsetXChange.resizer', - 'offsetYChange.resizer' + 'xChange.tr', + 'yChange.tr', + 'widthChange.tr', + 'heightChange.tr', + 'scaleXChange.tr', + 'scaleYChange.tr', + 'skewXChange.tr', + 'skewYChange.tr', + 'rotationChange.tr', + 'offsetXChange.tr', + 'offsetYChange.tr', + 'transformsEnabledChange.tr', + 'strokeWidthChange.tr' ].join(' '); var ANGLES = { @@ -205,15 +191,36 @@ export class Transformer extends Group { this._node = node; this._resetTransformCache(); - node.on(TRANSFORM_CHANGE_STR, this._resetTransformCache.bind(this)); - node.on( - REDRAW_CHANGE_STR, - function() { - if (!this._transforming) { - this.update(); - } - }.bind(this) - ); + const additionalEvents = node._attrsAffectingSize + .map(prop => prop + 'Change.tr') + .join(' '); + + const upChange = () => { + this._resetTransformCache(); + if (!this._transforming) { + this.update(); + } + }; + node.on(additionalEvents, upChange); + + node.on(TRANSFORM_CHANGE_STR, upChange); + + // node.on( + // additionalEvents, + // function() { + // if (!this._transforming) { + // this.update(); + // } + // }.bind(this) + // ); + // node.on( + // REDRAW_CHANGE_STR, + // function() { + // if (!this._transforming) { + // this.update(); + // } + // }.bind(this) + // ); // we may need it if we set not in initial props // so elements are not defined yet @@ -221,7 +228,7 @@ export class Transformer extends Group { if (elementsCreated) { this.update(); } - return this; + return this; } // TODO: add docs, use overloaded setter/getter getNode() { @@ -237,7 +244,7 @@ export class Transformer extends Group { */ detach() { if (this.getNode()) { - this.getNode().off('.resizer'); + this.getNode().off('.tr'); this._node = undefined; } this._resetTransformCache(); diff --git a/src/shapes/Wedge.ts b/src/shapes/Wedge.ts index afde52d4..06408b71 100644 --- a/src/shapes/Wedge.ts +++ b/src/shapes/Wedge.ts @@ -50,14 +50,10 @@ export class Wedge extends Shape { return this.radius() * 2; } setWidth(width) { - if (this.radius() !== width / 2) { - this.radius(width / 2); - } + this.radius(width / 2); } setHeight(height) { - if (this.radius() !== height / 2) { - this.radius(height / 2); - } + this.radius(height / 2); } radius: GetSet; @@ -67,6 +63,7 @@ export class Wedge extends Shape { Wedge.prototype.className = 'Wedge'; Wedge.prototype._centroid = true; +Wedge.prototype._attrsAffectingSize = ['radius']; /** * get/set radius diff --git a/test/unit/shapes/Transformer-test.js b/test/unit/shapes/Transformer-test.js index c7921ab5..cfb7a1cd 100644 --- a/test/unit/shapes/Transformer-test.js +++ b/test/unit/shapes/Transformer-test.js @@ -1713,4 +1713,435 @@ suite('Transformer', function() { assert.equal(rect.width(), 100); assert.equal(rect.height(), 100); }); + + test('attrs change - arc', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Arc({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + innerRadius: 40, + outerRadius: 70, + angle: 60, + fill: 'yellow', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.outerRadius(100); + + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + shape.innerRadius(200); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + layer.draw(); + }); + + test('attrs change - line', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Arrow({ + x: stage.getWidth() / 4, + y: stage.getHeight() / 4, + points: [0, 0, stage.width() / 2, stage.height() / 2], + pointerLength: 20, + pointerWidth: 20, + fill: 'black', + stroke: 'black', + strokeWidth: 4, + draggable: true + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + layer.draw(); + + shape.points([10, 10, 100, 10]); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + layer.draw(); + assert.deepEqual(shape.getClientRect(), rect); + + shape.strokeWidth(10); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + layer.draw(); + assert.deepEqual(shape.getClientRect(), rect); + }); + + test('attrs change - circle', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Circle({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + radius: 40, + fill: 'yellow', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.radius(100); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + }); + + test('attrs change - ellipse', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Ellipse({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + radius: { + x: 100, + y: 50 + }, + fill: 'yellow', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.radiusX(120); + + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + shape.radiusY(100); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + layer.draw(); + }); + + test('attrs change - rect', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Rect({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + width: 100, + height: 100, + fill: 'yellow', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.width(120); + + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + shape.height(110); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + layer.draw(); + }); + + test('attrs change - path', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Path({ + x: 50, + y: 40, + data: + 'M12.582,9.551C3.251,16.237,0.921,29.021,7.08,38.564l-2.36,1.689l4.893,2.262l4.893,2.262l-0.568-5.36l-0.567-5.359l-2.365,1.694c-4.657-7.375-2.83-17.185,4.352-22.33c7.451-5.338,17.817-3.625,23.156,3.824c5.337,7.449,3.625,17.813-3.821,23.152l2.857,3.988c9.617-6.893,11.827-20.277,4.935-29.896C35.591,4.87,22.204,2.658,12.582,9.551z', + fill: 'green' + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.data('M200,100h100v50z'); + layer.draw(); + + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + }); + + test('attrs change - regular polygon', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.RegularPolygon({ + x: 100, + y: 150, + sides: 6, + radius: 70, + fill: 'red', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.radius(100); + layer.draw(); + + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + }); + + test('attrs change - ring', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Ring({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + innerRadius: 40, + outerRadius: 70, + fill: 'yellow', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.outerRadius(100); + + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + shape.innerRadius(200); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + layer.draw(); + }); + + test('attrs change - star', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Star({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + numPoints: 6, + innerRadius: 40, + outerRadius: 70, + fill: 'yellow', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.outerRadius(100); + + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + shape.innerRadius(200); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + + layer.draw(); + }); + + test('attrs change - wedge', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Wedge({ + x: stage.getWidth() / 2, + y: stage.getHeight() / 2, + radius: 70, + angle: 60, + fill: 'red', + stroke: 'black', + strokeWidth: 4 + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.radius(100); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect); + }); + + test('attrs change - text', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.Text({ + x: stage.getWidth() / 2, + y: 15, + text: 'Simple Text', + fontSize: 60, + fontFamily: 'Calibri', + fill: 'green' + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.text('Simple'); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change text'); + + shape.fontSize(30); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change font size'); + + shape.padding(10); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change padding'); + + shape.lineHeight(2); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change line height'); + + shape.width(30); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect), 'change width'; + + shape.height(30); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change height'); + }); + + test('attrs change - text path', function() { + var stage = addStage(); + var layer = new Konva.Layer(); + stage.add(layer); + + var shape = new Konva.TextPath({ + x: 0, + y: 50, + fill: '#333', + fontSize: 16, + fontFamily: 'Arial', + text: + "All the world's a stage, and all the men and women merely players.", + data: 'M10,10 C0,0 10,150 100,100 S300,150 400,50' + }); + layer.add(shape); + + var tr = new Konva.Transformer({ + node: shape + }); + layer.add(tr); + + shape.text('Simple'); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change text'); + + shape.fontSize(30); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect, 'change font size'); + + shape.data('M10,10 C0,0 10,150 100,100 S300,150 400,50'); + layer.draw(); + var rect = Object.assign({}, tr._getNodeRect()); + delete rect.rotation; + assert.deepEqual(shape.getClientRect(), rect), 'change data'; + }); });