diff --git a/CHANGELOG.md b/CHANGELOG.md index 3a402bb4..6cf86f5a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,11 +5,13 @@ This project adheres to [Semantic Versioning](http://semver.org/). ## Not released: -* `inherit` option is removed from `visible` and `listening`. They now just have boolean values `true` or `false`. If you do `group.listeng(false);` then whole group and all its children will be removed from the hitgraph (and they will not listen to events); +* `inherit` option is removed from `visible` and `listening`. They now just have boolean values `true` or `false`. If you do `group.listening(false);` then whole group and all its children will be removed from the hitgraph (and they will not listen to events); * `layer.hitGraphEnabled()` is deprecated. Just use `layer.listening(false)` instead * Some performance fixes and code size optimizations * Better support for font families with spaces inside (like `Font Awesome 5`). * Fix possible `dblclick` and `dbltap` triggers +* Deprecate `Konva.FastLayer`. Use `new Konva.Layer({ listening: false });` instead. +* Up to 20% performance boost for many moving nodes. ## 6.0.0 - 2020-05-08 diff --git a/konva.js b/konva.js index a9f2138f..fa75e8a7 100644 --- a/konva.js +++ b/konva.js @@ -8,7 +8,7 @@ * Konva JavaScript Framework v6.0.0 * http://konvajs.org/ * Licensed under the MIT - * Date: Tue Jun 09 2020 + * Date: Wed Jun 10 2020 * * Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS) * Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva) @@ -301,8 +301,17 @@ var Transform = /** @class */ (function () { function Transform(m) { if (m === void 0) { m = [1, 0, 0, 1, 0, 0]; } + this.dirty = false; this.m = (m && m.slice()) || [1, 0, 0, 1, 0, 0]; } + Transform.prototype.reset = function () { + this.m[0] = 1; + this.m[1] = 0; + this.m[2] = 0; + this.m[3] = 1; + this.m[4] = 0; + this.m[5] = 0; + }; /** * Copy Konva.Transform object * @method @@ -314,6 +323,14 @@ Transform.prototype.copy = function () { return new Transform(this.m); }; + Transform.prototype.copyInto = function (tr) { + tr.m[0] = this.m[0]; + tr.m[1] = this.m[1]; + tr.m[2] = this.m[2]; + tr.m[3] = this.m[3]; + tr.m[4] = this.m[4]; + tr.m[5] = this.m[5]; + }; /** * Transform point * @method @@ -325,7 +342,7 @@ var m = this.m; return { x: m[0] * point.x + m[2] * point.y + m[4], - y: m[1] * point.x + m[3] * point.y + m[5] + y: m[1] * point.x + m[3] * point.y + m[5], }; }; /** @@ -385,7 +402,7 @@ Transform.prototype.getTranslation = function () { return { x: this.m[4], - y: this.m[5] + y: this.m[5], }; }; /** @@ -491,7 +508,7 @@ scaleX: 0, scaleY: 0, skewX: 0, - skewY: 0 + skewY: 0, }; // Apply the QR-like decomposition. if (a != 0 || b != 0) { @@ -666,7 +683,7 @@ white: [255, 255, 255], whitesmoke: [245, 245, 245], yellow: [255, 255, 0], - yellowgreen: [154, 205, 5] + yellowgreen: [154, 205, 5], }, RGB_REGEX = /rgb\((\d{1,3}),(\d{1,3}),(\d{1,3})\)/, animQueue = []; /** * @namespace Util @@ -789,7 +806,7 @@ return { r: (bigint >> 16) & 255, g: (bigint >> 8) & 255, - b: bigint & 255 + b: bigint & 255, }; }, /** @@ -833,7 +850,7 @@ return { r: rgb[0], g: rgb[1], - b: rgb[2] + b: rgb[2], }; } else if (color[0] === HASH) { @@ -846,7 +863,7 @@ return { r: parseInt(rgb[1], 10), g: parseInt(rgb[2], 10), - b: parseInt(rgb[3], 10) + b: parseInt(rgb[3], 10), }; } else { @@ -854,7 +871,7 @@ return { r: 0, g: 0, - b: 0 + b: 0, }; } }, @@ -879,7 +896,7 @@ r: c[0], g: c[1], b: c[2], - a: 1 + a: 1, }; }, // Parse rgb(n, n, n) @@ -891,7 +908,7 @@ r: parts[0], g: parts[1], b: parts[2], - a: 1 + a: 1, }; } }, @@ -904,7 +921,7 @@ r: parts[0], g: parts[1], b: parts[2], - a: parts[3] + a: parts[3], }; } }, @@ -915,7 +932,7 @@ r: parseInt(str.slice(1, 3), 16), g: parseInt(str.slice(3, 5), 16), b: parseInt(str.slice(5, 7), 16), - a: 1 + a: 1, }; } }, @@ -926,7 +943,7 @@ r: parseInt(str[1] + str[1], 16), g: parseInt(str[2] + str[2], 16), b: parseInt(str[3] + str[3], 16), - a: 1 + a: 1, }; } }, @@ -948,7 +965,7 @@ r: Math.round(val), g: Math.round(val), b: Math.round(val), - a: 1 + a: 1, }; } if (l < 0.5) { @@ -985,7 +1002,7 @@ r: Math.round(rgb[0]), g: Math.round(rgb[1]), b: Math.round(rgb[2]), - a: 1 + a: 1, }; } }, @@ -1145,13 +1162,13 @@ for (n = 0; n < startArray.length; n += 2) { start.push({ x: startArray[n], - y: startArray[n + 1] + y: startArray[n + 1], }); } for (n = 0; n < endArray.length; n += 2) { end.push({ x: endArray[n], - y: endArray[n + 1] + y: endArray[n + 1], }); } var newStart = []; @@ -1206,7 +1223,7 @@ else { return evt.changedTouches[0].identifier; } - } + }, }; function _formatValue(val) { @@ -2660,9 +2677,16 @@ Node.prototype.getChildren = function () { return emptyChildren; }; - /** @lends Konva.Node.prototype */ Node.prototype._clearCache = function (attr) { - if (attr) { + // if we want to clear transform cache + // we don't really need to remove it from the cache + // but instead mark as "dirty" + // so we don't need to create a new instance next time + if ((attr === TRANSFORM || attr === ABSOLUTE_TRANSFORM) && + this._cache.get(attr)) { + this._cache.get(attr).dirty = true; + } + else if (attr) { this._cache.delete(attr); } else { @@ -2671,8 +2695,12 @@ }; Node.prototype._getCache = function (attr, privateGetter) { var cache = this._cache.get(attr); + // for transform the cache can be NOT empty + // but we still need to recalculate it if it is dirty + var isTransform = attr === TRANSFORM || attr === ABSOLUTE_TRANSFORM; + var invalid = cache === undefined || (isTransform && cache.dirty === true); // if not cached, we need to set it using the private getter method. - if (cache === undefined) { + if (invalid) { cache = privateGetter.call(this); this._cache.set(attr, cache); } @@ -3934,12 +3962,13 @@ } else { // try to use a cached value + at = this._cache.get(ABSOLUTE_TRANSFORM) || new Transform(); if (this.parent) { // transform will be cached - at = this.parent.getAbsoluteTransform().copy(); + this.parent.getAbsoluteTransform().copyInto(at); } else { - at = new Transform(); + at.reset(); } var transformsEnabled = this.transformsEnabled(); if (transformsEnabled === 'all') { @@ -3948,6 +3977,7 @@ else if (transformsEnabled === 'position') { at.translate(this.x() - this.offsetX(), this.y() - this.offsetY()); } + at.dirty = false; return at; } }; @@ -4009,7 +4039,9 @@ return this._getCache(TRANSFORM, this._getTransform); }; Node.prototype._getTransform = function () { - var m = new Transform(), x = this.x(), y = this.y(), rotation = Konva.getAngle(this.rotation()), scaleX = this.scaleX(), scaleY = this.scaleY(), skewX = this.skewX(), skewY = this.skewY(), offsetX = this.offsetX(), offsetY = this.offsetY(); + var m = this._cache.get(TRANSFORM) || new Transform(); + m.reset(); + var x = this.x(), y = this.y(), rotation = Konva.getAngle(this.rotation()), scaleX = this.scaleX(), scaleY = this.scaleY(), skewX = this.skewX(), skewY = this.skewY(), offsetX = this.offsetX(), offsetY = this.offsetY(); if (x !== 0 || y !== 0) { m.translate(x, y); } @@ -4025,6 +4057,7 @@ if (offsetX !== 0 || offsetY !== 0) { m.translate(-1 * offsetX, -1 * offsetY); } + m.dirty = false; return m; }; /** @@ -5465,7 +5498,8 @@ } if (cachedSceneCanvas) { context.save(); - layer._applyTransform(this, context, top); + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedSceneCanvas(context); context.restore(); } @@ -5481,7 +5515,8 @@ var layer = this.getLayer(), canvas = can || (layer && layer.hitCanvas), context = canvas && canvas.getContext(), cachedCanvas = this._getCanvasCache(), cachedHitCanvas = cachedCanvas && cachedCanvas.hit; if (cachedHitCanvas) { context.save(); - layer._applyTransform(this, context, top); + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedHitCanvas(context); context.restore(); } @@ -5490,9 +5525,8 @@ } return this; }; - // TODO: create ClipContainer Container.prototype._drawChildren = function (drawMethod, canvas, top) { - var layer = this.getLayer(), context = canvas && canvas.getContext(), clipWidth = this.clipWidth(), clipHeight = this.clipHeight(), clipFunc = this.clipFunc(), hasClip = (clipWidth && clipHeight) || clipFunc; + var context = canvas && canvas.getContext(), clipWidth = this.clipWidth(), clipHeight = this.clipHeight(), clipFunc = this.clipFunc(), hasClip = (clipWidth && clipHeight) || clipFunc; var selfCache = top === this; if (hasClip) { context.save(); @@ -7101,7 +7135,8 @@ // if node is cached we just need to draw from cache if (cachedCanvas) { context.save(); - layer._applyTransform(this, context, top); + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedSceneCanvas(context); context.restore(); return this; @@ -7158,7 +7193,8 @@ } if (cachedHitCanvas) { context.save(); - layer._applyTransform(this, context, top); + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedHitCanvas(context); context.restore(); return this; @@ -7291,7 +7327,6 @@ * // set hit stroke width always equals to scene stroke width * shape.hitStrokeWidth('auto'); */ - // TODO: probably we should deprecate it Factory.addGetterSetter(Shape, 'strokeHitEnabled', true, getBooleanValidator()); /** * **deprecated, use hitStrokeWidth instead!** get/set strokeHitEnabled property. Useful for performance optimization. @@ -8512,14 +8547,6 @@ } return this; }; - // the apply transform method is handled by the Layer and FastLayer class - // because it is up to the layer to decide if an absolute or relative transform - // should be used - // TODO: remove that method - Layer.prototype._applyTransform = function (shape, context, top) { - var m = shape.getAbsoluteTransform(top).getMatrix(); - context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); - }; /** * get visible intersection shape. This is the preferred * method for determining if a point intersects a shape or not @@ -15791,7 +15818,6 @@ * transformer.padding(10); */ Factory.addGetterSetter(Transformer, 'padding', 0, getNumberValidator()); - // TODO: that property is deprecated Factory.addGetterSetter(Transformer, 'node'); /** * get/set attached nodes of the Transformer. Transformer will adapt to their size and listen to their events diff --git a/konva.min.js b/konva.min.js index 0c770244..77ca1b91 100644 --- a/konva.min.js +++ b/konva.min.js @@ -3,10 +3,10 @@ * Konva JavaScript Framework v6.0.0 * http://konvajs.org/ * Licensed under the MIT - * Date: Tue Jun 09 2020 + * Date: Wed Jun 10 2020 * * 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;function t(t){var e=t.toLowerCase(),i=/(chrome)[ /]([\w.]+)/.exec(e)||/(webkit)[ /]([\w.]+)/.exec(e)||/(opera)(?:.*version|)[ /]([\w.]+)/.exec(e)||/(msie) ([\w.]+)/.exec(e)||e.indexOf("compatible")<0&&/(mozilla)(?:.*? rv:([\w.]+)|)/.exec(e)||[],n=!!t.match(/Android|BlackBerry|iPhone|iPad|iPod|Opera Mini|IEMobile/i),r=!!t.match(/IEMobile/i);return{browser:i[1]||"",version:i[2]||"0",isIE:function(t){var e=t.indexOf("msie ");if(0>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 p?{r:(e=p[t])[0],g:e[1],b:e[2]}:"#"===t[0]?this._hexToRgb(t.substring(1)):"rgb("===t.substr(0,4)?(e=u.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",A._namedColorToRBA(t)||A._hex3ColorToRGBA(t)||A._hex6ColorToRGBA(t)||A._rgbColorToRGBA(t)||A._rgbaColorToRGBA(t)||A._hslColorToRGBA(t)},_namedColorToRBA:function(t){var e=p[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}},_hslColorToRGBA:function(t){if(/hsl\((\d+),\s*([\d.]+)%,\s*([\d.]+)%\)/g.test(t)){var e=/hsl\((\d+),\s*([\d.]+)%,\s*([\d.]+)%\)/g.exec(t),i=(e[0],e.slice(1)),n=Number(i[0])/360,r=Number(i[1])/100,o=Number(i[2])/100,a=void 0,s=void 0,h=void 0;if(0==r)return h=255*o,{r:Math.round(h),g:Math.round(h),b:Math.round(h),a:1};for(var l=2*o-(a=o<.5?o*(1+r):o+r-o*r),c=[0,0,0],d=0;d<3;d++)(s=n+1/3*-(d-1))<0&&s++,1t.x+t.width||e.x+e.widtht.y+t.height||e.y+e.heighte.length&&(r=e,e=t,t=r),n=0;n=this.parent.children.length)&&A.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},dt.prototype.getAbsoluteOpacity=function(){return this._getCache(Z,this._getAbsoluteOpacity)},dt.prototype._getAbsoluteOpacity=function(){var t=this.opacity(),e=this.getParent();return e&&!e._isUnderCache&&(t*=e.getAbsoluteOpacity()),t},dt.prototype.moveTo=function(t){return this.getParent()!==t&&(this._remove(),t.add(this)),this},dt.prototype.toObject=function(){var t,e,i,n={},r=this.getAttrs();for(t in n.attrs={},r)e=r[t],A.isObject(e)&&!A._isPlainObject(e)&&!A._isArray(e)||(i="function"==typeof this[t]&&this[t],delete r[t],(i?i.call(this):null)!==(r[t]=e)&&(n.attrs[t]=e));return n.className=this.getClassName(),A._prepareToStringify(n)},dt.prototype.toJSON=function(){return JSON.stringify(this.toObject())},dt.prototype.getParent=function(){return this.parent},dt.prototype.findAncestors=function(t,e,i){var n=[];e&&this._isMatch(t)&&n.push(this);for(var r=this.parent;r;){if(r===i)return n;r._isMatch(t)&&n.push(r),r=r.parent}return n},dt.prototype.isAncestorOf=function(t){return!1},dt.prototype.findAncestor=function(t,e,i){return this.findAncestors(t,e,i)[0]},dt.prototype._isMatch=function(t){if(!t)return!1;if("function"==typeof t)return t(this);for(var e,i=t.replace(/ /g,"").split(","),n=i.length,r=0;rthis.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())},Ae.prototype.getTime=function(){return this._time},Ae.prototype.setPosition=function(t){this.prevPos=this._pos,this.propFunc(t),this._pos=t},Ae.prototype.getPosition=function(t){return void 0===t&&(t=this._time),this.func(t,this.begin,this._change,this.duration)},Ae.prototype.play=function(){this.state=2,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onPlay")},Ae.prototype.reverse=function(){this.state=3,this._time=this.duration-this._time,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onReverse")},Ae.prototype.seek=function(t){this.pause(),this._time=t,this.update(),this.fire("onSeek")},Ae.prototype.reset=function(){this.pause(),this._time=0,this.update(),this.fire("onReset")},Ae.prototype.finish=function(){this.pause(),this._time=this.duration,this.update(),this.fire("onFinish")},Ae.prototype.update=function(){this.setPosition(this.getPosition(this._time))},Ae.prototype.onEnterFrame=function(){var t=this.getTimer()-this._startTime;2===this.state?this.setTime(t):3===this.state&&this.setTime(this.duration-t)},Ae.prototype.pause=function(){this.state=1,this.fire("onPause")},Ae.prototype.getTimer=function(){return(new Date).getTime()},Ae);function Ae(t,e,i,n,r,o,a){this.prop=t,this.propFunc=e,this.begin=n,this._pos=n,this.duration=o,this._change=0,this.prevPos=0,this.yoyo=a,this._time=0,this._position=0,this._startTime=0,this._finish=0,this.func=i,this._change=r-this.begin,this.pause()}var Me=(Ge.prototype._addAttr=function(t,e){var i,n,r,o,a,s,h,l,c=this.node,d=c._id,p=Ge.tweens[d][t];if(p&&delete Ge.attrs[d][p][t],i=c.getAttr(t),A._isArray(e))if(n=[],o=Math.max(e.length,i.length),"points"===t&&e.length!==i.length&&(e.length>i.length?(s=i,i=A._prepareArrayForTween(i,e,c.closed())):(a=e,e=A._prepareArrayForTween(e,i,c.closed()))),0===t.indexOf("fill"))for(r=0;rthis.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 r=this.dataArray[i],o=r.points;switch(r.command){case"L":return ci.getPointOnLine(t,r.start.x,r.start.y,o[0],o[1]);case"C":return ci.getPointOnCubicBezier(t/r.pathLength,r.start.x,r.start.y,o[0],o[1],o[2],o[3],o[4],o[5]);case"Q":return ci.getPointOnQuadraticBezier(t/r.pathLength,r.start.x,r.start.y,o[0],o[1],o[2],o[3]);case"A":var a=o[0],s=o[1],h=o[2],l=o[3],c=o[4],d=o[5],p=o[6];return c+=d*t/r.pathLength,ci.getPointOnEllipticalArc(a,s,h,l,c,p)}return null},ci.getLineLength=function(t,e,i,n){return Math.sqrt((i-t)*(i-t)+(n-e)*(n-e))},ci.getPointOnLine=function(t,e,i,n,r,o,a){void 0===o&&(o=e),void 0===a&&(a=i);var s=(r-i)/(n-e+1e-8),h=Math.sqrt(t*t/(1+s*s));n>>1,A=_.slice(0,1+T),M=this._getTextWidth(A)+y;M<=l?(w=1+T,P=A+(g?"…":""),k=M):C=T}if(!P)break;if(!f||0<(S=(" "===(x=_[P.length])||"-"===x)&&k<=l?P.length:Math.max(P.lastIndexOf(" "),P.lastIndexOf("-"))+1)&&(w=S,P=P.slice(0,w),k=this._getTextWidth(P)),P=P.trimRight(),this._addTextLine(P),i=Math.max(i,k),d+=n,!u||s&&ce?g=li.getPointOnLine(e,f.x,f.y,y.points[0],y.points[1],f.x,f.y):y=void 0;break;case"A":var a=y.points[4],s=y.points[5],h=y.points[4]+s;0===m?m=a+1e-8:iy.pathLength?1e-8:e/y.pathLength:il.x?-1:1,d=this.findOne(".top-left").y()>l.y?-1:1,w=s*this.cos*c,C=s*this.sin*d,this.findOne(".top-left").x(l.x-w),this.findOne(".top-left").y(l.y-C)):"top-center"===this._movingAnchorName?this.findOne(".top-left").y(e.y()):"top-right"===this._movingAnchorName?(a&&(l=p?{x:this.width()/2,y:this.height()/2}:{x:this.findOne(".bottom-left").x(),y:this.findOne(".bottom-left").y()},s=Math.sqrt(Math.pow(e.x()-l.x,2)+Math.pow(l.y-e.y(),2)),c=this.findOne(".top-right").x()l.y?-1:1,w=s*this.cos*c,C=s*this.sin*d,this.findOne(".top-right").x(l.x+w),this.findOne(".top-right").y(l.y-C)),h=e.position(),this.findOne(".top-left").y(h.y),this.findOne(".bottom-right").x(h.x)):"middle-left"===this._movingAnchorName?this.findOne(".top-left").x(e.x()):"middle-right"===this._movingAnchorName?this.findOne(".bottom-right").x(e.x()):"bottom-left"===this._movingAnchorName?(a&&(l=p?{x:this.width()/2,y:this.height()/2}:{x:this.findOne(".top-right").x(),y:this.findOne(".top-right").y()},s=Math.sqrt(Math.pow(l.x-e.x(),2)+Math.pow(e.y()-l.y,2)),c=l.x>N,0!==w?(w=255/w,P[a]=(h*B>>N)*w,P[a+1]=(l*B>>N)*w,P[a+2]=(c*B>>N)*w):P[a]=P[a+1]=P[a+2]=0,h-=p,l-=u,c-=f,d-=g,p-=O.r,u-=O.g,f-=O.b,g-=O.a,r=s+((r=i+e+1)>N,0>N)*w,P[r+1]=(l*B>>N)*w,P[r+2]=(c*B>>N)*w):P[r]=P[r+1]=P[r+2]=0,h-=p,l-=u,c-=f,d-=g,p-=O.r,u-=O.g,f-=O.b,g-=O.a,r=i+((r=n+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 p?{r:(e=p[t])[0],g:e[1],b:e[2]}:"#"===t[0]?this._hexToRgb(t.substring(1)):"rgb("===t.substr(0,4)?(e=u.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",A._namedColorToRBA(t)||A._hex3ColorToRGBA(t)||A._hex6ColorToRGBA(t)||A._rgbColorToRGBA(t)||A._rgbaColorToRGBA(t)||A._hslColorToRGBA(t)},_namedColorToRBA:function(t){var e=p[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}},_hslColorToRGBA:function(t){if(/hsl\((\d+),\s*([\d.]+)%,\s*([\d.]+)%\)/g.test(t)){var e=/hsl\((\d+),\s*([\d.]+)%,\s*([\d.]+)%\)/g.exec(t),i=(e[0],e.slice(1)),n=Number(i[0])/360,r=Number(i[1])/100,o=Number(i[2])/100,a=void 0,s=void 0,h=void 0;if(0==r)return h=255*o,{r:Math.round(h),g:Math.round(h),b:Math.round(h),a:1};for(var c=2*o-(a=o<.5?o*(1+r):o+r-o*r),l=[0,0,0],d=0;d<3;d++)(s=n+1/3*-(d-1))<0&&s++,1t.x+t.width||e.x+e.widtht.y+t.height||e.y+e.heighte.length&&(r=e,e=t,t=r),n=0;n=this.parent.children.length)&&A.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},dt.prototype.getAbsoluteOpacity=function(){return this._getCache(Z,this._getAbsoluteOpacity)},dt.prototype._getAbsoluteOpacity=function(){var t=this.opacity(),e=this.getParent();return e&&!e._isUnderCache&&(t*=e.getAbsoluteOpacity()),t},dt.prototype.moveTo=function(t){return this.getParent()!==t&&(this._remove(),t.add(this)),this},dt.prototype.toObject=function(){var t,e,i,n={},r=this.getAttrs();for(t in n.attrs={},r)e=r[t],A.isObject(e)&&!A._isPlainObject(e)&&!A._isArray(e)||(i="function"==typeof this[t]&&this[t],delete r[t],(i?i.call(this):null)!==(r[t]=e)&&(n.attrs[t]=e));return n.className=this.getClassName(),A._prepareToStringify(n)},dt.prototype.toJSON=function(){return JSON.stringify(this.toObject())},dt.prototype.getParent=function(){return this.parent},dt.prototype.findAncestors=function(t,e,i){var n=[];e&&this._isMatch(t)&&n.push(this);for(var r=this.parent;r;){if(r===i)return n;r._isMatch(t)&&n.push(r),r=r.parent}return n},dt.prototype.isAncestorOf=function(t){return!1},dt.prototype.findAncestor=function(t,e,i){return this.findAncestors(t,e,i)[0]},dt.prototype._isMatch=function(t){if(!t)return!1;if("function"==typeof t)return t(this);for(var e,i=t.replace(/ /g,"").split(","),n=i.length,r=0;rthis.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())},Ae.prototype.getTime=function(){return this._time},Ae.prototype.setPosition=function(t){this.prevPos=this._pos,this.propFunc(t),this._pos=t},Ae.prototype.getPosition=function(t){return void 0===t&&(t=this._time),this.func(t,this.begin,this._change,this.duration)},Ae.prototype.play=function(){this.state=2,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onPlay")},Ae.prototype.reverse=function(){this.state=3,this._time=this.duration-this._time,this._startTime=this.getTimer()-this._time,this.onEnterFrame(),this.fire("onReverse")},Ae.prototype.seek=function(t){this.pause(),this._time=t,this.update(),this.fire("onSeek")},Ae.prototype.reset=function(){this.pause(),this._time=0,this.update(),this.fire("onReset")},Ae.prototype.finish=function(){this.pause(),this._time=this.duration,this.update(),this.fire("onFinish")},Ae.prototype.update=function(){this.setPosition(this.getPosition(this._time))},Ae.prototype.onEnterFrame=function(){var t=this.getTimer()-this._startTime;2===this.state?this.setTime(t):3===this.state&&this.setTime(this.duration-t)},Ae.prototype.pause=function(){this.state=1,this.fire("onPause")},Ae.prototype.getTimer=function(){return(new Date).getTime()},Ae);function Ae(t,e,i,n,r,o,a){this.prop=t,this.propFunc=e,this.begin=n,this._pos=n,this.duration=o,this._change=0,this.prevPos=0,this.yoyo=a,this._time=0,this._position=0,this._startTime=0,this._finish=0,this.func=i,this._change=r-this.begin,this.pause()}var Me=(Ge.prototype._addAttr=function(t,e){var i,n,r,o,a,s,h,c,l=this.node,d=l._id,p=Ge.tweens[d][t];if(p&&delete Ge.attrs[d][p][t],i=l.getAttr(t),A._isArray(e))if(n=[],o=Math.max(e.length,i.length),"points"===t&&e.length!==i.length&&(e.length>i.length?(s=i,i=A._prepareArrayForTween(i,e,l.closed())):(a=e,e=A._prepareArrayForTween(e,i,l.closed()))),0===t.indexOf("fill"))for(r=0;rthis.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 r=this.dataArray[i],o=r.points;switch(r.command){case"L":return li.getPointOnLine(t,r.start.x,r.start.y,o[0],o[1]);case"C":return li.getPointOnCubicBezier(t/r.pathLength,r.start.x,r.start.y,o[0],o[1],o[2],o[3],o[4],o[5]);case"Q":return li.getPointOnQuadraticBezier(t/r.pathLength,r.start.x,r.start.y,o[0],o[1],o[2],o[3]);case"A":var a=o[0],s=o[1],h=o[2],c=o[3],l=o[4],d=o[5],p=o[6];return l+=d*t/r.pathLength,li.getPointOnEllipticalArc(a,s,h,c,l,p)}return null},li.getLineLength=function(t,e,i,n){return Math.sqrt((i-t)*(i-t)+(n-e)*(n-e))},li.getPointOnLine=function(t,e,i,n,r,o,a){void 0===o&&(o=e),void 0===a&&(a=i);var s=(r-i)/(n-e+1e-8),h=Math.sqrt(t*t/(1+s*s));n>>1,A=_.slice(0,1+T),M=this._getTextWidth(A)+y;M<=c?(w=1+T,P=A+(g?"…":""),k=M):C=T}if(!P)break;if(!f||0<(S=(" "===(x=_[P.length])||"-"===x)&&k<=c?P.length:Math.max(P.lastIndexOf(" "),P.lastIndexOf("-"))+1)&&(w=S,P=P.slice(0,w),k=this._getTextWidth(P)),P=P.trimRight(),this._addTextLine(P),i=Math.max(i,k),d+=n,!u||s&&le?g=ci.getPointOnLine(e,f.x,f.y,y.points[0],y.points[1],f.x,f.y):y=void 0;break;case"A":var a=y.points[4],s=y.points[5],h=y.points[4]+s;0===m?m=a+1e-8:iy.pathLength?1e-8:e/y.pathLength:ic.x?-1:1,d=this.findOne(".top-left").y()>c.y?-1:1,w=s*this.cos*l,C=s*this.sin*d,this.findOne(".top-left").x(c.x-w),this.findOne(".top-left").y(c.y-C)):"top-center"===this._movingAnchorName?this.findOne(".top-left").y(e.y()):"top-right"===this._movingAnchorName?(a&&(c=p?{x:this.width()/2,y:this.height()/2}:{x:this.findOne(".bottom-left").x(),y:this.findOne(".bottom-left").y()},s=Math.sqrt(Math.pow(e.x()-c.x,2)+Math.pow(c.y-e.y(),2)),l=this.findOne(".top-right").x()c.y?-1:1,w=s*this.cos*l,C=s*this.sin*d,this.findOne(".top-right").x(c.x+w),this.findOne(".top-right").y(c.y-C)),h=e.position(),this.findOne(".top-left").y(h.y),this.findOne(".bottom-right").x(h.x)):"middle-left"===this._movingAnchorName?this.findOne(".top-left").x(e.x()):"middle-right"===this._movingAnchorName?this.findOne(".bottom-right").x(e.x()):"bottom-left"===this._movingAnchorName?(a&&(c=p?{x:this.width()/2,y:this.height()/2}:{x:this.findOne(".top-right").x(),y:this.findOne(".top-right").y()},s=Math.sqrt(Math.pow(c.x-e.x(),2)+Math.pow(e.y()-c.y,2)),l=c.x>N,0!==w?(w=255/w,P[a]=(h*B>>N)*w,P[a+1]=(c*B>>N)*w,P[a+2]=(l*B>>N)*w):P[a]=P[a+1]=P[a+2]=0,h-=p,c-=u,l-=f,d-=g,p-=O.r,u-=O.g,f-=O.b,g-=O.a,r=s+((r=i+e+1)>N,0>N)*w,P[r+1]=(c*B>>N)*w,P[r+2]=(l*B>>N)*w):P[r]=P[r+1]=P[r+2]=0,h-=p,c-=u,l-=f,d-=g,p-=O.r,u-=O.g,f-=O.b,g-=O.a,r=i+((r=n+R) extends Node< if (cachedSceneCanvas) { context.save(); - layer._applyTransform(this, context, top); + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedSceneCanvas(context); context.restore(); } else { @@ -365,7 +366,8 @@ export abstract class Container extends Node< if (cachedHitCanvas) { context.save(); - layer._applyTransform(this, context, top); + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedHitCanvas(context); context.restore(); } else { @@ -373,10 +375,8 @@ export abstract class Container extends Node< } return this; } - // TODO: create ClipContainer _drawChildren(drawMethod, canvas, top) { - var layer = this.getLayer(), - context = canvas && canvas.getContext(), + var context = canvas && canvas.getContext(), clipWidth = this.clipWidth(), clipHeight = this.clipHeight(), clipFunc = this.clipFunc(), diff --git a/src/Layer.ts b/src/Layer.ts index 0bcf34eb..ef5f1b9d 100644 --- a/src/Layer.ts +++ b/src/Layer.ts @@ -310,15 +310,6 @@ export abstract class Layer extends Container { return this; } - // the apply transform method is handled by the Layer and FastLayer class - // because it is up to the layer to decide if an absolute or relative transform - // should be used - // TODO: remove that method - _applyTransform(shape, context, top) { - var m = shape.getAbsoluteTransform(top).getMatrix(); - context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); - } - /** * get visible intersection shape. This is the preferred * method for determining if a point intersects a shape or not diff --git a/src/Node.ts b/src/Node.ts index 5eea72b8..003cca11 100644 --- a/src/Node.ts +++ b/src/Node.ts @@ -241,9 +241,17 @@ export abstract class Node { return emptyChildren; } - /** @lends Konva.Node.prototype */ _clearCache(attr?: string) { - if (attr) { + // if we want to clear transform cache + // we don't really need to remove it from the cache + // but instead mark as "dirty" + // so we don't need to create a new instance next time + if ( + (attr === TRANSFORM || attr === ABSOLUTE_TRANSFORM) && + this._cache.get(attr) + ) { + (this._cache.get(attr) as Transform).dirty = true; + } else if (attr) { this._cache.delete(attr); } else { this._cache.clear(); @@ -252,8 +260,13 @@ export abstract class Node { _getCache(attr: string, privateGetter: Function) { var cache = this._cache.get(attr); + // for transform the cache can be NOT empty + // but we still need to recalculate it if it is dirty + var isTransform = attr === TRANSFORM || attr === ABSOLUTE_TRANSFORM; + var invalid = cache === undefined || (isTransform && cache.dirty === true); + // if not cached, we need to set it using the private getter method. - if (cache === undefined) { + if (invalid) { cache = privateGetter.call(this); this._cache.set(attr, cache); } @@ -1690,11 +1703,12 @@ export abstract class Node { return at; } else { // try to use a cached value + at = this._cache.get(ABSOLUTE_TRANSFORM) || new Transform(); if (this.parent) { // transform will be cached - at = this.parent.getAbsoluteTransform().copy(); + this.parent.getAbsoluteTransform().copyInto(at); } else { - at = new Transform(); + at.reset(); } var transformsEnabled = this.transformsEnabled(); if (transformsEnabled === 'all') { @@ -1702,6 +1716,7 @@ export abstract class Node { } else if (transformsEnabled === 'position') { at.translate(this.x() - this.offsetX(), this.y() - this.offsetY()); } + at.dirty = false; return at; } } @@ -1766,8 +1781,10 @@ export abstract class Node { return this._getCache(TRANSFORM, this._getTransform) as Transform; } _getTransform(): Transform { - var m = new Transform(), - x = this.x(), + var m: Transform = this._cache.get(TRANSFORM) || new Transform(); + m.reset(); + + var x = this.x(), y = this.y(), rotation = Konva.getAngle(this.rotation()), scaleX = this.scaleX(), @@ -1793,6 +1810,8 @@ export abstract class Node { m.translate(-1 * offsetX, -1 * offsetY); } + m.dirty = false; + return m; } /** diff --git a/src/Shape.ts b/src/Shape.ts index 6283000c..5ca625af 100644 --- a/src/Shape.ts +++ b/src/Shape.ts @@ -567,7 +567,9 @@ export class Shape extends Node< // if node is cached we just need to draw from cache if (cachedCanvas) { context.save(); - layer._applyTransform(this, context, top); + + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); this._drawCachedSceneCanvas(context); context.restore(); return this; @@ -647,7 +649,10 @@ export class Shape extends Node< if (cachedHitCanvas) { context.save(); - layer._applyTransform(this, context, top); + + var m = this.getAbsoluteTransform(top).getMatrix(); + context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); + this._drawCachedHitCanvas(context); context.restore(); return this; @@ -874,7 +879,6 @@ Factory.addGetterSetter( * shape.hitStrokeWidth('auto'); */ -// TODO: probably we should deprecate it Factory.addGetterSetter(Shape, 'strokeHitEnabled', true, getBooleanValidator()); /** diff --git a/src/Util.ts b/src/Util.ts index 9aa73f7b..e0fdbbe8 100644 --- a/src/Util.ts +++ b/src/Util.ts @@ -49,7 +49,7 @@ export class Collection { } static _mapMethod(methodName: any) { - Collection.prototype[methodName] = function() { + Collection.prototype[methodName] = function () { var len = this.length, i; @@ -62,7 +62,7 @@ export class Collection { }; } - static mapMethods = function(constructor: Function) { + static mapMethods = function (constructor: Function) { var prot = constructor.prototype; for (var methodName in prot) { Collection._mapMethod(methodName); @@ -83,7 +83,7 @@ Collection.prototype = [] as any; * shape.setX(10); * }); */ -Collection.prototype.each = function(func) { +Collection.prototype.each = function (func) { for (var n = 0; n < this.length; n++) { func(this[n], n); } @@ -93,7 +93,7 @@ Collection.prototype.each = function(func) { * @method * @name Konva.Collection#toArray */ -Collection.prototype.toArray = function() { +Collection.prototype.toArray = function () { var arr = [], len = this.length, n; @@ -130,9 +130,18 @@ Collection.prototype.toArray = function() { */ export class Transform { m: Array; + dirty = false; constructor(m = [1, 0, 0, 1, 0, 0]) { this.m = (m && m.slice()) || [1, 0, 0, 1, 0, 0]; } + reset() { + this.m[0] = 1; + this.m[1] = 0; + this.m[2] = 0; + this.m[3] = 1; + this.m[4] = 0; + this.m[5] = 0; + } /** * Copy Konva.Transform object * @method @@ -144,6 +153,14 @@ export class Transform { copy() { return new Transform(this.m); } + copyInto(tr: Transform) { + tr.m[0] = this.m[0]; + tr.m[1] = this.m[1]; + tr.m[2] = this.m[2]; + tr.m[3] = this.m[3]; + tr.m[4] = this.m[4]; + tr.m[5] = this.m[5]; + } /** * Transform point * @method @@ -155,7 +172,7 @@ export class Transform { var m = this.m; return { x: m[0] * point.x + m[2] * point.y + m[4], - y: m[1] * point.x + m[3] * point.y + m[5] + y: m[1] * point.x + m[3] * point.y + m[5], }; } /** @@ -215,7 +232,7 @@ export class Transform { getTranslation() { return { x: this.m[4], - y: this.m[5] + y: this.m[5], }; } /** @@ -334,7 +351,7 @@ export class Transform { scaleX: 0, scaleY: 0, skewX: 0, - skewY: 0 + skewY: 0, }; // Apply the QR-like decomposition. @@ -525,7 +542,7 @@ var OBJECT_ARRAY = '[object Array]', white: [255, 255, 255], whitesmoke: [245, 245, 245], yellow: [255, 255, 0], - yellowgreen: [154, 205, 5] + yellowgreen: [154, 205, 5], }, RGB_REGEX = /rgb\((\d{1,3}),(\d{1,3}),(\d{1,3})\)/, animQueue: Array = []; @@ -592,10 +609,10 @@ export const Util = { requestAnimFrame(callback: Function) { animQueue.push(callback); if (animQueue.length === 1) { - requestAnimationFrame(function() { + requestAnimationFrame(function () { const queue = animQueue; animQueue = []; - queue.forEach(function(cb) { + queue.forEach(function (cb) { cb(); }); }); @@ -646,7 +663,7 @@ export const Util = { _urlToImage(url: string, callback: Function) { // if arg is a string, then it's a data url var imageObj = new glob.Image(); - imageObj.onload = function() { + imageObj.onload = function () { callback(imageObj); }; imageObj.src = url; @@ -660,7 +677,7 @@ export const Util = { return { r: (bigint >> 16) & 255, g: (bigint >> 8) & 255, - b: bigint & 255 + b: bigint & 255, }; }, /** @@ -704,7 +721,7 @@ export const Util = { return { r: rgb[0], g: rgb[1], - b: rgb[2] + b: rgb[2], }; } else if (color[0] === HASH) { // hex @@ -715,14 +732,14 @@ export const Util = { return { r: parseInt(rgb[1], 10), g: parseInt(rgb[2], 10), - b: parseInt(rgb[3], 10) + b: parseInt(rgb[3], 10), }; } else { // default return { r: 0, g: 0, - b: 0 + b: 0, }; } }, @@ -749,7 +766,7 @@ export const Util = { r: c[0], g: c[1], b: c[2], - a: 1 + a: 1, }; }, // Parse rgb(n, n, n) @@ -761,7 +778,7 @@ export const Util = { r: parts[0], g: parts[1], b: parts[2], - a: 1 + a: 1, }; } }, @@ -774,7 +791,7 @@ export const Util = { r: parts[0], g: parts[1], b: parts[2], - a: parts[3] + a: parts[3], }; } }, @@ -785,7 +802,7 @@ export const Util = { r: parseInt(str.slice(1, 3), 16), g: parseInt(str.slice(3, 5), 16), b: parseInt(str.slice(5, 7), 16), - a: 1 + a: 1, }; } }, @@ -796,7 +813,7 @@ export const Util = { r: parseInt(str[1] + str[1], 16), g: parseInt(str[2] + str[2], 16), b: parseInt(str[3] + str[3], 16), - a: 1 + a: 1, }; } }, @@ -821,7 +838,7 @@ export const Util = { r: Math.round(val), g: Math.round(val), b: Math.round(val), - a: 1 + a: 1, }; } @@ -861,7 +878,7 @@ export const Util = { r: Math.round(rgb[0]), g: Math.round(rgb[1]), b: Math.round(rgb[2]), - a: 1 + a: 1, }; } }, @@ -1016,7 +1033,7 @@ export const Util = { _getProjectionToLine(pt: Vector2d, line, isClosed) { var pc = Util.cloneObject(pt); var dist = Number.MAX_VALUE; - line.forEach(function(p1, i) { + line.forEach(function (p1, i) { if (!isClosed && i === line.length - 1) { return; } @@ -1052,18 +1069,18 @@ export const Util = { for (n = 0; n < startArray.length; n += 2) { start.push({ x: startArray[n], - y: startArray[n + 1] + y: startArray[n + 1], }); } for (n = 0; n < endArray.length; n += 2) { end.push({ x: endArray[n], - y: endArray[n + 1] + y: endArray[n + 1], }); } var newStart = []; - end.forEach(function(point) { + end.forEach(function (point) { var pr = Util._getProjectionToLine(point, start, isClosed); newStart.push(pr.x); newStart.push(pr.y); @@ -1118,5 +1135,5 @@ export const Util = { } else { return evt.changedTouches[0].identifier; } - } + }, }; diff --git a/src/shapes/Transformer.ts b/src/shapes/Transformer.ts index b29c17c6..53b29429 100644 --- a/src/shapes/Transformer.ts +++ b/src/shapes/Transformer.ts @@ -1531,7 +1531,6 @@ Factory.addGetterSetter(Transformer, 'ignoreStroke', false); */ Factory.addGetterSetter(Transformer, 'padding', 0, getNumberValidator()); -// TODO: that property is deprecated Factory.addGetterSetter(Transformer, 'node'); /** diff --git a/test/performance/bunnies.html b/test/performance/bunnies.html index 6f5e62ae..61cc43bf 100644 --- a/test/performance/bunnies.html +++ b/test/performance/bunnies.html @@ -10,7 +10,8 @@
- + +