2021-05-05 22:54:03 +08:00
|
|
|
import { Util } from '../Util';
|
2023-03-15 23:30:11 +08:00
|
|
|
import { Context } from '../Context';
|
2021-05-05 22:54:03 +08:00
|
|
|
import { Factory } from '../Factory';
|
|
|
|
import { Shape, ShapeConfig } from '../Shape';
|
|
|
|
import { Konva } from '../Global';
|
2019-02-25 01:06:04 +08:00
|
|
|
import {
|
|
|
|
getNumberValidator,
|
|
|
|
getStringValidator,
|
2020-05-15 00:13:47 +08:00
|
|
|
getNumberOrAutoValidator,
|
2020-09-08 00:03:20 +08:00
|
|
|
getBooleanValidator,
|
2021-05-05 22:54:03 +08:00
|
|
|
} from '../Validators';
|
|
|
|
import { _registerNode } from '../Global';
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
import { GetSet } from '../types';
|
|
|
|
|
2020-09-14 22:46:26 +08:00
|
|
|
export function stringToArray(string: string) {
|
|
|
|
// we need to use `Array.from` because it can split unicode string correctly
|
|
|
|
// we also can use some regexp magic from lodash:
|
|
|
|
// https://github.com/lodash/lodash/blob/fb1f99d9d90ad177560d771bc5953a435b2dc119/lodash.toarray/index.js#L256
|
|
|
|
// but I decided it is too much code for that small fix
|
|
|
|
return Array.from(string);
|
|
|
|
}
|
|
|
|
|
2019-03-10 23:31:13 +08:00
|
|
|
export interface TextConfig extends ShapeConfig {
|
|
|
|
text?: string;
|
|
|
|
fontFamily?: string;
|
|
|
|
fontSize?: number;
|
|
|
|
fontStyle?: string;
|
2020-11-13 12:38:23 +08:00
|
|
|
fontVariant?: string;
|
|
|
|
textDecoration?: string;
|
2019-03-10 23:31:13 +08:00
|
|
|
align?: string;
|
|
|
|
verticalAlign?: string;
|
|
|
|
padding?: number;
|
|
|
|
lineHeight?: number;
|
2021-01-27 00:43:52 +08:00
|
|
|
letterSpacing?: number;
|
2019-03-10 23:31:13 +08:00
|
|
|
wrap?: string;
|
|
|
|
ellipsis?: boolean;
|
|
|
|
}
|
|
|
|
|
2019-01-02 04:59:27 +08:00
|
|
|
// constants
|
|
|
|
var AUTO = 'auto',
|
|
|
|
//CANVAS = 'canvas',
|
|
|
|
CENTER = 'center',
|
|
|
|
JUSTIFY = 'justify',
|
|
|
|
CHANGE_KONVA = 'Change.konva',
|
|
|
|
CONTEXT_2D = '2d',
|
|
|
|
DASH = '-',
|
|
|
|
LEFT = 'left',
|
|
|
|
TEXT = 'text',
|
|
|
|
TEXT_UPPER = 'Text',
|
|
|
|
TOP = 'top',
|
|
|
|
BOTTOM = 'bottom',
|
|
|
|
MIDDLE = 'middle',
|
|
|
|
NORMAL = 'normal',
|
|
|
|
PX_SPACE = 'px ',
|
|
|
|
SPACE = ' ',
|
|
|
|
RIGHT = 'right',
|
|
|
|
WORD = 'word',
|
|
|
|
CHAR = 'char',
|
|
|
|
NONE = 'none',
|
|
|
|
ELLIPSIS = '…',
|
|
|
|
ATTR_CHANGE_LIST = [
|
|
|
|
'fontFamily',
|
|
|
|
'fontSize',
|
|
|
|
'fontStyle',
|
|
|
|
'fontVariant',
|
|
|
|
'padding',
|
|
|
|
'align',
|
|
|
|
'verticalAlign',
|
|
|
|
'lineHeight',
|
|
|
|
'text',
|
|
|
|
'width',
|
|
|
|
'height',
|
|
|
|
'wrap',
|
|
|
|
'ellipsis',
|
2020-05-15 00:13:47 +08:00
|
|
|
'letterSpacing',
|
2019-01-02 04:59:27 +08:00
|
|
|
],
|
|
|
|
// cached variables
|
|
|
|
attrChangeListLen = ATTR_CHANGE_LIST.length;
|
2020-06-26 01:18:31 +08:00
|
|
|
|
|
|
|
function normalizeFontFamily(fontFamily: string) {
|
|
|
|
return fontFamily
|
|
|
|
.split(',')
|
|
|
|
.map((family) => {
|
|
|
|
family = family.trim();
|
|
|
|
const hasSpace = family.indexOf(' ') >= 0;
|
|
|
|
const hasQuotes = family.indexOf('"') >= 0 || family.indexOf("'") >= 0;
|
|
|
|
if (hasSpace && !hasQuotes) {
|
|
|
|
family = `"${family}"`;
|
|
|
|
}
|
|
|
|
return family;
|
|
|
|
})
|
|
|
|
.join(', ');
|
|
|
|
}
|
|
|
|
|
2023-03-15 23:30:11 +08:00
|
|
|
var dummyContext: CanvasRenderingContext2D;
|
2019-01-02 04:59:27 +08:00
|
|
|
function getDummyContext() {
|
|
|
|
if (dummyContext) {
|
|
|
|
return dummyContext;
|
|
|
|
}
|
2023-03-15 23:30:11 +08:00
|
|
|
dummyContext = Util.createCanvasElement().getContext(CONTEXT_2D) as CanvasRenderingContext2D;
|
2019-01-02 04:59:27 +08:00
|
|
|
return dummyContext;
|
|
|
|
}
|
|
|
|
|
2023-03-15 23:30:11 +08:00
|
|
|
function _fillFunc(context: Context) {
|
2019-02-24 09:54:20 +08:00
|
|
|
context.fillText(this._partialText, this._partialTextX, this._partialTextY);
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
2023-03-15 23:30:11 +08:00
|
|
|
function _strokeFunc(context: Context) {
|
2019-02-24 09:54:20 +08:00
|
|
|
context.strokeText(this._partialText, this._partialTextX, this._partialTextY);
|
2019-02-21 00:01:38 +08:00
|
|
|
}
|
|
|
|
|
2023-03-15 23:30:11 +08:00
|
|
|
function checkDefaultFill(config: TextConfig) {
|
2019-02-21 00:01:38 +08:00
|
|
|
config = config || {};
|
|
|
|
|
|
|
|
// set default color to black
|
|
|
|
if (
|
|
|
|
!config.fillLinearGradientColorStops &&
|
|
|
|
!config.fillRadialGradientColorStops &&
|
|
|
|
!config.fillPatternImage
|
|
|
|
) {
|
|
|
|
config.fill = config.fill || 'black';
|
|
|
|
}
|
|
|
|
return config;
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Text constructor
|
|
|
|
* @constructor
|
|
|
|
* @memberof Konva
|
|
|
|
* @augments Konva.Shape
|
|
|
|
* @param {Object} config
|
|
|
|
* @param {String} [config.fontFamily] default is Arial
|
|
|
|
* @param {Number} [config.fontSize] in pixels. Default is 12
|
2021-03-15 20:34:20 +08:00
|
|
|
* @param {String} [config.fontStyle] can be 'normal', 'bold', 'italic' or even 'italic bold'. Default is 'normal'
|
2019-01-02 04:59:27 +08:00
|
|
|
* @param {String} [config.fontVariant] can be normal or small-caps. Default is normal
|
|
|
|
* @param {String} [config.textDecoration] can be line-through, underline or empty string. Default is empty string.
|
|
|
|
* @param {String} config.text
|
|
|
|
* @param {String} [config.align] can be left, center, or right
|
|
|
|
* @param {String} [config.verticalAlign] can be top, middle or bottom
|
|
|
|
* @param {Number} [config.padding]
|
|
|
|
* @param {Number} [config.lineHeight] default is 1
|
2019-01-11 21:51:46 +08:00
|
|
|
* @param {String} [config.wrap] can be "word", "char", or "none". Default is word
|
2019-01-02 04:59:27 +08:00
|
|
|
* @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
|
|
|
|
* @@nodeParams
|
|
|
|
* @example
|
|
|
|
* var text = new Konva.Text({
|
|
|
|
* x: 10,
|
|
|
|
* y: 15,
|
|
|
|
* text: 'Simple Text',
|
|
|
|
* fontSize: 30,
|
|
|
|
* fontFamily: 'Calibri',
|
|
|
|
* fill: 'green'
|
|
|
|
* });
|
|
|
|
*/
|
2019-03-10 23:31:13 +08:00
|
|
|
export class Text extends Shape<TextConfig> {
|
2022-02-24 09:21:15 +08:00
|
|
|
textArr: Array<{ text: string; width: number; lastInParagraph: boolean }>;
|
2019-02-24 09:54:20 +08:00
|
|
|
_partialText: string;
|
|
|
|
_partialTextX = 0;
|
|
|
|
_partialTextY = 0;
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
textWidth: number;
|
|
|
|
textHeight: number;
|
2019-03-10 23:31:13 +08:00
|
|
|
constructor(config?: TextConfig) {
|
2019-02-21 00:01:38 +08:00
|
|
|
super(checkDefaultFill(config));
|
2019-01-02 04:59:27 +08:00
|
|
|
// update text data for certain attr changes
|
|
|
|
for (var n = 0; n < attrChangeListLen; n++) {
|
|
|
|
this.on(ATTR_CHANGE_LIST[n] + CHANGE_KONVA, this._setTextData);
|
|
|
|
}
|
|
|
|
this._setTextData();
|
|
|
|
}
|
|
|
|
|
2023-03-15 23:30:11 +08:00
|
|
|
_sceneFunc(context: Context) {
|
2020-09-18 02:25:46 +08:00
|
|
|
var textArr = this.textArr,
|
|
|
|
textArrLen = textArr.length;
|
|
|
|
|
|
|
|
if (!this.text()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-01-02 04:59:27 +08:00
|
|
|
var padding = this.padding(),
|
2019-02-19 21:41:32 +08:00
|
|
|
fontSize = this.fontSize(),
|
|
|
|
lineHeightPx = this.lineHeight() * fontSize,
|
2019-01-02 04:59:27 +08:00
|
|
|
verticalAlign = this.verticalAlign(),
|
|
|
|
alignY = 0,
|
|
|
|
align = this.align(),
|
|
|
|
totalWidth = this.getWidth(),
|
|
|
|
letterSpacing = this.letterSpacing(),
|
|
|
|
fill = this.fill(),
|
2019-01-16 02:39:40 +08:00
|
|
|
textDecoration = this.textDecoration(),
|
|
|
|
shouldUnderline = textDecoration.indexOf('underline') !== -1,
|
|
|
|
shouldLineThrough = textDecoration.indexOf('line-through') !== -1,
|
2019-01-02 04:59:27 +08:00
|
|
|
n;
|
|
|
|
|
2019-02-21 00:01:38 +08:00
|
|
|
var translateY = 0;
|
2019-02-21 00:33:00 +08:00
|
|
|
var translateY = lineHeightPx / 2;
|
2019-02-21 00:01:38 +08:00
|
|
|
|
|
|
|
var lineTranslateX = 0;
|
|
|
|
var lineTranslateY = 0;
|
|
|
|
|
2019-01-02 04:59:27 +08:00
|
|
|
context.setAttr('font', this._getContextFont());
|
|
|
|
|
|
|
|
context.setAttr('textBaseline', MIDDLE);
|
2019-01-11 21:51:46 +08:00
|
|
|
|
2019-01-02 04:59:27 +08:00
|
|
|
context.setAttr('textAlign', LEFT);
|
|
|
|
|
|
|
|
// handle vertical alignment
|
|
|
|
if (verticalAlign === MIDDLE) {
|
|
|
|
alignY = (this.getHeight() - textArrLen * lineHeightPx - padding * 2) / 2;
|
|
|
|
} else if (verticalAlign === BOTTOM) {
|
|
|
|
alignY = this.getHeight() - textArrLen * lineHeightPx - padding * 2;
|
|
|
|
}
|
|
|
|
|
2019-02-21 00:33:00 +08:00
|
|
|
context.translate(padding, alignY + padding);
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
// draw text lines
|
|
|
|
for (n = 0; n < textArrLen; n++) {
|
2019-02-21 00:01:38 +08:00
|
|
|
var lineTranslateX = 0;
|
|
|
|
var lineTranslateY = 0;
|
2019-01-02 04:59:27 +08:00
|
|
|
var obj = textArr[n],
|
|
|
|
text = obj.text,
|
|
|
|
width = obj.width,
|
2022-02-24 09:21:15 +08:00
|
|
|
lastLine = obj.lastInParagraph,
|
2019-01-02 04:59:27 +08:00
|
|
|
spacesNumber,
|
|
|
|
oneWord,
|
|
|
|
lineWidth;
|
|
|
|
|
|
|
|
// horizontal alignment
|
|
|
|
context.save();
|
|
|
|
if (align === RIGHT) {
|
2019-02-21 00:01:38 +08:00
|
|
|
lineTranslateX += totalWidth - width - padding * 2;
|
2019-01-02 04:59:27 +08:00
|
|
|
} else if (align === CENTER) {
|
2019-02-21 00:01:38 +08:00
|
|
|
lineTranslateX += (totalWidth - width - padding * 2) / 2;
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
|
|
|
|
2019-01-16 02:39:40 +08:00
|
|
|
if (shouldUnderline) {
|
2019-01-02 04:59:27 +08:00
|
|
|
context.save();
|
|
|
|
context.beginPath();
|
|
|
|
|
2019-02-21 00:01:38 +08:00
|
|
|
context.moveTo(
|
|
|
|
lineTranslateX,
|
|
|
|
translateY + lineTranslateY + Math.round(fontSize / 2)
|
|
|
|
);
|
2019-01-02 04:59:27 +08:00
|
|
|
spacesNumber = text.split(' ').length - 1;
|
|
|
|
oneWord = spacesNumber === 0;
|
|
|
|
lineWidth =
|
2023-01-20 07:42:25 +08:00
|
|
|
align === JUSTIFY && !lastLine ? totalWidth - padding * 2 : width;
|
2019-02-21 00:01:38 +08:00
|
|
|
context.lineTo(
|
|
|
|
lineTranslateX + Math.round(lineWidth),
|
|
|
|
translateY + lineTranslateY + Math.round(fontSize / 2)
|
|
|
|
);
|
2019-01-28 04:43:50 +08:00
|
|
|
|
|
|
|
// I have no idea what is real ratio
|
2019-01-02 04:59:27 +08:00
|
|
|
// just /15 looks good enough
|
|
|
|
context.lineWidth = fontSize / 15;
|
2023-01-20 07:42:25 +08:00
|
|
|
|
|
|
|
const gradient = this._getLinearGradient();
|
|
|
|
context.strokeStyle = gradient || fill;
|
2019-01-02 04:59:27 +08:00
|
|
|
context.stroke();
|
|
|
|
context.restore();
|
|
|
|
}
|
2019-01-16 02:39:40 +08:00
|
|
|
if (shouldLineThrough) {
|
2019-01-02 04:59:27 +08:00
|
|
|
context.save();
|
|
|
|
context.beginPath();
|
2019-02-21 00:01:38 +08:00
|
|
|
context.moveTo(lineTranslateX, translateY + lineTranslateY);
|
2019-01-02 04:59:27 +08:00
|
|
|
spacesNumber = text.split(' ').length - 1;
|
|
|
|
oneWord = spacesNumber === 0;
|
|
|
|
lineWidth =
|
|
|
|
align === JUSTIFY && lastLine && !oneWord
|
|
|
|
? totalWidth - padding * 2
|
|
|
|
: width;
|
2019-02-21 00:01:38 +08:00
|
|
|
context.lineTo(
|
|
|
|
lineTranslateX + Math.round(lineWidth),
|
|
|
|
translateY + lineTranslateY
|
|
|
|
);
|
2019-01-02 04:59:27 +08:00
|
|
|
context.lineWidth = fontSize / 15;
|
2023-01-20 07:42:25 +08:00
|
|
|
const gradient = this._getLinearGradient();
|
|
|
|
context.strokeStyle = gradient || fill;
|
2019-01-02 04:59:27 +08:00
|
|
|
context.stroke();
|
|
|
|
context.restore();
|
|
|
|
}
|
|
|
|
if (letterSpacing !== 0 || align === JUSTIFY) {
|
|
|
|
// var words = text.split(' ');
|
|
|
|
spacesNumber = text.split(' ').length - 1;
|
2020-09-14 22:46:26 +08:00
|
|
|
var array = stringToArray(text);
|
|
|
|
for (var li = 0; li < array.length; li++) {
|
|
|
|
var letter = array[li];
|
2019-01-02 04:59:27 +08:00
|
|
|
// skip justify for the last line
|
2022-02-24 09:21:15 +08:00
|
|
|
if (letter === ' ' && !lastLine && align === JUSTIFY) {
|
2020-07-03 06:27:30 +08:00
|
|
|
lineTranslateX += (totalWidth - padding * 2 - width) / spacesNumber;
|
2019-02-21 00:01:38 +08:00
|
|
|
// context.translate(
|
|
|
|
// Math.floor((totalWidth - padding * 2 - width) / spacesNumber),
|
|
|
|
// 0
|
|
|
|
// );
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
2019-02-24 09:54:20 +08:00
|
|
|
this._partialTextX = lineTranslateX;
|
|
|
|
this._partialTextY = translateY + lineTranslateY;
|
|
|
|
this._partialText = letter;
|
2019-01-02 04:59:27 +08:00
|
|
|
context.fillStrokeShape(this);
|
2020-07-03 06:27:30 +08:00
|
|
|
lineTranslateX += this.measureSize(letter).width + letterSpacing;
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
|
|
|
} else {
|
2019-02-24 09:54:20 +08:00
|
|
|
this._partialTextX = lineTranslateX;
|
|
|
|
this._partialTextY = translateY + lineTranslateY;
|
|
|
|
this._partialText = text;
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
context.fillStrokeShape(this);
|
|
|
|
}
|
|
|
|
context.restore();
|
|
|
|
if (textArrLen > 1) {
|
2019-02-21 00:01:38 +08:00
|
|
|
translateY += lineHeightPx;
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-15 23:30:11 +08:00
|
|
|
_hitFunc(context: Context) {
|
2019-01-02 04:59:27 +08:00
|
|
|
var width = this.getWidth(),
|
|
|
|
height = this.getHeight();
|
|
|
|
|
|
|
|
context.beginPath();
|
|
|
|
context.rect(0, 0, width, height);
|
|
|
|
context.closePath();
|
|
|
|
context.fillStrokeShape(this);
|
|
|
|
}
|
2023-03-15 23:30:11 +08:00
|
|
|
setText(text: string) {
|
2020-03-27 03:41:06 +08:00
|
|
|
var str = Util._isString(text)
|
|
|
|
? text
|
|
|
|
: text === null || text === undefined
|
|
|
|
? ''
|
|
|
|
: text + '';
|
2019-01-02 04:59:27 +08:00
|
|
|
this._setAttr(TEXT, str);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
getWidth() {
|
|
|
|
var isAuto = this.attrs.width === AUTO || this.attrs.width === undefined;
|
|
|
|
return isAuto ? this.getTextWidth() + this.padding() * 2 : this.attrs.width;
|
|
|
|
}
|
|
|
|
getHeight() {
|
|
|
|
var isAuto = this.attrs.height === AUTO || this.attrs.height === undefined;
|
|
|
|
return isAuto
|
2019-02-19 21:41:32 +08:00
|
|
|
? this.fontSize() * this.textArr.length * this.lineHeight() +
|
2019-01-02 04:59:27 +08:00
|
|
|
this.padding() * 2
|
|
|
|
: this.attrs.height;
|
|
|
|
}
|
|
|
|
/**
|
2019-01-06 16:01:20 +08:00
|
|
|
* get pure text width without padding
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#getTextWidth
|
2019-01-02 04:59:27 +08:00
|
|
|
* @returns {Number}
|
|
|
|
*/
|
|
|
|
getTextWidth() {
|
|
|
|
return this.textWidth;
|
|
|
|
}
|
|
|
|
getTextHeight() {
|
2019-02-19 21:41:32 +08:00
|
|
|
Util.warn(
|
|
|
|
'text.getTextHeight() method is deprecated. Use text.height() - for full height and text.fontSize() - for one line height.'
|
|
|
|
);
|
2019-01-02 04:59:27 +08:00
|
|
|
return this.textHeight;
|
|
|
|
}
|
2019-01-24 21:45:17 +08:00
|
|
|
|
2019-02-14 12:41:32 +08:00
|
|
|
/**
|
|
|
|
* measure string with the font of current text shape.
|
|
|
|
* That method can't handle multiline text.
|
|
|
|
* @method
|
|
|
|
* @name Konva.Text#measureSize
|
2019-02-19 01:12:03 +08:00
|
|
|
* @param {String} [text] text to measure
|
2019-02-14 12:41:32 +08:00
|
|
|
* @returns {Object} { width , height} of measured text
|
|
|
|
*/
|
|
|
|
measureSize(text) {
|
2019-01-02 04:59:27 +08:00
|
|
|
var _context = getDummyContext(),
|
|
|
|
fontSize = this.fontSize(),
|
|
|
|
metrics;
|
|
|
|
|
|
|
|
_context.save();
|
|
|
|
_context.font = this._getContextFont();
|
|
|
|
|
|
|
|
metrics = _context.measureText(text);
|
|
|
|
_context.restore();
|
|
|
|
return {
|
|
|
|
width: metrics.width,
|
2020-05-15 00:13:47 +08:00
|
|
|
height: fontSize,
|
2019-01-02 04:59:27 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
_getContextFont() {
|
|
|
|
return (
|
|
|
|
this.fontStyle() +
|
|
|
|
SPACE +
|
|
|
|
this.fontVariant() +
|
|
|
|
SPACE +
|
2020-05-15 00:25:27 +08:00
|
|
|
(this.fontSize() + PX_SPACE) +
|
2020-05-15 00:13:47 +08:00
|
|
|
// wrap font family into " so font families with spaces works ok
|
2020-06-26 01:18:31 +08:00
|
|
|
normalizeFontFamily(this.fontFamily())
|
2019-01-02 04:59:27 +08:00
|
|
|
);
|
|
|
|
}
|
2023-03-15 23:30:11 +08:00
|
|
|
_addTextLine(line: string) {
|
2023-01-20 07:42:25 +08:00
|
|
|
const align = this.align();
|
|
|
|
if (align === JUSTIFY) {
|
2019-01-02 04:59:27 +08:00
|
|
|
line = line.trim();
|
|
|
|
}
|
|
|
|
var width = this._getTextWidth(line);
|
2022-02-24 09:21:15 +08:00
|
|
|
return this.textArr.push({
|
|
|
|
text: line,
|
|
|
|
width: width,
|
|
|
|
lastInParagraph: false,
|
|
|
|
});
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
2023-03-15 23:30:11 +08:00
|
|
|
_getTextWidth(text: string) {
|
2019-01-12 02:46:44 +08:00
|
|
|
var letterSpacing = this.letterSpacing();
|
2019-01-02 04:59:27 +08:00
|
|
|
var length = text.length;
|
|
|
|
return (
|
|
|
|
getDummyContext().measureText(text).width +
|
2019-01-12 02:46:44 +08:00
|
|
|
(length ? letterSpacing * (length - 1) : 0)
|
2019-01-02 04:59:27 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
_setTextData() {
|
|
|
|
var lines = this.text().split('\n'),
|
|
|
|
fontSize = +this.fontSize(),
|
|
|
|
textWidth = 0,
|
|
|
|
lineHeightPx = this.lineHeight() * fontSize,
|
|
|
|
width = this.attrs.width,
|
|
|
|
height = this.attrs.height,
|
2019-01-07 03:06:37 +08:00
|
|
|
fixedWidth = width !== AUTO && width !== undefined,
|
|
|
|
fixedHeight = height !== AUTO && height !== undefined,
|
2019-01-02 04:59:27 +08:00
|
|
|
padding = this.padding(),
|
|
|
|
maxWidth = width - padding * 2,
|
|
|
|
maxHeightPx = height - padding * 2,
|
|
|
|
currentHeightPx = 0,
|
|
|
|
wrap = this.wrap(),
|
|
|
|
// align = this.align(),
|
|
|
|
shouldWrap = wrap !== NONE,
|
|
|
|
wrapAtWord = wrap !== CHAR && shouldWrap,
|
2020-09-16 22:55:40 +08:00
|
|
|
shouldAddEllipsis = this.ellipsis();
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
this.textArr = [];
|
|
|
|
getDummyContext().font = this._getContextFont();
|
2019-01-20 10:29:52 +08:00
|
|
|
var additionalWidth = shouldAddEllipsis ? this._getTextWidth(ELLIPSIS) : 0;
|
2019-01-02 04:59:27 +08:00
|
|
|
for (var i = 0, max = lines.length; i < max; ++i) {
|
|
|
|
var line = lines[i];
|
|
|
|
|
|
|
|
var lineWidth = this._getTextWidth(line);
|
|
|
|
if (fixedWidth && lineWidth > maxWidth) {
|
|
|
|
/*
|
|
|
|
* if width is fixed and line does not fit entirely
|
|
|
|
* break the line into multiple fitting lines
|
|
|
|
*/
|
|
|
|
while (line.length > 0) {
|
|
|
|
/*
|
|
|
|
* use binary search to find the longest substring that
|
|
|
|
* that would fit in the specified width
|
|
|
|
*/
|
|
|
|
var low = 0,
|
|
|
|
high = line.length,
|
|
|
|
match = '',
|
|
|
|
matchWidth = 0;
|
|
|
|
while (low < high) {
|
|
|
|
var mid = (low + high) >>> 1,
|
|
|
|
substr = line.slice(0, mid + 1),
|
|
|
|
substrWidth = this._getTextWidth(substr) + additionalWidth;
|
|
|
|
if (substrWidth <= maxWidth) {
|
|
|
|
low = mid + 1;
|
2020-12-19 00:12:19 +08:00
|
|
|
match = substr;
|
2019-01-02 04:59:27 +08:00
|
|
|
matchWidth = substrWidth;
|
|
|
|
} else {
|
|
|
|
high = mid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* 'low' is now the index of the substring end
|
|
|
|
* 'match' is the substring
|
|
|
|
* 'matchWidth' is the substring width in px
|
|
|
|
*/
|
|
|
|
if (match) {
|
|
|
|
// a fitting substring was found
|
|
|
|
if (wrapAtWord) {
|
|
|
|
// try to find a space or dash where wrapping could be done
|
|
|
|
var wrapIndex;
|
|
|
|
var nextChar = line[match.length];
|
|
|
|
var nextIsSpaceOrDash = nextChar === SPACE || nextChar === DASH;
|
|
|
|
if (nextIsSpaceOrDash && matchWidth <= maxWidth) {
|
|
|
|
wrapIndex = match.length;
|
|
|
|
} else {
|
|
|
|
wrapIndex =
|
|
|
|
Math.max(match.lastIndexOf(SPACE), match.lastIndexOf(DASH)) +
|
|
|
|
1;
|
|
|
|
}
|
|
|
|
if (wrapIndex > 0) {
|
|
|
|
// re-cut the substring found at the space/dash position
|
|
|
|
low = wrapIndex;
|
|
|
|
match = match.slice(0, low);
|
|
|
|
matchWidth = this._getTextWidth(match);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if (align === 'right') {
|
2019-11-13 03:38:36 +08:00
|
|
|
match = match.trimRight();
|
2019-01-02 04:59:27 +08:00
|
|
|
// }
|
|
|
|
this._addTextLine(match);
|
|
|
|
textWidth = Math.max(textWidth, matchWidth);
|
|
|
|
currentHeightPx += lineHeightPx;
|
2020-09-08 00:03:20 +08:00
|
|
|
|
2022-08-05 23:07:22 +08:00
|
|
|
var shouldHandleEllipsis =
|
|
|
|
this._shouldHandleEllipsis(currentHeightPx);
|
2022-08-01 02:09:00 +08:00
|
|
|
if (shouldHandleEllipsis) {
|
2022-08-05 23:07:22 +08:00
|
|
|
this._tryToAddEllipsisToLastLine();
|
2019-01-02 04:59:27 +08:00
|
|
|
/*
|
|
|
|
* stop wrapping if wrapping is disabled or if adding
|
|
|
|
* one more line would overflow the fixed height
|
|
|
|
*/
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
line = line.slice(low);
|
|
|
|
line = line.trimLeft();
|
|
|
|
if (line.length > 0) {
|
|
|
|
// Check if the remaining text would fit on one line
|
|
|
|
lineWidth = this._getTextWidth(line);
|
|
|
|
if (lineWidth <= maxWidth) {
|
|
|
|
// if it does, add the line and break out of the loop
|
|
|
|
this._addTextLine(line);
|
|
|
|
currentHeightPx += lineHeightPx;
|
|
|
|
textWidth = Math.max(textWidth, lineWidth);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// not even one character could fit in the element, abort
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// element width is automatically adjusted to max line width
|
|
|
|
this._addTextLine(line);
|
|
|
|
currentHeightPx += lineHeightPx;
|
|
|
|
textWidth = Math.max(textWidth, lineWidth);
|
2022-08-29 22:38:49 +08:00
|
|
|
if (this._shouldHandleEllipsis(currentHeightPx) && i < max - 1) {
|
2022-08-05 23:07:22 +08:00
|
|
|
this._tryToAddEllipsisToLastLine();
|
|
|
|
}
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
|
|
|
// if element height is fixed, abort if adding one more line would overflow
|
2022-02-24 09:21:15 +08:00
|
|
|
if (this.textArr[this.textArr.length - 1]) {
|
|
|
|
this.textArr[this.textArr.length - 1].lastInParagraph = true;
|
|
|
|
}
|
2023-01-20 23:53:49 +08:00
|
|
|
if (fixedHeight && currentHeightPx + lineHeightPx > maxHeightPx) {
|
|
|
|
break;
|
|
|
|
}
|
2019-01-02 04:59:27 +08:00
|
|
|
}
|
|
|
|
this.textHeight = fontSize;
|
|
|
|
// var maxTextWidth = 0;
|
|
|
|
// for(var j = 0; j < this.textArr.length; j++) {
|
|
|
|
// maxTextWidth = Math.max(maxTextWidth, this.textArr[j].width);
|
|
|
|
// }
|
|
|
|
this.textWidth = textWidth;
|
|
|
|
}
|
|
|
|
|
2022-08-01 02:09:00 +08:00
|
|
|
/**
|
|
|
|
* whether to handle ellipsis, there are two cases:
|
|
|
|
* 1. the current line is the last line
|
|
|
|
* 2. wrap is NONE
|
|
|
|
* @param {Number} currentHeightPx
|
2022-08-05 23:07:22 +08:00
|
|
|
* @returns
|
2022-08-01 02:09:00 +08:00
|
|
|
*/
|
2022-08-05 23:07:22 +08:00
|
|
|
_shouldHandleEllipsis(currentHeightPx: number): boolean {
|
2022-08-01 02:09:00 +08:00
|
|
|
var fontSize = +this.fontSize(),
|
|
|
|
lineHeightPx = this.lineHeight() * fontSize,
|
|
|
|
height = this.attrs.height,
|
|
|
|
fixedHeight = height !== AUTO && height !== undefined,
|
|
|
|
padding = this.padding(),
|
|
|
|
maxHeightPx = height - padding * 2,
|
|
|
|
wrap = this.wrap(),
|
|
|
|
shouldWrap = wrap !== NONE;
|
|
|
|
|
2022-08-05 23:07:22 +08:00
|
|
|
return (
|
|
|
|
!shouldWrap ||
|
|
|
|
(fixedHeight && currentHeightPx + lineHeightPx > maxHeightPx)
|
|
|
|
);
|
2022-08-01 02:09:00 +08:00
|
|
|
}
|
|
|
|
|
2022-08-05 23:07:22 +08:00
|
|
|
_tryToAddEllipsisToLastLine(): void {
|
2022-08-01 02:09:00 +08:00
|
|
|
var width = this.attrs.width,
|
|
|
|
fixedWidth = width !== AUTO && width !== undefined,
|
|
|
|
padding = this.padding(),
|
|
|
|
maxWidth = width - padding * 2,
|
|
|
|
shouldAddEllipsis = this.ellipsis();
|
2022-08-05 23:07:22 +08:00
|
|
|
|
2022-08-01 02:09:00 +08:00
|
|
|
var lastLine = this.textArr[this.textArr.length - 1];
|
2022-08-05 23:07:22 +08:00
|
|
|
if (!lastLine || !shouldAddEllipsis) {
|
2022-08-01 02:09:00 +08:00
|
|
|
return;
|
|
|
|
}
|
2022-08-05 23:07:22 +08:00
|
|
|
|
2022-08-01 02:09:00 +08:00
|
|
|
if (fixedWidth) {
|
2022-08-05 23:07:22 +08:00
|
|
|
var haveSpace = this._getTextWidth(lastLine.text + ELLIPSIS) < maxWidth;
|
2022-08-01 02:09:00 +08:00
|
|
|
if (!haveSpace) {
|
2022-08-05 23:07:22 +08:00
|
|
|
lastLine.text = lastLine.text.slice(0, lastLine.text.length - 3);
|
2022-08-01 02:09:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.textArr.splice(this.textArr.length - 1, 1);
|
|
|
|
this._addTextLine(lastLine.text + ELLIPSIS);
|
|
|
|
}
|
|
|
|
|
2019-01-02 04:59:27 +08:00
|
|
|
// for text we can't disable stroke scaling
|
|
|
|
// if we do, the result will be unexpected
|
|
|
|
getStrokeScaleEnabled() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
fontFamily: GetSet<string, this>;
|
|
|
|
fontSize: GetSet<number, this>;
|
|
|
|
fontStyle: GetSet<string, this>;
|
|
|
|
fontVariant: GetSet<string, this>;
|
|
|
|
align: GetSet<string, this>;
|
|
|
|
letterSpacing: GetSet<number, this>;
|
|
|
|
verticalAlign: GetSet<string, this>;
|
|
|
|
padding: GetSet<number, this>;
|
|
|
|
lineHeight: GetSet<number, this>;
|
|
|
|
textDecoration: GetSet<string, this>;
|
|
|
|
text: GetSet<string, this>;
|
|
|
|
wrap: GetSet<string, this>;
|
|
|
|
ellipsis: GetSet<boolean, this>;
|
|
|
|
}
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Text.prototype._fillFunc = _fillFunc;
|
|
|
|
Text.prototype._strokeFunc = _strokeFunc;
|
|
|
|
Text.prototype.className = TEXT_UPPER;
|
2019-02-19 21:36:16 +08:00
|
|
|
Text.prototype._attrsAffectingSize = [
|
|
|
|
'text',
|
|
|
|
'fontSize',
|
|
|
|
'padding',
|
|
|
|
'wrap',
|
2020-05-15 00:13:47 +08:00
|
|
|
'lineHeight',
|
2021-03-04 01:06:37 +08:00
|
|
|
'letterSpacing',
|
2019-02-19 21:36:16 +08:00
|
|
|
];
|
2019-02-27 21:06:04 +08:00
|
|
|
_registerNode(Text);
|
2019-01-06 16:01:20 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set width of text area, which includes padding.
|
|
|
|
* @name Konva.Text#width
|
|
|
|
* @method
|
|
|
|
* @param {Number} width
|
|
|
|
* @returns {Number}
|
|
|
|
* @example
|
|
|
|
* // get width
|
|
|
|
* var width = text.width();
|
|
|
|
*
|
|
|
|
* // set width
|
|
|
|
* text.width(20);
|
|
|
|
*
|
|
|
|
* // set to auto
|
|
|
|
* text.width('auto');
|
|
|
|
* text.width() // will return calculated width, and not "auto"
|
|
|
|
*/
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.overWriteSetter(Text, 'width', getNumberOrAutoValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
/**
|
|
|
|
* get/set the height of the text area, which takes into account multi-line text, line heights, and padding.
|
|
|
|
* @name Konva.Text#height
|
|
|
|
* @method
|
|
|
|
* @param {Number} height
|
|
|
|
* @returns {Number}
|
|
|
|
* @example
|
|
|
|
* // get height
|
|
|
|
* var height = text.height();
|
|
|
|
*
|
|
|
|
* // set height
|
|
|
|
* text.height(20);
|
|
|
|
*
|
|
|
|
* // set to auto
|
|
|
|
* text.height('auto');
|
|
|
|
* text.height() // will return calculated height, and not "auto"
|
|
|
|
*/
|
2019-01-02 04:59:27 +08:00
|
|
|
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.overWriteSetter(Text, 'height', getNumberOrAutoValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set font family
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#fontFamily
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} fontFamily
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get font family
|
|
|
|
* var fontFamily = text.fontFamily();
|
|
|
|
*
|
|
|
|
* // set font family
|
|
|
|
* text.fontFamily('Arial');
|
|
|
|
*/
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'fontFamily', 'Arial');
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set font size in pixels
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#fontSize
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {Number} fontSize
|
|
|
|
* @returns {Number}
|
|
|
|
* @example
|
|
|
|
* // get font size
|
|
|
|
* var fontSize = text.fontSize();
|
|
|
|
*
|
|
|
|
* // set font size to 22px
|
|
|
|
* text.fontSize(22);
|
|
|
|
*/
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.addGetterSetter(Text, 'fontSize', 12, getNumberValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
2021-03-15 20:34:20 +08:00
|
|
|
* get/set font style. Can be 'normal', 'italic', or 'bold' or even 'italic bold'. 'normal' is the default.
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#fontStyle
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} fontStyle
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get font style
|
|
|
|
* var fontStyle = text.fontStyle();
|
|
|
|
*
|
|
|
|
* // set font style
|
|
|
|
* text.fontStyle('bold');
|
|
|
|
*/
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'fontStyle', NORMAL);
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
2019-01-06 16:01:20 +08:00
|
|
|
* get/set font variant. Can be 'normal' or 'small-caps'. 'normal' is the default.
|
|
|
|
* @name Konva.Text#fontVariant
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} fontVariant
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get font variant
|
|
|
|
* var fontVariant = text.fontVariant();
|
|
|
|
*
|
|
|
|
* // set font variant
|
|
|
|
* text.fontVariant('small-caps');
|
|
|
|
*/
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'fontVariant', NORMAL);
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
2019-01-06 16:01:20 +08:00
|
|
|
* get/set padding
|
|
|
|
* @name Konva.Text#padding
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {Number} padding
|
|
|
|
* @returns {Number}
|
|
|
|
* @example
|
|
|
|
* // get padding
|
|
|
|
* var padding = text.padding();
|
|
|
|
*
|
|
|
|
* // set padding to 10 pixels
|
|
|
|
* text.padding(10);
|
|
|
|
*/
|
|
|
|
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.addGetterSetter(Text, 'padding', 0, getNumberValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set horizontal align of text. Can be 'left', 'center', 'right' or 'justify'
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#align
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} align
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get text align
|
|
|
|
* var align = text.align();
|
|
|
|
*
|
|
|
|
* // center text
|
|
|
|
* text.align('center');
|
|
|
|
*
|
|
|
|
* // align text to right
|
|
|
|
* text.align('right');
|
|
|
|
*/
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'align', LEFT);
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set vertical align of text. Can be 'top', 'middle', 'bottom'.
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#verticalAlign
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} verticalAlign
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get text vertical align
|
|
|
|
* var verticalAlign = text.verticalAlign();
|
|
|
|
*
|
|
|
|
* // center text
|
|
|
|
* text.verticalAlign('middle');
|
|
|
|
*/
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'verticalAlign', TOP);
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set line height. The default is 1.
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#lineHeight
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {Number} lineHeight
|
|
|
|
* @returns {Number}
|
|
|
|
* @example
|
|
|
|
* // get line height
|
|
|
|
* var lineHeight = text.lineHeight();
|
|
|
|
*
|
|
|
|
* // set the line height
|
|
|
|
* text.lineHeight(2);
|
|
|
|
*/
|
|
|
|
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.addGetterSetter(Text, 'lineHeight', 1, getNumberValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
2019-01-11 21:51:46 +08:00
|
|
|
* 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.
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#wrap
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} wrap
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get wrap
|
|
|
|
* var wrap = text.wrap();
|
|
|
|
*
|
|
|
|
* // set wrap
|
|
|
|
* text.wrap('word');
|
|
|
|
*/
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'wrap', WORD);
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
2020-09-08 00:03:20 +08:00
|
|
|
* get/set ellipsis. Can be true or false. Default is false. If ellipses is true,
|
|
|
|
* Konva will add "..." at the end of the text if it doesn't have enough space to write characters.
|
|
|
|
* That is possible only when you limit both width and height of the text
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#ellipsis
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {Boolean} ellipsis
|
|
|
|
* @returns {Boolean}
|
|
|
|
* @example
|
2020-09-08 00:03:20 +08:00
|
|
|
* // get ellipsis param, returns true or false
|
2019-01-02 04:59:27 +08:00
|
|
|
* var ellipsis = text.ellipsis();
|
|
|
|
*
|
|
|
|
* // set ellipsis
|
|
|
|
* text.ellipsis(true);
|
|
|
|
*/
|
|
|
|
|
2020-09-08 00:03:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'ellipsis', false, getBooleanValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* set letter spacing property. Default value is 0.
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#letterSpacing
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {Number} letterSpacing
|
|
|
|
*/
|
|
|
|
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.addGetterSetter(Text, 'letterSpacing', 0, getNumberValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set text
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#text
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} text
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get text
|
|
|
|
* var text = text.text();
|
|
|
|
*
|
|
|
|
* // set text
|
|
|
|
* text.text('Hello world!');
|
|
|
|
*/
|
|
|
|
|
2019-02-25 01:06:04 +08:00
|
|
|
Factory.addGetterSetter(Text, 'text', '', getStringValidator());
|
2019-01-02 04:59:27 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* get/set text decoration of a text. Possible values are 'underline', 'line-through' or combination of these values separated by space
|
2019-01-06 16:01:20 +08:00
|
|
|
* @name Konva.Text#textDecoration
|
2019-01-02 04:59:27 +08:00
|
|
|
* @method
|
|
|
|
* @param {String} textDecoration
|
|
|
|
* @returns {String}
|
|
|
|
* @example
|
|
|
|
* // get text decoration
|
|
|
|
* var textDecoration = text.textDecoration();
|
|
|
|
*
|
|
|
|
* // underline text
|
|
|
|
* text.textDecoration('underline');
|
|
|
|
*
|
|
|
|
* // strike text
|
|
|
|
* text.textDecoration('line-through');
|
|
|
|
*
|
|
|
|
* // underline and strike text
|
|
|
|
* text.textDecoration('underline line-through');
|
|
|
|
*/
|
|
|
|
|
2019-01-06 16:01:20 +08:00
|
|
|
Factory.addGetterSetter(Text, 'textDecoration', '');
|