diff --git a/Thorfile b/Thorfile index 7ba0834c..eab292b7 100644 --- a/Thorfile +++ b/Thorfile @@ -17,7 +17,6 @@ class Build < Thor "tests/js/unit/layerTests.js", "tests/js/unit/shapeTests.js", "tests/js/unit/ddTests.js", - "tests/js/unit/customShapeTests.js", "tests/js/unit/animationTests.js", "tests/js/unit/transitionTests.js", "tests/js/unit/shapes/rectTests.js", diff --git a/src/Shape.js b/src/Shape.js index 564f9a84..6ef1ec41 100644 --- a/src/Shape.js +++ b/src/Shape.js @@ -64,7 +64,6 @@ Kinetic.Shape = (function() { Shape.prototype = { _initShape: function(config) { this.nodeType = 'Shape'; - this.appliedShadow = false; // set colorKey var shapes = Kinetic.Global.shapes; @@ -104,36 +103,35 @@ Kinetic.Shape = (function() { * @name stroke * @methodOf Kinetic.Shape.prototype */ - stroke: function(context) { + stroke: function(context, stroke, strokeWidth, shadow) { if(context.type === 'scene') { - this._strokeScene(context); + return this._strokeScene(context, stroke, strokeWidth, shadow); } else if(context.type === 'hit') { - this._strokeHit(context); + return this._strokeHit(context, strokeWidth); } + return false; }, - _strokeScene: function(context) { - var strokeWidth = this.getStrokeWidth(), stroke = this.getStroke(); + _strokeScene: function(context, stroke, strokeWidth, shadow) { if(stroke || strokeWidth) { - var appliedShadow = false; - context.save(); - if(this.attrs.shadow && !this.appliedShadow) { - appliedShadow = this._applyShadow(context); - } - + var appliedShadow = this._applyShadow(context, shadow); context.lineWidth = strokeWidth || 2; context.strokeStyle = stroke || 'black'; context.stroke(context); context.restore(); if(appliedShadow) { - this.stroke(context); + if(shadow.opacity) { + this._strokeScene(context, stroke); + } + return true; } } + return false; }, - _strokeHit: function(context) { - var strokeWidth = this.getStrokeWidth(), stroke = this.colorKey; + _strokeHit: function(context, strokeWidth) { + var stroke = this.colorKey; if(stroke || strokeWidth) { context.save(); context.lineWidth = strokeWidth || 2; @@ -141,6 +139,7 @@ Kinetic.Shape = (function() { context.stroke(context); context.restore(); } + return false; }, _getFillType: function(fill) { var type = Kinetic.Type; @@ -163,27 +162,42 @@ Kinetic.Shape = (function() { return 'UNKNOWN'; } }, + /** + * helper method to fill and stroke shape based on fill and stroke properties, + * and also automatically handle shadows, line joins, and line caps + * @name render + * @methodOf Kinetic.Shape.prototype + */ + render: function(context) { + this.applyLineJoin(context, this.getLineJoin()); + this.applyLineCap(context, this.getLineCap()); + if(context.type === 'scene') { + this.applyOpacity(context); + } + + var appliedShadow = this.fill(context, this.getFill(), this.getShadow()); + this.stroke(context, this.getStroke(), this.getStrokeWidth(), appliedShadow ? null : this.getShadow()); + }, /** * helper method to fill the shape * @name fill * @methodOf Kinetic.Shape.prototype - * */ - fill: function(context) { + */ + fill: function(context, fill, shadow) { if(context.type === 'scene') { - this._fillScene(context); + return this._fillScene(context, fill, shadow); } else if(context.type === 'hit') { - this._fillHit(context); + return this._fillHit(context); } + return false; }, - _fillScene: function(context) { - var appliedShadow = false, fill = this.getFill(), fillType = this._getFillType(fill); + _fillScene: function(context, fill, shadow) { + var fillType = this._getFillType(fill); if(fill) { context.save(); - if(this.attrs.shadow && !this.appliedShadow) { - appliedShadow = this._applyShadow(context); - } + var appliedShadow = this._applyShadow(context, shadow); var s = fill.start; var e = fill.end; @@ -233,36 +247,36 @@ Kinetic.Shape = (function() { context.fill(context); break; } - context.restore(); - } - if(appliedShadow) { - this.fill(context); + context.restore(); + + if(appliedShadow) { + if(shadow.opacity) { + this._fillScene(context, fill); + } + return true; + } } + return false; }, _fillHit: function(context) { context.save(); context.fillStyle = this.colorKey; context.fill(context); context.restore(); + return false; }, /** - * helper method to draw an image and apply - * a shadow if needed + * helper method to draw an image * @name drawImage * @methodOf Kinetic.Shape.prototype */ drawImage: function() { - var appliedShadow = false; var context = arguments[0]; context.save(); var a = Array.prototype.slice.call(arguments); if(a.length === 6 || a.length === 10) { - if(this.attrs.shadow && !this.appliedShadow) { - appliedShadow = this._applyShadow(context); - } - if(a.length === 6) { context.drawImage(a[1], a[2], a[3], a[4], a[5]); } @@ -272,10 +286,6 @@ Kinetic.Shape = (function() { } context.restore(); - - if(appliedShadow) { - this.drawImage.apply(this, a); - } }, applyOpacity: function(context) { var absOpacity = this.getAbsoluteOpacity(); @@ -288,9 +298,10 @@ Kinetic.Shape = (function() { * based on the applyLineJoin property * @name lineJoin * @methodOf Kinetic.Shape.prototype + * @param {CanvasContext} context + * @param {String} lineJoin */ - applyLineJoin: function(context) { - var lineJoin = this.attrs.lineJoin; + applyLineJoin: function(context, lineJoin) { if(lineJoin) { context.lineJoin = lineJoin; } @@ -300,9 +311,10 @@ Kinetic.Shape = (function() { * based on the lineCap property * @name applyLineCap * @methodOf Kinetic.Shape.prototype + * @param {CanvasContext} context + * @param {String} lineCap */ - applyLineCap: function(context) { - var lineCap = this.attrs.lineCap; + applyLineCap: function(context, lineCap) { if(lineCap) { context.lineCap = lineCap; } @@ -379,30 +391,24 @@ Kinetic.Shape = (function() { _get: function(selector) { return this.nodeType === selector || this.shapeType === selector ? [this] : []; }, - /** - * apply shadow. return true if shadow was applied - * and false if it was not - */ - _applyShadow: function(context) { - var s = this.attrs.shadow; - if(s) { + _applyShadow: function(context, shadow) { + if(shadow) { var aa = this.getAbsoluteOpacity(); // defaults - var color = s.color ? s.color : 'black'; - var blur = s.blur ? s.blur : 5; - var offset = s.offset ? s.offset : { + var color = shadow.color || 'black'; + var blur = shadow.blur || 5; + var offset = shadow.offset || { x: 0, y: 0 }; - if(s.opacity) { - context.globalAlpha = s.opacity * aa; + if(shadow.opacity) { + context.globalAlpha = shadow.opacity * aa; } context.shadowColor = color; context.shadowBlur = blur; context.shadowOffsetX = offset.x; context.shadowOffsetY = offset.y; - this.appliedShadow = true; return true; } @@ -447,11 +453,6 @@ Kinetic.Shape = (function() { context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); } - this.applyOpacity(context); - this.applyLineJoin(context); - this.applyLineCap(context); - this.appliedShadow = false; - drawFunc.call(this, context); context.restore(); } @@ -474,12 +475,7 @@ Kinetic.Shape = (function() { var node = family[n], t = node.getTransform(), m = t.getMatrix(); context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); } - - this.applyOpacity(context); - this.applyLineJoin(context); - this.applyLineCap(context); - this.appliedShadow = false; - + drawFunc.call(this, context); context.restore(); } @@ -503,10 +499,6 @@ Kinetic.Shape = (function() { context.transform(m[0], m[1], m[2], m[3], m[4], m[5]); } - // don't draw shadows on hit context - this.applyLineJoin(context); - this.applyLineCap(context); - drawFunc.call(this, context); context.restore(); } @@ -523,7 +515,7 @@ Kinetic.Shape = (function() { Kinetic.Global.extend(Shape, Kinetic.Node); // add getters and setters - Kinetic.Node.addGettersSetters(Shape, ['stroke', 'lineJoin', 'strokeWidth', 'drawFunc', 'drawHitFunc', 'cornerRadius']); + Kinetic.Node.addGettersSetters(Shape, ['stroke', 'lineJoin', 'lineCap', 'strokeWidth', 'drawFunc', 'drawHitFunc', 'cornerRadius']); Kinetic.Node.addGetters(Shape, ['shadow', 'fill']); /** diff --git a/src/shapes/Circle.js b/src/shapes/Circle.js index d6005592..006545a3 100644 --- a/src/shapes/Circle.js +++ b/src/shapes/Circle.js @@ -27,8 +27,7 @@ Kinetic.Circle.prototype = { context.beginPath(); context.arc(0, 0, this.getRadius(), 0, Math.PI * 2, true); context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); }, getWidth: function() { return this.getRadius() * 2; diff --git a/src/shapes/Ellipse.js b/src/shapes/Ellipse.js index f3329310..c383d571 100644 --- a/src/shapes/Ellipse.js +++ b/src/shapes/Ellipse.js @@ -36,8 +36,7 @@ Kinetic.Ellipse.prototype = { context.arc(0, 0, r.x, 0, Math.PI * 2, true); context.restore(); context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); }, /** * set radius diff --git a/src/shapes/Image.js b/src/shapes/Image.js index eafe8e0d..1e382dd9 100644 --- a/src/shapes/Image.js +++ b/src/shapes/Image.js @@ -36,8 +36,7 @@ Kinetic.Image.prototype = { context.beginPath(); context.rect(0, 0, width, height); context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); if(this.attrs.image) { // if cropping @@ -55,19 +54,16 @@ Kinetic.Image.prototype = { } }, drawHitFunc: function(context) { - var width = this.getWidth(), height = this.getHeight(), imageBuffer = this.imageBuffer; + var width = this.getWidth(), height = this.getHeight(), imageBuffer = this.imageBuffer, appliedShadow = false; context.beginPath(); context.rect(0, 0, width, height); context.closePath(); + this.render(context); if(imageBuffer) { this.drawImage(context, this.imageBuffer, 0, 0, width, height); } - else { - this.fill(context); - } - this.stroke(context); }, /** * apply filter diff --git a/src/shapes/Line.js b/src/shapes/Line.js index a4a83a75..624fb89b 100644 --- a/src/shapes/Line.js +++ b/src/shapes/Line.js @@ -47,7 +47,7 @@ Kinetic.Line.prototype = { } } - this.stroke(context); + this.stroke(context, this.getStroke(), this.getStrokeWidth(), this.getShadow()); }, /** * set points array diff --git a/src/shapes/Path.js b/src/shapes/Path.js index 7abb89d1..38b8d1d0 100644 --- a/src/shapes/Path.js +++ b/src/shapes/Path.js @@ -68,8 +68,7 @@ Kinetic.Path.prototype = { break; } } - this.fill(context); - this.stroke(context); + this.render(context); } }; Kinetic.Global.extend(Kinetic.Path, Kinetic.Shape); diff --git a/src/shapes/Polygon.js b/src/shapes/Polygon.js index 4b01b588..ab0befc3 100644 --- a/src/shapes/Polygon.js +++ b/src/shapes/Polygon.js @@ -30,8 +30,7 @@ Kinetic.Polygon.prototype = { context.lineTo(this.attrs.points[n].x, this.attrs.points[n].y); } context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); }, /** * set points array diff --git a/src/shapes/Rect.js b/src/shapes/Rect.js index 3e9efac7..e08bb2cf 100644 --- a/src/shapes/Rect.js +++ b/src/shapes/Rect.js @@ -42,8 +42,7 @@ Kinetic.Rect.prototype = { } context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); } }; diff --git a/src/shapes/RegularPolygon.js b/src/shapes/RegularPolygon.js index 164c2b74..5e7d3c82 100644 --- a/src/shapes/RegularPolygon.js +++ b/src/shapes/RegularPolygon.js @@ -34,8 +34,7 @@ Kinetic.RegularPolygon.prototype = { context.lineTo(x, y); } context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); } }; Kinetic.Global.extend(Kinetic.RegularPolygon, Kinetic.Shape); diff --git a/src/shapes/Sprite.js b/src/shapes/Sprite.js index abd06508..3b005e40 100644 --- a/src/shapes/Sprite.js +++ b/src/shapes/Sprite.js @@ -38,8 +38,7 @@ Kinetic.Sprite.prototype = { context.beginPath(); context.rect(0, 0, f.width, f.height); context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); if(this.attrs.image) { @@ -58,8 +57,8 @@ Kinetic.Sprite.prototype = { context.beginPath(); context.rect(0, 0, f.width, f.height); context.closePath(); - this.fill(context); - this.stroke(context); + this.fill(context, this.getFill(), null); + this.stroke(context, this.getStroke(), this.getStrokeWidth(), null); }, /** * start sprite animation diff --git a/src/shapes/Star.js b/src/shapes/Star.js index d566206b..19b386da 100644 --- a/src/shapes/Star.js +++ b/src/shapes/Star.js @@ -37,8 +37,7 @@ Kinetic.Star.prototype = { } context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); } }; Kinetic.Global.extend(Kinetic.Star, Kinetic.Shape); diff --git a/src/shapes/Text.js b/src/shapes/Text.js index b27c2612..36adfd7f 100644 --- a/src/shapes/Text.js +++ b/src/shapes/Text.js @@ -68,8 +68,7 @@ Kinetic.Text.prototype = { } context.closePath(); - this.fill(context); - this.stroke(context); + this.render(context); /* * draw text */ @@ -85,14 +84,8 @@ Kinetic.Text.prototype = { context.translate(0, p + this.getTextHeight() / 2); // draw text lines - var appliedShadow = this.appliedShadow; for(var n = 0; n < textArr.length; n++) { var text = textArr[n]; - /* - * need to reset appliedShadow flag so that shadows - * are appropriately applied to each line of text - */ - this.appliedShadow = appliedShadow; // horizontal alignment context.save(); @@ -103,24 +96,50 @@ Kinetic.Text.prototype = { context.translate((this.getWidth() - this._getTextSize(text).width - p * 2) / 2, 0); } - this.fillText(context, text); - this.strokeText(context, text); + var appliedShadow = this.fillText(context, text, this.getTextFill(), this.getShadow()); + this.strokeText(context, text, this.getTextStroke(), this.getTextStrokeWidth(), appliedShadow ? null : this.getShadow()); context.restore(); context.translate(0, lineHeightPx); } context.restore(); }, + drawHitFunc: function(context) { + // draw rect + context.beginPath(); + var boxWidth = this.getWidth(); + var boxHeight = this.getHeight(); + + if(this.attrs.cornerRadius === 0) { + // simple rect - don't bother doing all that complicated maths stuff. + context.rect(0, 0, boxWidth, boxHeight); + } + else { + // arcTo would be nicer, but browser support is patchy (Opera) + context.moveTo(this.attrs.cornerRadius, 0); + context.lineTo(boxWidth - this.attrs.cornerRadius, 0); + context.arc(boxWidth - this.attrs.cornerRadius, this.attrs.cornerRadius, this.attrs.cornerRadius, Math.PI * 3 / 2, 0, false); + context.lineTo(boxWidth, boxHeight - this.attrs.cornerRadius); + context.arc(boxWidth - this.attrs.cornerRadius, boxHeight - this.attrs.cornerRadius, this.attrs.cornerRadius, 0, Math.PI / 2, false); + context.lineTo(this.attrs.cornerRadius, boxHeight); + context.arc(this.attrs.cornerRadius, boxHeight - this.attrs.cornerRadius, this.attrs.cornerRadius, Math.PI / 2, Math.PI, false); + context.lineTo(0, this.attrs.cornerRadius); + context.arc(this.attrs.cornerRadius, this.attrs.cornerRadius, this.attrs.cornerRadius, Math.PI, Math.PI * 3 / 2, false); + } + context.closePath(); + + this.render(context); + }, /** - * set text - * @name setText - * @methodOf Kinetic.Text.prototype - * @param {String} text - */ - setText: function(text) { - var str = Kinetic.Type._isString(text) ? text : text.toString(); - this.setAttr('text', str); - }, + * set text + * @name setText + * @methodOf Kinetic.Text.prototype + * @param {String} text + */ + setText: function(text) { + var str = Kinetic.Type._isString(text) ? text : text.toString(); + this.setAttr('text', str); + }, /** * get width * @name getWidth @@ -172,36 +191,38 @@ Kinetic.Text.prototype = { * @name fillText * @methodOf Kinetic.Text.prototype */ - fillText: function(context, text) { + fillText: function(context, text, textFill, shadow) { if(context.type === 'scene') { - this._fillTextScene(context, text); + return this._fillTextScene(context, text, textFill, shadow); } - else if(context.type === 'buffer') { - this._fillTextBuffer(context, text); + else if(context.type === 'hit') { + return this._fillTextHit(context, text); } + return false; }, - _fillTextScene: function(context, text) { - var appliedShadow = false; - if(this.attrs.textFill) { + _fillTextScene: function(context, text, textFill, shadow) { + if(textFill) { context.save(); - if(this.attrs.shadow && !this.appliedShadow) { - appliedShadow = this._applyShadow(context); + var appliedShadow = this._applyShadow(context, shadow); + context.fillStyle = textFill; + context.fillText(text, 0, 0); + context.restore(); + + if(appliedShadow) { + if(shadow.opacity) { + this._fillTextScene(context, text, textFill); + return true; + } } - context.fillStyle = this.attrs.textFill; - context.fillText(text, 0, 0); - context.restore(); - } - if(appliedShadow) { - this.fillText(context, text, 0, 0); } + return false; }, - _fillTextBuffer: function(context, text) { - if(this.attrs.textFill) { - context.save(); - context.fillStyle = this.colorKey; - context.fillText(text, 0, 0); - context.restore(); - } + _fillTextHit: function(context, text) { + context.save(); + context.fillStyle = this.colorKey; + context.fillText(text, 0, 0); + context.restore(); + return false; }, /** * helper method to stroke text @@ -210,46 +231,46 @@ Kinetic.Text.prototype = { * @methodOf Kinetic.Shape.prototype * @param {String} text */ - strokeText: function(context, text) { + strokeText: function(context, text, textStroke, textStrokeWidth, shadow) { if(context.type === 'scene') { - this._strokeTextScene(context, text); + this._strokeTextScene(context, text, textStroke, textStrokeWidth, shadow); } - else if(context.type === 'buffer') { - this._strokeTextBuffer(context, text); + else if(context.type === 'hit') { + this._strokeTextHit(context, text, textStrokeWidth); } + return false; }, - _strokeTextScene: function(context, text) { - var appliedShadow = false; - - if(this.attrs.textStroke || this.attrs.textStrokeWidth) { + _strokeTextScene: function(context, text, textStroke, textStrokeWidth, shadow) { + if(textStroke || textStrokeWidth) { context.save(); - if(this.attrs.shadow && !this.appliedShadow) { - appliedShadow = this._applyShadow(context); + var appliedShadow = this._applyShadow(context, shadow); + // defaults + textStroke = textStroke || 'black'; + textStrokeWidth = textStrokeWidth || 2; + context.lineWidth = textStrokeWidth; + context.strokeStyle = textStroke; + context.strokeText(text, 0, 0); + context.restore(); + + if(appliedShadow) { + if(shadow.opacity) { + this._strokeTextScene(context, text, textStroke, textStrokeWidth); + return true; + } } - // defaults - var textStroke = this.attrs.textStroke ? this.attrs.textStroke : 'black'; - var textStrokeWidth = this.attrs.textStrokeWidth ? this.attrs.textStrokeWidth : 2; - context.lineWidth = textStrokeWidth; - context.strokeStyle = textStroke; - context.strokeText(text, 0, 0); - context.restore(); - } - - if(appliedShadow) { - this.strokeText(context, text, 0, 0); } + return false; }, - _strokeTextBuffer: function(context, text) { - if(this.attrs.textStroke || this.attrs.textStrokeWidth) { - context.save(); - // defaults - var textStroke = this.colorKey ? this.colorKey : 'black'; - var textStrokeWidth = this.attrs.textStrokeWidth ? this.attrs.textStrokeWidth : 2; - context.lineWidth = textStrokeWidth; - context.strokeStyle = textStroke; - context.strokeText(text, 0, 0); - context.restore(); - } + _strokeTextHit: function(context, text, textStrokeWidth) { + context.save(); + // defaults + var textStroke = this.colorKey ? this.colorKey : 'black'; + var textStrokeWidth = textStrokeWidth || 2; + context.lineWidth = textStrokeWidth; + context.strokeStyle = textStroke; + context.strokeText(text, 0, 0); + context.restore(); + return false; }, /** * set text data. wrap logic and width and height setting occurs diff --git a/src/shapes/TextPath.js b/src/shapes/TextPath.js index 3ded1781..e99b177f 100644 --- a/src/shapes/TextPath.js +++ b/src/shapes/TextPath.js @@ -68,11 +68,9 @@ Kinetic.TextPath.prototype = { var ht = parseFloat(this.attrs.fontSize); context.translate(p0.x, p0.y); - context.rotate(glyphInfo[i].rotation); - this.fillText(context, glyphInfo[i].text); - this.strokeText(context, glyphInfo[i].text); + this.render(context); context.restore(); @@ -310,10 +308,10 @@ Kinetic.Node.addGetters(Kinetic.TextPath, ['text']); // reference Text methods Kinetic.TextPath.prototype.fillText = Kinetic.Text.prototype.fillText; Kinetic.TextPath.prototype._fillTextScene = Kinetic.Text.prototype._fillTextScene; -Kinetic.TextPath.prototype._fillTextBuffer = Kinetic.Text.prototype._fillTextBuffer; +Kinetic.TextPath.prototype._fillTextHit = Kinetic.Text.prototype._fillTextHit; Kinetic.TextPath.prototype.strokeText = Kinetic.Text.prototype.strokeText; Kinetic.TextPath.prototype._strokeTextScene = Kinetic.Text.prototype._strokeTextScene; -Kinetic.TextPath.prototype._strokeTextBuffer = Kinetic.Text.prototype._strokeTextBuffer; +Kinetic.TextPath.prototype._strokeTextHit = Kinetic.Text.prototype._strokeTextHit; /** * set font family * @name setFontFamily diff --git a/tests/dataUrls/cloneGroup.js b/tests/dataUrls/cloneGroup.js new file mode 100644 index 00000000..a367ba22 --- /dev/null +++ b/tests/dataUrls/cloneGroup.js @@ -0,0 +1 @@ +var cloneGroup = ''; \ No newline at end of file diff --git a/tests/dataUrls/customShapeTwoFills.js b/tests/dataUrls/customShapeTwoFills.js new file mode 100644 index 00000000..319ff2c6 --- /dev/null +++ b/tests/dataUrls/customShapeTwoFills.js @@ -0,0 +1 @@ +var customShapeTwoFills = ''; diff --git a/tests/html/unitTests.html b/tests/html/unitTests.html index a6af0ee7..bf54f400 100644 --- a/tests/html/unitTests.html +++ b/tests/html/unitTests.html @@ -10,6 +10,8 @@ + + diff --git a/tests/js/unit/customShapeTests.js b/tests/js/unit/customShapeTests.js deleted file mode 100644 index 504e3c44..00000000 --- a/tests/js/unit/customShapeTests.js +++ /dev/null @@ -1,91 +0,0 @@ -Test.Modules['CUSTOM SHAPE'] = { - 'custom shape with fill, stroke, and strokeWidth': function(containerId) { - var stage = new Kinetic.Stage({ - container: containerId, - width: 578, - height: 200 - }); - var layer = new Kinetic.Layer(); - var shape = new Kinetic.Shape({ - drawFunc: function(context) { - context.beginPath(); - context.moveTo(0, 0); - context.lineTo(100, 0); - context.lineTo(100, 100); - context.closePath(); - this.fill(context); - this.stroke(context); - }, - x: 200, - y: 100, - fill: 'green', - stroke: 'blue', - strokeWidth: 5 - }); - - layer.add(shape); - stage.add(layer); - }, - 'change custom shape draw func': function(containerId) { - var stage = new Kinetic.Stage({ - container: containerId, - width: 578, - height: 200 - }); - var layer = new Kinetic.Layer(); - var shape = new Kinetic.Shape({ - drawFunc: function(context) { - context.beginPath(); - context.moveTo(0, 0); - context.lineTo(100, 0); - context.lineTo(100, 100); - context.closePath(); - this.fill(context); - this.stroke(context); - }, - x: 200, - y: 100, - fill: 'green', - stroke: 'blue', - strokeWidth: 5 - }); - - shape.setDrawFunc(function(context) { - context.beginPath(); - context.moveTo(0, 0); - context.lineTo(200, 0); - context.lineTo(200, 100); - context.closePath(); - this.fill(context); - this.stroke(context); - }); - var rect = new Kinetic.Rect({ - x: 10, - y: 10, - width: 100, - height: 50, - fill: 'green', - stroke: 'black', - strokeWidth: 4, - draggable: true - }); - - rect.setDrawFunc(function(context) { - context.beginPath(); - context.moveTo(0, 0); - context.lineTo(200, 0); - context.lineTo(200, 100); - context.closePath(); - this.fill(context); - this.stroke(context); - }); - - layer.add(shape); - layer.add(rect); - stage.add(layer); - - var dataUrl = layer.toDataURL(); - - test(dataUrls['SHAPE - change custom shape draw func'] === dataUrl, 'problem with setDrawFunc'); - } -}; diff --git a/tests/js/unit/nodeTests.js b/tests/js/unit/nodeTests.js index 457184df..8d4d58e1 100644 --- a/tests/js/unit/nodeTests.js +++ b/tests/js/unit/nodeTests.js @@ -388,6 +388,8 @@ Test.Modules.NODE = { stage.draw(); + warn(layer.toDataURL() === cloneGroup, 'problem cloning group'); + }, 'test on attr change': function(containerId) { diff --git a/tests/js/unit/shapeTests.js b/tests/js/unit/shapeTests.js index 84ae7855..ad2e30b5 100644 --- a/tests/js/unit/shapeTests.js +++ b/tests/js/unit/shapeTests.js @@ -1,5 +1,5 @@ Test.Modules.SHAPE = { - 'SHAPE - test intersects()': function(containerId) { + 'test intersects()': function(containerId) { var stage = new Kinetic.Stage({ container: containerId, width: 578, @@ -44,5 +44,146 @@ Test.Modules.SHAPE = { y: 153 }) === false, '(303, 153) should not intersect the shape'); + }, + 'custom shape with two fills and two strokes': function(containerId) { + var stage = new Kinetic.Stage({ + container: containerId, + width: 578, + height: 200 + }); + var layer = new Kinetic.Layer(); + + var drawTriangle = function(context) { + context.beginPath(); + context.moveTo(200, 50); + context.lineTo(420, 80); + context.quadraticCurveTo(300, 100, 260, 170); + context.closePath(); + this.fill(context, 'red'); + this.stroke(context, 'black', this.getStrokeWidth(), { + color: 'black', + offset: { + x: 20, + y: 20 + }, + opacity: 0.5 + }); + + context.beginPath(); + context.moveTo(300, 150); + context.lineTo(520, 180); + context.quadraticCurveTo(400, 200, 360, 270); + context.closePath(); + + this.fill(context, 'green', { + color: 'black', + offset: { + x: 20, + y: 20 + }, + opacity: 0.5 + }); + + this.stroke(context, 'yellow', this.getStrokeWidth()); + }; + var triangle = new Kinetic.Shape({ + drawFunc: drawTriangle, + fill: "#00D2FF", + stroke: "black", + strokeWidth: 4, + id: 'myTriangle', + draggable: true + }); + + stage.add(layer.add(triangle)); + + warn(layer.toDataURL() === customShapeTwoFills, 'problem with custom shape with two fills'); + + }, + 'custom shape with fill, stroke, and strokeWidth': function(containerId) { + var stage = new Kinetic.Stage({ + container: containerId, + width: 578, + height: 200 + }); + var layer = new Kinetic.Layer(); + var shape = new Kinetic.Shape({ + drawFunc: function(context) { + context.beginPath(); + context.moveTo(0, 0); + context.lineTo(100, 0); + context.lineTo(100, 100); + context.closePath(); + this.fill(context); + this.stroke(context); + }, + x: 200, + y: 100, + fill: 'green', + stroke: 'blue', + strokeWidth: 5 + }); + + layer.add(shape); + stage.add(layer); + }, + 'change custom shape draw func': function(containerId) { + var stage = new Kinetic.Stage({ + container: containerId, + width: 578, + height: 200 + }); + var layer = new Kinetic.Layer(); + var shape = new Kinetic.Shape({ + drawFunc: function(context) { + context.beginPath(); + context.moveTo(0, 0); + context.lineTo(100, 0); + context.lineTo(100, 100); + context.closePath(); + this.render(context); + }, + x: 200, + y: 100, + fill: 'green', + stroke: 'blue', + strokeWidth: 5 + }); + + shape.setDrawFunc(function(context) { + context.beginPath(); + context.moveTo(0, 0); + context.lineTo(200, 0); + context.lineTo(200, 100); + context.closePath(); + this.render(context); + }); + var rect = new Kinetic.Rect({ + x: 10, + y: 10, + width: 100, + height: 50, + fill: 'green', + stroke: 'black', + strokeWidth: 4, + draggable: true + }); + + rect.setDrawFunc(function(context) { + context.beginPath(); + context.moveTo(0, 0); + context.lineTo(200, 0); + context.lineTo(200, 100); + context.closePath(); + this.render(context); + }); + + layer.add(shape); + layer.add(rect); + stage.add(layer); + + var dataUrl = layer.toDataURL(); + + test(dataUrls['SHAPE - change custom shape draw func'] === dataUrl, 'problem with setDrawFunc'); } };