[d565449] | 1 | "use strict";
|
---|
| 2 | Object.defineProperty(exports, "__esModule", { value: true });
|
---|
| 3 | exports.TextPath = void 0;
|
---|
| 4 | const Util_1 = require("../Util");
|
---|
| 5 | const Factory_1 = require("../Factory");
|
---|
| 6 | const Shape_1 = require("../Shape");
|
---|
| 7 | const Path_1 = require("./Path");
|
---|
| 8 | const Text_1 = require("./Text");
|
---|
| 9 | const Validators_1 = require("../Validators");
|
---|
| 10 | const Global_1 = require("../Global");
|
---|
[0c6b92a] | 11 | const EMPTY_STRING = '', NORMAL = 'normal';
|
---|
[d565449] | 12 | function _fillFunc(context) {
|
---|
| 13 | context.fillText(this.partialText, 0, 0);
|
---|
| 14 | }
|
---|
| 15 | function _strokeFunc(context) {
|
---|
| 16 | context.strokeText(this.partialText, 0, 0);
|
---|
| 17 | }
|
---|
| 18 | class TextPath extends Shape_1.Shape {
|
---|
| 19 | constructor(config) {
|
---|
| 20 | super(config);
|
---|
| 21 | this.dummyCanvas = Util_1.Util.createCanvasElement();
|
---|
| 22 | this.dataArray = [];
|
---|
| 23 | this._readDataAttribute();
|
---|
| 24 | this.on('dataChange.konva', function () {
|
---|
| 25 | this._readDataAttribute();
|
---|
| 26 | this._setTextData();
|
---|
| 27 | });
|
---|
| 28 | this.on('textChange.konva alignChange.konva letterSpacingChange.konva kerningFuncChange.konva fontSizeChange.konva fontFamilyChange.konva', this._setTextData);
|
---|
| 29 | this._setTextData();
|
---|
| 30 | }
|
---|
| 31 | _getTextPathLength() {
|
---|
| 32 | return Path_1.Path.getPathLength(this.dataArray);
|
---|
| 33 | }
|
---|
| 34 | _getPointAtLength(length) {
|
---|
| 35 | if (!this.attrs.data) {
|
---|
| 36 | return null;
|
---|
| 37 | }
|
---|
| 38 | const totalLength = this.pathLength;
|
---|
| 39 | if (length - 1 > totalLength) {
|
---|
| 40 | return null;
|
---|
| 41 | }
|
---|
| 42 | return Path_1.Path.getPointAtLengthOfDataArray(length, this.dataArray);
|
---|
| 43 | }
|
---|
| 44 | _readDataAttribute() {
|
---|
| 45 | this.dataArray = Path_1.Path.parsePathData(this.attrs.data);
|
---|
| 46 | this.pathLength = this._getTextPathLength();
|
---|
| 47 | }
|
---|
| 48 | _sceneFunc(context) {
|
---|
| 49 | context.setAttr('font', this._getContextFont());
|
---|
| 50 | context.setAttr('textBaseline', this.textBaseline());
|
---|
| 51 | context.setAttr('textAlign', 'left');
|
---|
| 52 | context.save();
|
---|
[0c6b92a] | 53 | const textDecoration = this.textDecoration();
|
---|
| 54 | const fill = this.fill();
|
---|
| 55 | const fontSize = this.fontSize();
|
---|
| 56 | const glyphInfo = this.glyphInfo;
|
---|
[d565449] | 57 | if (textDecoration === 'underline') {
|
---|
| 58 | context.beginPath();
|
---|
| 59 | }
|
---|
[0c6b92a] | 60 | for (let i = 0; i < glyphInfo.length; i++) {
|
---|
[d565449] | 61 | context.save();
|
---|
[0c6b92a] | 62 | const p0 = glyphInfo[i].p0;
|
---|
[d565449] | 63 | context.translate(p0.x, p0.y);
|
---|
| 64 | context.rotate(glyphInfo[i].rotation);
|
---|
| 65 | this.partialText = glyphInfo[i].text;
|
---|
| 66 | context.fillStrokeShape(this);
|
---|
| 67 | if (textDecoration === 'underline') {
|
---|
| 68 | if (i === 0) {
|
---|
| 69 | context.moveTo(0, fontSize / 2 + 1);
|
---|
| 70 | }
|
---|
| 71 | context.lineTo(fontSize, fontSize / 2 + 1);
|
---|
| 72 | }
|
---|
| 73 | context.restore();
|
---|
| 74 | }
|
---|
| 75 | if (textDecoration === 'underline') {
|
---|
| 76 | context.strokeStyle = fill;
|
---|
| 77 | context.lineWidth = fontSize / 20;
|
---|
| 78 | context.stroke();
|
---|
| 79 | }
|
---|
| 80 | context.restore();
|
---|
| 81 | }
|
---|
| 82 | _hitFunc(context) {
|
---|
| 83 | context.beginPath();
|
---|
[0c6b92a] | 84 | const glyphInfo = this.glyphInfo;
|
---|
[d565449] | 85 | if (glyphInfo.length >= 1) {
|
---|
[0c6b92a] | 86 | const p0 = glyphInfo[0].p0;
|
---|
[d565449] | 87 | context.moveTo(p0.x, p0.y);
|
---|
| 88 | }
|
---|
[0c6b92a] | 89 | for (let i = 0; i < glyphInfo.length; i++) {
|
---|
| 90 | const p1 = glyphInfo[i].p1;
|
---|
[d565449] | 91 | context.lineTo(p1.x, p1.y);
|
---|
| 92 | }
|
---|
| 93 | context.setAttr('lineWidth', this.fontSize());
|
---|
| 94 | context.setAttr('strokeStyle', this.colorKey);
|
---|
| 95 | context.stroke();
|
---|
| 96 | }
|
---|
| 97 | getTextWidth() {
|
---|
| 98 | return this.textWidth;
|
---|
| 99 | }
|
---|
| 100 | getTextHeight() {
|
---|
| 101 | Util_1.Util.warn('text.getTextHeight() method is deprecated. Use text.height() - for full height and text.fontSize() - for one line height.');
|
---|
| 102 | return this.textHeight;
|
---|
| 103 | }
|
---|
| 104 | setText(text) {
|
---|
| 105 | return Text_1.Text.prototype.setText.call(this, text);
|
---|
| 106 | }
|
---|
| 107 | _getContextFont() {
|
---|
| 108 | return Text_1.Text.prototype._getContextFont.call(this);
|
---|
| 109 | }
|
---|
| 110 | _getTextSize(text) {
|
---|
[0c6b92a] | 111 | const dummyCanvas = this.dummyCanvas;
|
---|
| 112 | const _context = dummyCanvas.getContext('2d');
|
---|
[d565449] | 113 | _context.save();
|
---|
| 114 | _context.font = this._getContextFont();
|
---|
[0c6b92a] | 115 | const metrics = _context.measureText(text);
|
---|
[d565449] | 116 | _context.restore();
|
---|
| 117 | return {
|
---|
| 118 | width: metrics.width,
|
---|
| 119 | height: parseInt(`${this.fontSize()}`, 10),
|
---|
| 120 | };
|
---|
| 121 | }
|
---|
| 122 | _setTextData() {
|
---|
| 123 | const { width, height } = this._getTextSize(this.attrs.text);
|
---|
| 124 | this.textWidth = width;
|
---|
| 125 | this.textHeight = height;
|
---|
| 126 | this.glyphInfo = [];
|
---|
| 127 | if (!this.attrs.data) {
|
---|
| 128 | return null;
|
---|
| 129 | }
|
---|
| 130 | const letterSpacing = this.letterSpacing();
|
---|
| 131 | const align = this.align();
|
---|
| 132 | const kerningFunc = this.kerningFunc();
|
---|
| 133 | const textWidth = Math.max(this.textWidth + ((this.attrs.text || '').length - 1) * letterSpacing, 0);
|
---|
| 134 | let offset = 0;
|
---|
| 135 | if (align === 'center') {
|
---|
| 136 | offset = Math.max(0, this.pathLength / 2 - textWidth / 2);
|
---|
| 137 | }
|
---|
| 138 | if (align === 'right') {
|
---|
| 139 | offset = Math.max(0, this.pathLength - textWidth);
|
---|
| 140 | }
|
---|
| 141 | const charArr = (0, Text_1.stringToArray)(this.text());
|
---|
| 142 | let offsetToGlyph = offset;
|
---|
[0c6b92a] | 143 | for (let i = 0; i < charArr.length; i++) {
|
---|
[d565449] | 144 | const charStartPoint = this._getPointAtLength(offsetToGlyph);
|
---|
| 145 | if (!charStartPoint)
|
---|
| 146 | return;
|
---|
| 147 | let glyphWidth = this._getTextSize(charArr[i]).width + letterSpacing;
|
---|
| 148 | if (charArr[i] === ' ' && align === 'justify') {
|
---|
| 149 | const numberOfSpaces = this.text().split(' ').length - 1;
|
---|
| 150 | glyphWidth += (this.pathLength - textWidth) / numberOfSpaces;
|
---|
| 151 | }
|
---|
| 152 | const charEndPoint = this._getPointAtLength(offsetToGlyph + glyphWidth);
|
---|
| 153 | if (!charEndPoint)
|
---|
| 154 | return;
|
---|
| 155 | const width = Path_1.Path.getLineLength(charStartPoint.x, charStartPoint.y, charEndPoint.x, charEndPoint.y);
|
---|
| 156 | let kern = 0;
|
---|
| 157 | if (kerningFunc) {
|
---|
| 158 | try {
|
---|
| 159 | kern = kerningFunc(charArr[i - 1], charArr[i]) * this.fontSize();
|
---|
| 160 | }
|
---|
| 161 | catch (e) {
|
---|
| 162 | kern = 0;
|
---|
| 163 | }
|
---|
| 164 | }
|
---|
| 165 | charStartPoint.x += kern;
|
---|
| 166 | charEndPoint.x += kern;
|
---|
| 167 | this.textWidth += kern;
|
---|
| 168 | const midpoint = Path_1.Path.getPointOnLine(kern + width / 2.0, charStartPoint.x, charStartPoint.y, charEndPoint.x, charEndPoint.y);
|
---|
| 169 | const rotation = Math.atan2(charEndPoint.y - charStartPoint.y, charEndPoint.x - charStartPoint.x);
|
---|
| 170 | this.glyphInfo.push({
|
---|
| 171 | transposeX: midpoint.x,
|
---|
| 172 | transposeY: midpoint.y,
|
---|
| 173 | text: charArr[i],
|
---|
| 174 | rotation: rotation,
|
---|
| 175 | p0: charStartPoint,
|
---|
| 176 | p1: charEndPoint,
|
---|
| 177 | });
|
---|
| 178 | offsetToGlyph += glyphWidth;
|
---|
| 179 | }
|
---|
| 180 | }
|
---|
| 181 | getSelfRect() {
|
---|
| 182 | if (!this.glyphInfo.length) {
|
---|
| 183 | return {
|
---|
| 184 | x: 0,
|
---|
| 185 | y: 0,
|
---|
| 186 | width: 0,
|
---|
| 187 | height: 0,
|
---|
| 188 | };
|
---|
| 189 | }
|
---|
[0c6b92a] | 190 | const points = [];
|
---|
[d565449] | 191 | this.glyphInfo.forEach(function (info) {
|
---|
| 192 | points.push(info.p0.x);
|
---|
| 193 | points.push(info.p0.y);
|
---|
| 194 | points.push(info.p1.x);
|
---|
| 195 | points.push(info.p1.y);
|
---|
| 196 | });
|
---|
[0c6b92a] | 197 | let minX = points[0] || 0;
|
---|
| 198 | let maxX = points[0] || 0;
|
---|
| 199 | let minY = points[1] || 0;
|
---|
| 200 | let maxY = points[1] || 0;
|
---|
| 201 | let x, y;
|
---|
| 202 | for (let i = 0; i < points.length / 2; i++) {
|
---|
[d565449] | 203 | x = points[i * 2];
|
---|
| 204 | y = points[i * 2 + 1];
|
---|
| 205 | minX = Math.min(minX, x);
|
---|
| 206 | maxX = Math.max(maxX, x);
|
---|
| 207 | minY = Math.min(minY, y);
|
---|
| 208 | maxY = Math.max(maxY, y);
|
---|
| 209 | }
|
---|
[0c6b92a] | 210 | const fontSize = this.fontSize();
|
---|
[d565449] | 211 | return {
|
---|
| 212 | x: minX - fontSize / 2,
|
---|
| 213 | y: minY - fontSize / 2,
|
---|
| 214 | width: maxX - minX + fontSize,
|
---|
| 215 | height: maxY - minY + fontSize,
|
---|
| 216 | };
|
---|
| 217 | }
|
---|
| 218 | destroy() {
|
---|
| 219 | Util_1.Util.releaseCanvas(this.dummyCanvas);
|
---|
| 220 | return super.destroy();
|
---|
| 221 | }
|
---|
| 222 | }
|
---|
| 223 | exports.TextPath = TextPath;
|
---|
| 224 | TextPath.prototype._fillFunc = _fillFunc;
|
---|
| 225 | TextPath.prototype._strokeFunc = _strokeFunc;
|
---|
| 226 | TextPath.prototype._fillFuncHit = _fillFunc;
|
---|
| 227 | TextPath.prototype._strokeFuncHit = _strokeFunc;
|
---|
| 228 | TextPath.prototype.className = 'TextPath';
|
---|
| 229 | TextPath.prototype._attrsAffectingSize = ['text', 'fontSize', 'data'];
|
---|
| 230 | (0, Global_1._registerNode)(TextPath);
|
---|
| 231 | Factory_1.Factory.addGetterSetter(TextPath, 'data');
|
---|
| 232 | Factory_1.Factory.addGetterSetter(TextPath, 'fontFamily', 'Arial');
|
---|
| 233 | Factory_1.Factory.addGetterSetter(TextPath, 'fontSize', 12, (0, Validators_1.getNumberValidator)());
|
---|
| 234 | Factory_1.Factory.addGetterSetter(TextPath, 'fontStyle', NORMAL);
|
---|
| 235 | Factory_1.Factory.addGetterSetter(TextPath, 'align', 'left');
|
---|
| 236 | Factory_1.Factory.addGetterSetter(TextPath, 'letterSpacing', 0, (0, Validators_1.getNumberValidator)());
|
---|
| 237 | Factory_1.Factory.addGetterSetter(TextPath, 'textBaseline', 'middle');
|
---|
| 238 | Factory_1.Factory.addGetterSetter(TextPath, 'fontVariant', NORMAL);
|
---|
| 239 | Factory_1.Factory.addGetterSetter(TextPath, 'text', EMPTY_STRING);
|
---|
| 240 | Factory_1.Factory.addGetterSetter(TextPath, 'textDecoration', null);
|
---|
| 241 | Factory_1.Factory.addGetterSetter(TextPath, 'kerningFunc', null);
|
---|