bugs fixes, docs updates. fix #535

This commit is contained in:
Anton Lavrenov
2019-01-11 08:51:46 -05:00
parent a3767bdf3d
commit 830eb53650
15 changed files with 268 additions and 31 deletions

View File

@@ -8,6 +8,7 @@ This project adheres to [Semantic Versioning](http://semver.org/).
### Changed ### Changed
* Fixes inconsistent `layer.setSize()` method. Now it has same arguments as any container. * 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 * 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`
### Removed ### Removed
* `Konva.Util.addMethods` * `Konva.Util.addMethods`

1
konva.d.ts vendored
View File

@@ -1031,6 +1031,7 @@ declare namespace Konva {
lineHeight?: number; lineHeight?: number;
wrap?: string; wrap?: string;
ellipsis?: boolean; ellipsis?: boolean;
textDecoration?: string;
} }
class Text extends Shape { class Text extends Shape {

View File

@@ -8,7 +8,7 @@
* Konva JavaScript Framework v2.6.0 * Konva JavaScript Framework v2.6.0
* http://konvajs.github.io/ * http://konvajs.github.io/
* Licensed under the MIT * Licensed under the MIT
* Date: Sun Jan 06 2019 * Date: Fri Jan 11 2019
* *
* Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS) * Original work Copyright (C) 2011 - 2013 by Eric Rowell (KineticJS)
* Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva) * Modified work Copyright (C) 2014 - present by Anton Lavrenov (Konva)
@@ -1786,6 +1786,9 @@
if (fillPatternRotation) { if (fillPatternRotation) {
this.rotate(fillPatternRotation); this.rotate(fillPatternRotation);
} }
// TODO: optimize to fillPatternScaleX and fillPatternScaleY
// otherwise it is object (always true)
// do the same for offset
if (fillPatternScale) { if (fillPatternScale) {
this.scale(fillPatternScale.x, fillPatternScale.y); this.scale(fillPatternScale.x, fillPatternScale.y);
} }
@@ -1793,7 +1796,7 @@
this.translate(-1 * fillPatternOffset.x, -1 * fillPatternOffset.y); this.translate(-1 * fillPatternOffset.x, -1 * fillPatternOffset.y);
} }
this.setAttr('fillStyle', this.createPattern(shape.getFillPatternImage(), shape.getFillPatternRepeat() || 'repeat')); this.setAttr('fillStyle', this.createPattern(shape.getFillPatternImage(), shape.getFillPatternRepeat() || 'repeat'));
this.fill(); shape._fillFunc(this);
}; };
SceneContext.prototype._fillLinearGradient = function (shape) { SceneContext.prototype._fillLinearGradient = function (shape) {
var start = shape.getFillLinearGradientStartPoint(), end = shape.getFillLinearGradientEndPoint(), colorStops = shape.getFillLinearGradientColorStops(), grd = this.createLinearGradient(start.x, start.y, end.x, end.y); var start = shape.getFillLinearGradientStartPoint(), end = shape.getFillLinearGradientEndPoint(), colorStops = shape.getFillLinearGradientColorStops(), grd = this.createLinearGradient(start.x, start.y, end.x, end.y);
@@ -1813,7 +1816,7 @@
grd.addColorStop(colorStops[n], colorStops[n + 1]); grd.addColorStop(colorStops[n], colorStops[n + 1]);
} }
this.setAttr('fillStyle', grd); this.setAttr('fillStyle', grd);
this.fill(); shape._fillFunc(this);
}; };
SceneContext.prototype._fill = function (shape) { SceneContext.prototype._fill = function (shape) {
var hasColor = shape.fill(), fillPriority = shape.getFillPriority(); var hasColor = shape.fill(), fillPriority = shape.getFillPriority();
@@ -3908,7 +3911,8 @@
}()); }());
Node.prototype.nodeType = 'Node'; Node.prototype.nodeType = 'Node';
/** /**
* get/set zIndex relative to the node's siblings who share the same parent * 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.
* @name Konva.Node#zIndex * @name Konva.Node#zIndex
* @method * @method
* @param {Number} index * @param {Number} index
@@ -5032,6 +5036,8 @@
*/ */
Collection.mapMethods(Container); Collection.mapMethods(Container);
// TODO: add a warning if stage has too many layers
// TODO: remove "content" events from docs
// CONSTANTS // CONSTANTS
var STAGE$1 = 'Stage', STRING = 'string', PX = 'px', MOUSEOUT = 'mouseout', MOUSELEAVE$1 = 'mouseleave', MOUSEOVER = 'mouseover', MOUSEENTER$1 = 'mouseenter', MOUSEMOVE = 'mousemove', MOUSEDOWN = 'mousedown', MOUSEUP = 'mouseup', CONTEXTMENU = 'contextmenu', CLICK = 'click', DBL_CLICK = 'dblclick', TOUCHSTART = 'touchstart', TOUCHEND = 'touchend', TAP = 'tap', DBL_TAP = 'dbltap', TOUCHMOVE = 'touchmove', WHEEL = 'wheel', CONTENT_MOUSEOUT = 'contentMouseout', CONTENT_MOUSEOVER = 'contentMouseover', CONTENT_MOUSEMOVE = 'contentMousemove', CONTENT_MOUSEDOWN = 'contentMousedown', CONTENT_MOUSEUP = 'contentMouseup', CONTENT_CONTEXTMENU = 'contentContextmenu', CONTENT_CLICK = 'contentClick', CONTENT_DBL_CLICK = 'contentDblclick', CONTENT_TOUCHSTART = 'contentTouchstart', CONTENT_TOUCHEND = 'contentTouchend', CONTENT_DBL_TAP = 'contentDbltap', CONTENT_TAP = 'contentTap', CONTENT_TOUCHMOVE = 'contentTouchmove', CONTENT_WHEEL = 'contentWheel', DIV = 'div', RELATIVE = 'relative', KONVA_CONTENT = 'konvajs-content', SPACE$1 = ' ', UNDERSCORE = '_', CONTAINER = 'container', EMPTY_STRING$1 = '', EVENTS = [ var STAGE$1 = 'Stage', STRING = 'string', PX = 'px', MOUSEOUT = 'mouseout', MOUSELEAVE$1 = 'mouseleave', MOUSEOVER = 'mouseover', MOUSEENTER$1 = 'mouseenter', MOUSEMOVE = 'mousemove', MOUSEDOWN = 'mousedown', MOUSEUP = 'mouseup', CONTEXTMENU = 'contextmenu', CLICK = 'click', DBL_CLICK = 'dblclick', TOUCHSTART = 'touchstart', TOUCHEND = 'touchend', TAP = 'tap', DBL_TAP = 'dbltap', TOUCHMOVE = 'touchmove', WHEEL = 'wheel', CONTENT_MOUSEOUT = 'contentMouseout', CONTENT_MOUSEOVER = 'contentMouseover', CONTENT_MOUSEMOVE = 'contentMousemove', CONTENT_MOUSEDOWN = 'contentMousedown', CONTENT_MOUSEUP = 'contentMouseup', CONTENT_CONTEXTMENU = 'contentContextmenu', CONTENT_CLICK = 'contentClick', CONTENT_DBL_CLICK = 'contentDblclick', CONTENT_TOUCHSTART = 'contentTouchstart', CONTENT_TOUCHEND = 'contentTouchend', CONTENT_DBL_TAP = 'contentDbltap', CONTENT_TAP = 'contentTap', CONTENT_TOUCHMOVE = 'contentTouchmove', CONTENT_WHEEL = 'contentWheel', DIV = 'div', RELATIVE = 'relative', KONVA_CONTENT = 'konvajs-content', SPACE$1 = ' ', UNDERSCORE = '_', CONTAINER = 'container', EMPTY_STRING$1 = '', EVENTS = [
MOUSEDOWN, MOUSEDOWN,
@@ -6821,6 +6827,7 @@
var HAS_SHADOW = 'hasShadow'; var HAS_SHADOW = 'hasShadow';
var SHADOW_RGBA = 'shadowRGBA'; var SHADOW_RGBA = 'shadowRGBA';
// TODO: cache gradient from context
function _fillFunc(context) { function _fillFunc(context) {
context.fill(); context.fill();
} }
@@ -12282,6 +12289,10 @@
Factory.addGetterSetter(Star, 'outerRadius', 0, Validators.getNumberValidator()); Factory.addGetterSetter(Star, 'outerRadius', 0, Validators.getNumberValidator());
Collection.mapMethods(Star); Collection.mapMethods(Star);
// TODO:
// deprecate fill pattern image and fill gradient for text (and textpath?)
// we have API for that in docs
// I guess we should show a error or warning
// constants // constants
var AUTO = 'auto', var AUTO = 'auto',
//CANVAS = 'canvas', //CANVAS = 'canvas',
@@ -12333,7 +12344,7 @@
* @param {String} [config.verticalAlign] can be top, middle or bottom * @param {String} [config.verticalAlign] can be top, middle or bottom
* @param {Number} [config.padding] * @param {Number} [config.padding]
* @param {Number} [config.lineHeight] default is 1 * @param {Number} [config.lineHeight] default is 1
* @param {String} [config.wrap] can be word, char, or none. Default is word * @param {String} [config.wrap] can be "word", "char", or "none". Default is word
* @param {Boolean} [config.ellipsis] can be true or false. Default is false. if Konva.Text config is set to wrap="none" and ellipsis=true, then it will add "..." to the end * @param {Boolean} [config.ellipsis] can be true or false. Default is false. if Konva.Text config is set to wrap="none" and ellipsis=true, then it will add "..." to the end
* @param {String} [config.fill] fill color * @param {String} [config.fill] fill color
* @param {Image} [config.fillPatternImage] fill pattern image * @param {Image} [config.fillPatternImage] fill pattern image
@@ -12423,7 +12434,8 @@
config = config || {}; config = config || {};
// set default color to black // set default color to black
if (!config.fillLinearGradientColorStops && if (!config.fillLinearGradientColorStops &&
!config.fillRadialGradientColorStops) { !config.fillRadialGradientColorStops &&
!config.fillPatternImage) {
config.fill = config.fill || 'black'; config.fill = config.fill || 'black';
} }
_this = _super.call(this, config) || this; _this = _super.call(this, config) || this;
@@ -12440,6 +12452,7 @@
var padding = this.padding(), textHeight = this.getTextHeight(), lineHeightPx = this.lineHeight() * textHeight, textArr = this.textArr, textArrLen = textArr.length, verticalAlign = this.verticalAlign(), alignY = 0, align = this.align(), totalWidth = this.getWidth(), letterSpacing = this.letterSpacing(), textDecoration = this.textDecoration(), fill = this.fill(), fontSize = this.fontSize(), n; var padding = this.padding(), textHeight = this.getTextHeight(), lineHeightPx = this.lineHeight() * textHeight, textArr = this.textArr, textArrLen = textArr.length, verticalAlign = this.verticalAlign(), alignY = 0, align = this.align(), totalWidth = this.getWidth(), letterSpacing = this.letterSpacing(), textDecoration = this.textDecoration(), fill = this.fill(), fontSize = this.fontSize(), n;
context.setAttr('font', this._getContextFont()); context.setAttr('font', this._getContextFont());
context.setAttr('textBaseline', MIDDLE); context.setAttr('textBaseline', MIDDLE);
// TODO: do we have that property in context?
context.setAttr('textAlign', LEFT$1); context.setAttr('textAlign', LEFT$1);
// handle vertical alignment // handle vertical alignment
if (verticalAlign === MIDDLE) { if (verticalAlign === MIDDLE) {
@@ -12884,7 +12897,9 @@
*/ */
Factory.addGetterSetter(Text, 'lineHeight', 1, Validators.getNumberValidator()); Factory.addGetterSetter(Text, 'lineHeight', 1, Validators.getNumberValidator());
/** /**
* get/set wrap. Can be word, char, or none. Default is word. * get/set wrap. Can be "word", "char", or "none". Default is "word".
* In "word" wrapping any word still can be wrapped if it can't be placed in the required width
* without breaks.
* @name Konva.Text#wrap * @name Konva.Text#wrap
* @method * @method
* @param {String} wrap * @param {String} wrap
@@ -13171,10 +13186,10 @@
return this.textHeight; return this.textHeight;
}; };
TextPath.prototype.setText = function (text) { TextPath.prototype.setText = function (text) {
Text.prototype.setText.call(this, text); return Text.prototype.setText.call(this, text);
}; };
TextPath.prototype._getContextFont = function () { TextPath.prototype._getContextFont = function () {
Text.prototype._getContextFont.call(this); return Text.prototype._getContextFont.call(this);
}; };
TextPath.prototype._getTextSize = function (text) { TextPath.prototype._getTextSize = function (text) {
var dummyCanvas = this.dummyCanvas; var dummyCanvas = this.dummyCanvas;

4
konva.min.js vendored

File diff suppressed because one or more lines are too long

View File

@@ -483,9 +483,14 @@ export class SceneContext extends Context {
if (fillPatternX || fillPatternY) { if (fillPatternX || fillPatternY) {
this.translate(fillPatternX || 0, fillPatternY || 0); this.translate(fillPatternX || 0, fillPatternY || 0);
} }
if (fillPatternRotation) { if (fillPatternRotation) {
this.rotate(fillPatternRotation); this.rotate(fillPatternRotation);
} }
// TODO: optimize to fillPatternScaleX and fillPatternScaleY
// otherwise it is object (always true)
// do the same for offset
if (fillPatternScale) { if (fillPatternScale) {
this.scale(fillPatternScale.x, fillPatternScale.y); this.scale(fillPatternScale.x, fillPatternScale.y);
} }
@@ -500,7 +505,7 @@ export class SceneContext extends Context {
shape.getFillPatternRepeat() || 'repeat' shape.getFillPatternRepeat() || 'repeat'
) )
); );
this.fill(); shape._fillFunc(this);
} }
_fillLinearGradient(shape) { _fillLinearGradient(shape) {
var start = shape.getFillLinearGradientStartPoint(), var start = shape.getFillLinearGradientStartPoint(),
@@ -537,7 +542,7 @@ export class SceneContext extends Context {
grd.addColorStop(colorStops[n], colorStops[n + 1]); grd.addColorStop(colorStops[n], colorStops[n + 1]);
} }
this.setAttr('fillStyle', grd); this.setAttr('fillStyle', grd);
this.fill(); shape._fillFunc(this);
} }
_fill(shape) { _fill(shape) {
var hasColor = shape.fill(), var hasColor = shape.fill(),

View File

@@ -2153,7 +2153,8 @@ export abstract class Node {
Node.prototype.nodeType = 'Node'; Node.prototype.nodeType = 'Node';
/** /**
* get/set zIndex relative to the node's siblings who share the same parent * 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.
* @name Konva.Node#zIndex * @name Konva.Node#zIndex
* @method * @method
* @param {Number} index * @param {Number} index

View File

@@ -9,6 +9,8 @@ import { Context } from './Context';
var HAS_SHADOW = 'hasShadow'; var HAS_SHADOW = 'hasShadow';
var SHADOW_RGBA = 'shadowRGBA'; var SHADOW_RGBA = 'shadowRGBA';
// TODO: cache gradient from context
function _fillFunc(context) { function _fillFunc(context) {
context.fill(); context.fill();
} }

View File

@@ -1,13 +1,15 @@
import { Util, Collection } from './Util'; import { Util, Collection } from './Util';
import { Factory, Validators } from './Factory'; import { Factory } from './Factory';
import { Container } from './Container'; import { Container } from './Container';
import { document, isBrowser, getGlobalKonva, UA } from './Global'; import { document, isBrowser, getGlobalKonva, UA } from './Global';
import { Node } from './Node';
import { SceneCanvas, HitCanvas } from './Canvas'; import { SceneCanvas, HitCanvas } from './Canvas';
import { GetSet, Vector2d } from './types'; import { GetSet, Vector2d } from './types';
import { Shape } from './Shape'; import { Shape } from './Shape';
import { BaseLayer } from './BaseLayer'; import { BaseLayer } from './BaseLayer';
// TODO: add a warning if stage has too many layers
// TODO: remove "content" events from docs
// CONSTANTS // CONSTANTS
var STAGE = 'Stage', var STAGE = 'Stage',
STRING = 'string', STRING = 'string',

View File

@@ -5,6 +5,11 @@ import { UA } from '../Global';
import { GetSet } from '../types'; import { GetSet } from '../types';
// TODO:
// deprecate fill pattern image and fill gradient for text (and textpath?)
// we have API for that in docs
// I guess we should show a error or warning
// constants // constants
var AUTO = 'auto', var AUTO = 'auto',
//CANVAS = 'canvas', //CANVAS = 'canvas',
@@ -77,7 +82,7 @@ function _strokeFunc(context) {
* @param {String} [config.verticalAlign] can be top, middle or bottom * @param {String} [config.verticalAlign] can be top, middle or bottom
* @param {Number} [config.padding] * @param {Number} [config.padding]
* @param {Number} [config.lineHeight] default is 1 * @param {Number} [config.lineHeight] default is 1
* @param {String} [config.wrap] can be word, char, or none. Default is word * @param {String} [config.wrap] can be "word", "char", or "none". Default is word
* @param {Boolean} [config.ellipsis] can be true or false. Default is false. if Konva.Text config is set to wrap="none" and ellipsis=true, then it will add "..." to the end * @param {Boolean} [config.ellipsis] can be true or false. Default is false. if Konva.Text config is set to wrap="none" and ellipsis=true, then it will add "..." to the end
* @@shapeParams * @@shapeParams
* @@nodeParams * @@nodeParams
@@ -103,7 +108,8 @@ export class Text extends Shape {
// set default color to black // set default color to black
if ( if (
!config.fillLinearGradientColorStops && !config.fillLinearGradientColorStops &&
!config.fillRadialGradientColorStops !config.fillRadialGradientColorStops &&
!config.fillPatternImage
) { ) {
config.fill = config.fill || 'black'; config.fill = config.fill || 'black';
} }
@@ -139,6 +145,8 @@ export class Text extends Shape {
context.setAttr('font', this._getContextFont()); context.setAttr('font', this._getContextFont());
context.setAttr('textBaseline', MIDDLE); context.setAttr('textBaseline', MIDDLE);
// TODO: do we have that property in context?
context.setAttr('textAlign', LEFT); context.setAttr('textAlign', LEFT);
// handle vertical alignment // handle vertical alignment
@@ -672,7 +680,9 @@ Factory.addGetterSetter(Text, 'verticalAlign', TOP);
Factory.addGetterSetter(Text, 'lineHeight', 1, Validators.getNumberValidator()); Factory.addGetterSetter(Text, 'lineHeight', 1, Validators.getNumberValidator());
/** /**
* get/set wrap. Can be word, char, or none. Default is word. * get/set wrap. Can be "word", "char", or "none". Default is "word".
* In "word" wrapping any word still can be wrapped if it can't be placed in the required width
* without breaks.
* @name Konva.Text#wrap * @name Konva.Text#wrap
* @method * @method
* @param {String} wrap * @param {String} wrap

View File

@@ -184,11 +184,11 @@ export class TextPath extends Shape {
return this.textHeight; return this.textHeight;
} }
setText(text) { setText(text) {
Text.prototype.setText.call(this, text); return Text.prototype.setText.call(this, text);
} }
_getContextFont() { _getContextFont() {
Text.prototype._getContextFont.call(this); return Text.prototype._getContextFont.call(this);
} }
_getTextSize(text) { _getTextSize(text) {

View File

@@ -33,6 +33,8 @@
<div id="mocha"></div> <div id="mocha"></div>
<div id="konva-container"></div> <div id="konva-container"></div>
<!-- TODO: use parcel for tests -->
<script src="../node_modules/mocha/mocha.js"></script> <script src="../node_modules/mocha/mocha.js"></script>
<script src="../node_modules/chai/chai.js"></script> <script src="../node_modules/chai/chai.js"></script>
<script src="lib/stats.js"></script> <script src="lib/stats.js"></script>

View File

@@ -223,6 +223,7 @@ beforeEach(function() {
Konva.UA.mobile = false; Konva.UA.mobile = false;
afterEach(function() { afterEach(function() {
// can we destroy stage?
clearTimeout(Konva.stages[Konva.stages.length - 1].dblTimeout); clearTimeout(Konva.stages[Konva.stages.length - 1].dblTimeout);
// Konva.stages.forEach(function(stage) { // Konva.stages.forEach(function(stage) {
// stage.destroy(); // stage.destroy();

View File

@@ -148,7 +148,30 @@ suite('Circle', function() {
layer.add(group); layer.add(group);
stage.add(layer); stage.add(layer);
assert.equal(circle.getName(), 'myCircle'); var canvas = createCanvas();
var ctx = canvas.getContext('2d');
var start = { x: -35, y: -35 };
var end = { x: 35, y: 35 };
var colorStops = [0, 'red', 1, 'blue'];
var grd = ctx.createLinearGradient(start.x, start.y, end.x, end.y);
// build color stops
for (var n = 0; n < colorStops.length; n += 2) {
grd.addColorStop(colorStops[n], colorStops[n + 1]);
}
ctx.beginPath();
ctx.translate(circle.x(), circle.y());
ctx.arc(0, 0, 70, 0, Math.PI * 2, false);
ctx.closePath();
ctx.fillStyle = grd;
ctx.lineWidth = 4;
ctx.fill();
ctx.stroke();
compareLayerAndCanvas(layer, canvas, 200);
}); });
// ====================================================== // ======================================================

View File

@@ -737,28 +737,47 @@ suite('Text', function() {
}); });
}); });
test('gradient', function() { test('linear gradient', function() {
Konva.pixelRatio = 1; Konva.pixelRatio = 1;
var stage = addStage(); var stage = addStage();
var layer = new Konva.Layer(); var layer = new Konva.Layer();
var text = new Konva.Text({ var text = new Konva.Text({
fontSize: 100, fontSize: 50,
y: 10, y: 10,
x: 10, x: 10,
fillLinearGradientStartPoint: { x: -50, y: -50 }, fillLinearGradientStartPoint: { x: 0, y: 0 },
fillLinearGradientEndPoint: { x: 50, y: 50 }, fillLinearGradientEndPoint: { x: 300, y: 0 },
fillLinearGradientColorStops: [0, 'yellow', 1, 'yellow'], fillLinearGradientColorStops: [0, 'yellow', 0.5, 'yellow', 1, 'red'],
text: 'Text with gradient!!', text: 'Text with gradient!!',
draggable: true draggable: true
}); });
layer.add(text); layer.add(text);
stage.add(layer); stage.add(layer);
//stage.on('mousemove', function() { var canvas = createCanvas();
// console.log(stage.getPointerPosition()); var ctx = canvas.getContext('2d');
//});
var data = layer.getContext().getImageData(41, 50, 1, 1).data; ctx.fillStyle = 'green';
ctx.font = 'normal 50px Arial';
ctx.textBaseline = 'middle';
var start = { x: 0, y: 0 };
var end = { x: 300, y: 0 };
var colorStops = [0, 'yellow', 0.5, 'yellow', 1, 'red'];
var grd = ctx.createLinearGradient(start.x, start.y, end.x, end.y);
// build color stops
for (var n = 0; n < colorStops.length; n += 2) {
grd.addColorStop(colorStops[n], colorStops[n + 1]);
}
ctx.fillStyle = grd;
ctx.fillText(text.text(), text.x(), text.y() + text.fontSize() / 2);
compareLayerAndCanvas(layer, canvas, 200);
var data = layer.getContext().getImageData(25, 41, 1, 1).data;
delete Konva.pixelRatio; delete Konva.pixelRatio;
assert.equal(data[0], 255, 'full green'); assert.equal(data[0], 255, 'full green');
assert.equal(data[1], 255, 'full red'); assert.equal(data[1], 255, 'full red');
@@ -766,6 +785,102 @@ suite('Text', function() {
assert.equal(data[3], 255, '255 alpha - fully visible'); assert.equal(data[3], 255, '255 alpha - fully visible');
}); });
// TODO: how to make correct behavior?
test.skip('linear gradient multiline', function() {
Konva.pixelRatio = 1;
var stage = addStage();
var layer = new Konva.Layer();
var text = new Konva.Text({
fontSize: 50,
fillLinearGradientStartPoint: { x: 0, y: 0 },
fillLinearGradientEndPoint: { x: 0, y: 100 },
fillLinearGradientColorStops: [0, 'yellow', 1, 'red'],
text: 'Text with gradient!!\nText with gradient!!',
draggable: true
});
layer.add(text);
stage.add(layer);
var canvas = createCanvas();
var ctx = canvas.getContext('2d');
ctx.fillStyle = 'green';
ctx.font = 'normal 50px Arial';
ctx.textBaseline = 'middle';
var start = { x: 0, y: 0 };
var end = { x: 0, y: 100 };
var colorStops = [0, 'yellow', 1, 'red'];
var grd = ctx.createLinearGradient(start.x, start.y, end.x, end.y);
// build color stops
for (var n = 0; n < colorStops.length; n += 2) {
grd.addColorStop(colorStops[n], colorStops[n + 1]);
}
ctx.fillStyle = grd;
ctx.fillText(text.text(), text.x(), text.y() + text.fontSize() / 2);
ctx.fillText(
text.text(),
text.x(),
text.y() + text.fontSize() / 2 + text.fontSize()
);
compareLayerAndCanvas(layer, canvas, 200);
var data = layer.getContext().getImageData(25, 41, 1, 1).data;
delete Konva.pixelRatio;
assert.equal(data[0], 255, 'full green');
assert.equal(data[1], 255, 'full red');
assert.equal(data[2], 0, 'no blue');
assert.equal(data[3], 255, '255 alpha - fully visible');
});
test('radial gradient', function() {
var stage = addStage();
var layer = new Konva.Layer();
var text = new Konva.Text({
fontSize: 50,
y: 0,
x: 0,
fillRadialGradientStartPoint: { x: 100, y: 0 },
fillRadialGradientStartRadius: 0,
fillRadialGradientEndRadius: 100,
fillRadialGradientEndPoint: { x: 100, y: 0 },
fillRadialGradientColorStops: [0, 'yellow', 1, 'red'],
text: 'Text with gradient!!',
draggable: true
});
layer.add(text);
stage.add(layer);
var canvas = createCanvas();
var ctx = canvas.getContext('2d');
ctx.fillStyle = 'green';
ctx.font = 'normal 50px Arial';
ctx.textBaseline = 'middle';
var start = { x: 100, y: 0 };
var end = { x: 100, y: 0 };
var colorStops = [0, 'yellow', 1, 'red'];
var grd = ctx.createRadialGradient(start.x, start.y, 0, end.x, end.y, 100);
// build color stops
for (var n = 0; n < colorStops.length; n += 2) {
grd.addColorStop(colorStops[n], colorStops[n + 1]);
}
ctx.fillStyle = grd;
ctx.translate(0, 25);
ctx.fillText(text.text(), 0, 0);
compareLayerAndCanvas(layer, canvas, 200);
});
test('text should be centered in line height', function() { test('text should be centered in line height', function() {
var stage = addStage(); var stage = addStage();
var layer = new Konva.Layer(); var layer = new Konva.Layer();
@@ -837,4 +952,39 @@ suite('Text', function() {
assert.equal(lines[1].text, 'good text'); assert.equal(lines[1].text, 'good text');
layer.draw(); layer.draw();
}); });
test('image gradient for text', function(done) {
Konva.pixelRatio = 1;
var imageObj = new Image();
imageObj.onload = function() {
var stage = addStage();
var layer = new Konva.Layer();
var text = new Konva.Text({
text: 'Hello, this is some good text',
fontSize: 30,
fillPatternImage: imageObj
});
layer.add(text);
stage.add(layer);
var canvas = createCanvas();
var ctx = canvas.getContext('2d');
ctx.fillStyle = 'green';
ctx.font = 'normal normal 30px Arial';
ctx.textBaseline = 'middle';
var grd = ctx.createPattern(imageObj, 'repeat');
ctx.translate(0, 15);
ctx.fillStyle = grd;
ctx.fillText(text.text(), 0, 0);
compareLayerAndCanvas(layer, canvas, 200);
delete Konva.pixelRatio;
done();
};
imageObj.src = 'assets/darth-vader.jpg';
});
}); });

View File

@@ -516,6 +516,30 @@ suite('TextPath', function() {
assert.equal(called, true); assert.equal(called, true);
}); });
test.skip('linear gradient for path', function() {
var stage = addStage();
var layer = new Konva.Layer();
stage.add(layer);
const text = new Konva.TextPath({
x: 0,
y: 30,
text: 'AV',
fontSize: 20,
data: 'M0,0 L200,0',
fillLinearGradientStartPoint: { x: 0, y: 0 },
fillLinearGradientEndPoint: { x: 200, y: 0 },
fillLinearGradientColorStops: [0, 'yellow', 1, 'red'],
text: 'Text with gradient!!'
});
layer.add(text);
layer.draw();
var trace = layer.getContext().getTrace();
console.log(trace);
});
test('visual check for text path', function() { test('visual check for text path', function() {
var stage = addStage(); var stage = addStage();