From cb8d36c3937add27ebd169bdd5a08bf661b9e5ac Mon Sep 17 00:00:00 2001 From: Eric Rowell Date: Tue, 11 Dec 2012 00:08:59 -0800 Subject: [PATCH] fixed several bugs with toImage(). toImage() can now also accept x and y params. added extensive caching unit test. removed drawBuffer logic as it is no longer needed --- src/Container.js | 12 +--- src/Layer.js | 48 ++----------- src/Node.js | 17 +++-- src/Shape.js | 31 +-------- src/Stage.js | 2 + tests/assets/unitDataUrls.js | 1 + tests/js/unit/nodeTests.js | 131 ++++++++++++++++++++++++++++++----- 7 files changed, 135 insertions(+), 107 deletions(-) diff --git a/src/Container.js b/src/Container.js index 5caf22a4..b6fda05f 100644 --- a/src/Container.js +++ b/src/Container.js @@ -241,11 +241,11 @@ this.drawScene(); this.drawHit(); }, - drawScene: function() { + drawScene: function(canvas) { if(this.isVisible()) { var children = this.children, len = children.length; for(var n = 0; n < len; n++) { - children[n].drawScene(); + children[n].drawScene(canvas); } } }, @@ -256,14 +256,6 @@ children[n].drawHit(); } } - }, - drawBuffer: function(canvas) { - if(this.isVisible()) { - var children = this.children, len = children.length; - for(var n = 0; n < len; n++) { - children[n].drawBuffer(canvas); - } - } } }; Kinetic.Global.extend(Kinetic.Container, Kinetic.Node); diff --git a/src/Layer.js b/src/Layer.js index e55eb389..d3f45677 100644 --- a/src/Layer.js +++ b/src/Layer.js @@ -81,11 +81,12 @@ * @methodOf Kinetic.Layer.prototype * @param {Kinetic.Canvas} [canvas] */ - drawScene: function() { + drawScene: function(canvas) { + canvas = canvas || this.getCanvas(); if(this.attrs.clearBeforeDraw) { - this.getCanvas().clear(); + canvas.clear(); } - Kinetic.Container.prototype.drawScene.call(this); + Kinetic.Container.prototype.drawScene.call(this, canvas); }, /** * set before draw handler @@ -201,47 +202,6 @@ getLayer: function() { return this; }, - /** - * Creates a composite data URL. If MIME type is not - * specified, then "image/png" will result. For "image/jpeg", specify a quality - * level as quality (range 0.0 - 1.0). Note that this method works - * differently from toDataURL() for other nodes because it generates an absolute dataURL - * based on what's currently drawn on the layer, rather than drawing - * the current state of each child node - * @name toDataURL - * @methodOf Kinetic.Layer.prototype - * @param {Object} config - * @param {String} [config.mimeType] mime type. can be "image/png" or "image/jpeg". - * "image/png" is the default - * @param {Number} [config.width] data url image width - * @param {Number} [config.height] data url image height - * @param {Number} [config.quality] jpeg quality. If using an "image/jpeg" mimeType, - * you can specify the quality from 0 to 1, where 0 is very poor quality and 1 - * is very high quality - */ - toDataURL: function(config) { - var canvas; - var mimeType = config && config.mimeType ? config.mimeType : null; - var quality = config && config.quality ? config.quality : null; - - /* - * if layer is hidden, return blank canvas - * else if width and height are defined, create blank canvas and draw onto it - * else return canvas as is - */ - if(!this.isVisible()) { - var stage = this.getStage(); - canvas = new Kinetic.SceneCanvas(stage.getWidth(), stage.getHeight()); - } - else if(config && config.width && config.height) { - canvas = new Kinetic.SceneCanvas(config.width, config.height); - this.draw(canvas); - } - else { - canvas = this.getCanvas(); - } - return canvas.toDataURL(mimeType, quality); - }, /** * remove layer from stage */ diff --git a/src/Node.js b/src/Node.js index 696bf2c1..bb35844a 100644 --- a/src/Node.js +++ b/src/Node.js @@ -738,20 +738,25 @@ * is very high quality */ toDataURL: function(config) { - var mimeType = config && config.mimeType ? config.mimeType : null; - var quality = config && config.quality ? config.quality : null; - var canvas; + config = config || {}; + var mimeType = config.mimeType || null, quality = config.quality || null, canvas, context, x = config.x || 0, y = config.y || 0; - //if width and height are defined, create new canvas to draw on, else reuse stage hit canvas - if(config && config.width && config.height) { + //if width and height are defined, create new canvas to draw on, else reuse stage buffer canvas + if(config.width && config.height) { canvas = new Kinetic.SceneCanvas(config.width, config.height); } else { canvas = this.getStage().bufferCanvas; canvas.clear(); } + context = canvas.getContext(); + context.save(); + if(x || y) { + context.translate(-1 * x, -1 * y); + } + this.drawScene(canvas); + context.restore(); - this.drawBuffer(canvas); return canvas.toDataURL(mimeType, quality); }, /** diff --git a/src/Shape.js b/src/Shape.js index 56e6ff7c..063fada4 100644 --- a/src/Shape.js +++ b/src/Shape.js @@ -209,7 +209,7 @@ var stage = this.getStage(); var hitCanvas = stage.hitCanvas; hitCanvas.clear(); - this.drawBuffer(hitCanvas); + this.drawScene(hitCanvas); var p = hitCanvas.context.getImageData(Math.round(pos.x), Math.round(pos.y), 1, 1).data; return p[3] > 0; }, @@ -217,33 +217,8 @@ Kinetic.Node.prototype.remove.call(this); delete Kinetic.Global.shapes[this.colorKey]; }, - drawBuffer: function(canvas) { - var attrs = this.attrs, drawFunc = attrs.drawFunc, context = canvas.getContext(); - - if(drawFunc && this.isVisible()) { - var stage = this.getStage(), family = [], parent = this.parent; - - family.unshift(this); - while(parent) { - family.unshift(parent); - parent = parent.parent; - } - - context.save(); - canvas._applyOpacity(this); - canvas._applyLineJoin(this); - var len = family.length; - for(var n = 0; n < len; n++) { - var node = family[n], t = node.getTransform(), m = t.getMatrix(); - context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); - } - - drawFunc.call(this, canvas); - context.restore(); - } - }, - drawScene: function() { - var attrs = this.attrs, drawFunc = attrs.drawFunc, canvas = this.getLayer().getCanvas(), context = canvas.getContext(); + drawScene: function(canvas) { + var attrs = this.attrs, drawFunc = attrs.drawFunc, canvas = canvas || this.getLayer().getCanvas(), context = canvas.getContext(); if(drawFunc && this.isVisible()) { var stage = this.getStage(), family = [], parent = this.parent; diff --git a/src/Stage.js b/src/Stage.js index 13db2946..62ef9cc4 100644 --- a/src/Stage.js +++ b/src/Stage.js @@ -193,6 +193,8 @@ var height = config && config.height ? config.height : this.attrs.height; var canvas = new Kinetic.SceneCanvas(width, height); var context = canvas.getContext(); + + var layers = this.children; function drawLayer(n) { diff --git a/tests/assets/unitDataUrls.js b/tests/assets/unitDataUrls.js index cc2b18a8..a42c3fab 100644 --- a/tests/assets/unitDataUrls.js +++ b/tests/assets/unitDataUrls.js @@ -4,6 +4,7 @@ * urls will be slightly different from browser to browser */ var dataUrls = { + 'cache shape, group, layer, and stage': '', 'crop and scale image' : '', 'rotate wedge': '', 'wedge': '', diff --git a/tests/js/unit/nodeTests.js b/tests/js/unit/nodeTests.js index 7cbc457d..8f5a8d96 100644 --- a/tests/js/unit/nodeTests.js +++ b/tests/js/unit/nodeTests.js @@ -770,23 +770,116 @@ Test.Modules.NODE = { } }); - /* - group.toImage({ - callback: function(imageObj) { - test(Kinetic.Type._isElement(imageObj), 'group toImage() should be an image object'); - } - }); - layer.toImage({ - callback: function(imageObj) { - test(Kinetic.Type._isElement(imageObj), 'layer toImage() should be an image object'); - } - }); - stage.toImage({ - callback: function(imageObj) { - test(Kinetic.Type._isElement(imageObj), 'stage toImage() should be an image object'); - } - }); - */ + showHit(layer); + }, + 'cache shape, group, layer, and stage': function(containerId) { + var stage = new Kinetic.Stage({ + container: containerId, + width: 578, + height: 200 + }); + var layer = new Kinetic.Layer(); + var group = new Kinetic.Group(); + var rect = new Kinetic.Rect({ + x: 10, + y: 10, + width: 50, + height: 30, + fill: 'green', + stroke: 'black', + strokeWidth: 4, + draggable: true + }); + + group.add(rect); + layer.add(group); + stage.add(layer); + + // cache shape + rect.toImage({ + x: 8, + y: 8, + width: 54, + height: 34, + callback: function(imageObj) { + var cachedShape = new Kinetic.Image({ + image: imageObj, + x: 60, + y: 60, + draggable: true + }); + group.add(cachedShape); + layer.draw(); + + // cache group + group.toImage({ + x: 8, + y: 8, + width: 106, + height: 86, + callback: function(imageObj) { + var cachedGroup = new Kinetic.Image({ + image: imageObj, + x: 100, + y: 8, + draggable: true + }); + group.add(cachedGroup); + layer.draw(); + + + // cache layer + layer.toImage({ + x: 8, + y: 8, + width: 200, + height: 86, + callback: function(imageObj) { + + var cachedLayer = new Kinetic.Image({ + image: imageObj, + x: 190, + y: 8, + draggable: true + }); + group.add(cachedLayer); + layer.draw(); + + + var dataUrl = layer.toDataURL(); + + // cache stage + + stage.toImage({ + x: 8, + y: 8, + width: 400, + height: 86, + callback: function(imageObj) { + + var cachedStage = new Kinetic.Image({ + image: imageObj, + x: 8, + y: 100, + draggable: true + }); + group.add(cachedStage); + layer.draw(); + + var dataUrl = layer.toDataURL(); + //console.log(dataUrl); + + warn(dataUrl === dataUrls['cache shape, group, layer, and stage'], 'problem caching shape, group, layer, and stage'); + } + }); + + + } + }); + } + }); + } + }); showHit(layer); }, @@ -2009,7 +2102,7 @@ Test.Modules.NODE = { var group = new Kinetic.Group(); var drawTriangle = function(canvas) { - var context = canvas.getContext(); + var context = canvas.getContext(); context.beginPath(); context.moveTo(200, 50); context.lineTo(420, 80); @@ -2053,7 +2146,7 @@ Test.Modules.NODE = { }, 'load stage with custom shape using json': function(containerId) { var drawTriangle = function(canvas) { - var context = canvas.getContext(); + var context = canvas.getContext(); context.beginPath(); context.moveTo(200, 50); context.lineTo(420, 80);