UNPKG

@nmmty/lazycanvas

Version:

A simple way to interact with @napi-rs/canvas in an advanced way!

416 lines (415 loc) 17.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.TextLayer = void 0; const BaseLayer_1 = require("./BaseLayer"); const types_1 = require("../../types"); const LazyUtil_1 = require("../../utils/LazyUtil"); const utils_1 = require("../../utils/utils"); /** * Class representing a Text layer, extending the BaseLayer class. */ class TextLayer extends BaseLayer_1.BaseLayer { /** * The properties of the Text Layer. */ props; /** * Constructs a new TextLayer instance. * @param {ITextLayerProps} [props] - The properties of the Text layer. * @param {IBaseLayerMisc} [misc] - Miscellaneous options for the layer. */ constructor(props, misc) { super(types_1.LayerType.Text, props || {}, misc); this.props = props ? props : {}; this.props = this.validateProps(this.props); } /** * Sets the text of the text layer. * @param {string} [text] - The text content of the layer. * @returns {this} The current instance for chaining. */ setText(text) { this.props.text = text; return this; } /** * Sets the font of the text layer. * @param {string | { family: string; size: number; weight: AnyWeight }} [familyOrConfig] - The font family or configuration object. * @param {number} [size] - The font size (required if `familyOrConfig` is a string). * @param {AnyWeight} [weight] - The font weight (required if `familyOrConfig` is a string). * @returns {this} The current instance for chaining. * @throws {LazyError} If size or weight is not provided when `familyOrConfig` is a string. */ setFont(familyOrConfig, size, weight) { if (typeof familyOrConfig === "string") { if (!size) throw new LazyUtil_1.LazyError('The size of the font must be provided'); if (!weight) throw new LazyUtil_1.LazyError('The weight of the font must be provided'); this.props.font = { family: familyOrConfig, size, weight, }; } else { this.props.font = { family: familyOrConfig.family, size: familyOrConfig.size, weight: familyOrConfig.weight, }; } return this; } /** * Configures the multiline properties of the text layer. * @param {ScaleType} [width] - The width of the multiline text area. * @param {ScaleType} [height] - The height of the multiline text area. * @param {number} [spacing] - The spacing between lines (optional). * @returns {this} The current instance for chaining. */ setMultiline(width, height, spacing) { this.props.multiline = { enabled: true, spacing: spacing || 1.1, }; this.props.size = { width, height, }; return this; } /** * Sets the color of the text layer. * @param {ColorType} [color] - The base color of the text. * @param {SubStringColor[]} [sub] - Optional substring colors for partial text coloring. * @returns {this} The current instance for chaining. * @throws {LazyError} If the color is not provided or invalid. */ setColor(color, ...sub) { if (!color) throw new LazyUtil_1.LazyError('The color of the layer must be provided'); if (!(0, utils_1.isColor)(color)) throw new LazyUtil_1.LazyError('The color of the layer must be a valid color'); this.props.fillStyle = color; if (sub && sub.length > 0) { this.props.subStringColors = sub; } return this; } /** * Sets the alignment of the text layer. * @param {AnyTextAlign} [align] - The alignment of the text. * @returns {this} The current instance for chaining. */ setAlign(align) { this.props.align = align; return this; } /** * Sets the baseline of the text layer. * @param {AnyTextBaseline} [baseline] - The baseline of the text. * @returns {this} The current instance for chaining. */ setBaseline(baseline) { this.props.baseline = baseline; return this; } /** * Sets the direction of the text layer. * @param {AnyTextDirection} [direction] - The direction of the text. * @returns {this} The current instance for chaining. */ setDirection(direction) { this.props.direction = direction; return this; } /** * Configures the stroke properties of the text layer. * @param {number} [width] - The width of the stroke. * @param {string} [cap] - The cap style of the stroke. * @param {string} [join] - The join style of the stroke. * @param {number[]} [dash] - The dash pattern of the stroke. * @param {number} [dashOffset] - The dash offset of the stroke. * @param {number} [miterLimit] - The miter limit of the stroke. * @returns {this} The current instance for chaining. */ setStroke(width, cap, join, dash, dashOffset, miterLimit) { this.props.stroke = { width, cap: cap || 'butt', join: join || 'miter', dash: dash || [], dashOffset: dashOffset || 0, miterLimit: miterLimit || 10, }; this.props.filled = false; // Ensure filled is false when stroke is set return this; } /** * Sets the spacing between words in the text layer. * @param {number} [wordSpacing] - The spacing between words. * @returns {this} The current instance for chaining. */ setWordSpacing(wordSpacing) { this.props.wordSpacing = wordSpacing; return this; } /** * Sets the spacing between letters in the text layer. * @param {number} [letterSpacing] - The spacing between letters. * @returns {this} The current instance for chaining. */ setLetterSpacing(letterSpacing) { this.props.letterSpacing = letterSpacing; return this; } /** * Measures the dimensions of the text. * @param {SKRSContext2D} [ctx] - The canvas rendering context. * @param {Canvas | SvgCanvas} [canvas] - The canvas instance. * @returns {Object} The width and height of the text. */ measureText(ctx, canvas) { const w = (0, utils_1.parseToNormal)(this.props.size?.width, ctx, canvas); const h = (0, utils_1.parseToNormal)(this.props.size?.height, ctx, canvas, { width: w, height: 0 }, { vertical: true }); if (this.props.multiline.enabled) { return { width: w, height: h }; } else { ctx.font = `${this.props.font.weight} ${this.props.font.size}px ${this.props.font.family}`; let data = ctx.measureText(this.props.text); return { width: data.width, height: this.props.font.size }; } } /** * Draws the text layer on the canvas. * @param {SKRSContext2D} [ctx] - The canvas rendering context. * @param {Canvas | SvgCanvas} [canvas] - The canvas instance. * @param {LayersManager} [manager] - The layer's manager. * @param {boolean} [debug] - Whether to enable debug logging. */ async draw(ctx, canvas, manager, debug) { const parcer = (0, utils_1.parser)(ctx, canvas, manager); const { x, y, w } = parcer.parseBatch({ x: { v: this.props.x }, y: { v: this.props.y, options: LazyUtil_1.defaultArg.vl(true) }, w: { v: this.props.size?.width }, }); const h = parcer.parse(this.props.size?.height, LazyUtil_1.defaultArg.wh(w), LazyUtil_1.defaultArg.vl(true)); if (debug) LazyUtil_1.LazyLog.log('none', `TextLayer:`, { x, y, w, h }); ctx.save(); (0, utils_1.transform)(ctx, this.props.transform, { width: w, height: h, x, y, type: this.type }, { text: this.props.text, textAlign: this.props.align, fontSize: this.props.font.size, multiline: this.props.multiline.enabled }); ctx.beginPath(); (0, utils_1.drawShadow)(ctx, this.props.shadow); (0, utils_1.opacity)(ctx, this.props.opacity); (0, utils_1.filters)(ctx, this.props.filter); ctx.textAlign = this.props.align; if (this.props.letterSpacing) ctx.letterSpacing = `${this.props.letterSpacing}px`; if (this.props.wordSpacing) ctx.wordSpacing = `${this.props.wordSpacing}px`; if (this.props.baseline) ctx.textBaseline = this.props.baseline; if (this.props.direction) ctx.direction = this.props.direction; let fillStyle = await (0, utils_1.parseFillStyle)(ctx, this.props.fillStyle, { debug, layer: { width: w, height: h, x, y, align: 'center' }, manager }); if (this.props.multiline.enabled) { const words = this.props.text.split(' '); let lines = []; for (let fontSize = 1; fontSize <= this.props.font.size; fontSize++) { let lineHeight = fontSize * (this.props.multiline.spacing || 1.1); ctx.font = `${this.props.font.weight} ${fontSize}px ${this.props.font.family}`; let xm = x; let ym = y; lines = []; let line = ''; let charOffset = 0; // Track position in original text for (let word of words) { let linePlus = line + word + ' '; if (ctx.measureText(linePlus).width > w) { lines.push({ text: line, x: xm, y: ym, startOffset: charOffset }); charOffset += line.length; line = word + ' '; ym += lineHeight; } else { line = linePlus; } } lines.push({ text: line, x: xm, y: ym, startOffset: charOffset }); if (ym > ym + h) break; } for (let line of lines) { this.drawText(this.props, ctx, fillStyle, line.text, line.x, line.y, w, line.startOffset); } } else { ctx.font = `${this.props.font.weight} ${this.props.font.size}px ${this.props.font.family}`; this.drawText(this.props, ctx, fillStyle, this.props.text, x, y, w, 0); } ctx.closePath(); ctx.restore(); } /** * Draws the text on the canvas. * @param {ITextLayerProps} [props] - The properties of the text layer. * @param {SKRSContext2D} [ctx] - The canvas rendering context. * @param {string | CanvasGradient | CanvasPattern} [fillStyle] - The fill style for the text. * @param {string} [text] - The text content. * @param {number} [x] - The x-coordinate of the text. * @param {number} [y] - The y-coordinate of the text. * @param {number} [w] - The width of the text area. * @param {number} [textOffset] - The offset of this text segment in the original full text (for multiline support). */ drawText(props, ctx, fillStyle, text, x, y, w, textOffset = 0) { // If no substring colors are defined, draw normally if (!props.subStringColors || props.subStringColors.length === 0) { if (props.filled) { ctx.fillStyle = fillStyle; ctx.fillText(text, x, y, w); } else { ctx.strokeStyle = fillStyle; ctx.lineWidth = props.stroke?.width || 1; ctx.lineCap = props.stroke?.cap || 'butt'; ctx.lineJoin = props.stroke?.join || 'miter'; ctx.miterLimit = props.stroke?.miterLimit || 10; ctx.lineDashOffset = props.stroke?.dashOffset || 0; ctx.setLineDash(props.stroke?.dash || []); ctx.strokeText(text, x, y, w); } return; } // Draw text with substring colors const textLength = text.length; let currentX = x; // Save original text alignment and set to left for manual positioning const originalAlign = ctx.textAlign; ctx.textAlign = 'left'; // Adjust starting X based on text alignment const alignValue = props.align; if (alignValue === types_1.TextAlign.Center || alignValue === 'center') { const totalWidth = ctx.measureText(text).width; currentX = x - totalWidth / 2; } else if (alignValue === types_1.TextAlign.Right || alignValue === 'right' || alignValue === types_1.TextAlign.End || alignValue === 'end') { const totalWidth = ctx.measureText(text).width; currentX = x - totalWidth; } // Create segments based on substring colors const segments = []; // Sort substring colors by start position const sortedColors = [...props.subStringColors].sort((a, b) => a.start - b.start); let currentIndex = 0; for (const subColor of sortedColors) { // Adjust positions based on textOffset (for multiline support) const globalStart = subColor.start; const globalEnd = subColor.end; const lineStart = textOffset; const lineEnd = textOffset + textLength; // Skip if this color segment doesn't overlap with current line if (globalEnd <= lineStart || globalStart >= lineEnd) { continue; } // Calculate local positions within this line const localStart = Math.max(0, globalStart - lineStart); const localEnd = Math.min(textLength, globalEnd - lineStart); // Add base color segment before this substring color if (currentIndex < localStart) { segments.push({ text: text.substring(currentIndex, localStart), color: fillStyle, start: currentIndex, end: localStart }); } // Add colored substring if (localStart < localEnd) { segments.push({ text: text.substring(localStart, localEnd), color: subColor.color, start: localStart, end: localEnd }); currentIndex = localEnd; } } // Add remaining text with base color if (currentIndex < textLength) { segments.push({ text: text.substring(currentIndex), color: fillStyle, start: currentIndex, end: textLength }); } // Draw each segment for (const segment of segments) { if (segment.text.length === 0) continue; const segmentWidth = ctx.measureText(segment.text).width; if (props.filled) { ctx.fillStyle = segment.color; ctx.fillText(segment.text, currentX, y); } else { ctx.strokeStyle = segment.color; ctx.lineWidth = props.stroke?.width || 1; ctx.lineCap = props.stroke?.cap || 'butt'; ctx.lineJoin = props.stroke?.join || 'miter'; ctx.miterLimit = props.stroke?.miterLimit || 10; ctx.lineDashOffset = props.stroke?.dashOffset || 0; ctx.setLineDash(props.stroke?.dash || []); ctx.strokeText(segment.text, currentX, y); } currentX += segmentWidth; } // Restore original text alignment ctx.textAlign = originalAlign; } /** * Converts the Text layer to a JSON representation. * @returns {ITextLayer} The JSON representation of the Text layer. */ toJSON() { let data = super.toJSON(); let copy = { ...this.props }; for (const key of ['x', 'y', 'size.width', 'size.height', 'fillStyle']) { if (copy[key] && typeof copy[key] === 'object' && 'toJSON' in copy[key]) { copy[key] = copy[key].toJSON(); } } return { ...data, props: copy }; } /** * Validates the properties of the Text layer. * @param {ITextLayerProps} [data] - The properties to validate. * @returns {ITextLayerProps} The validated properties. */ validateProps(data) { return { ...super.validateProps(data), filled: data.filled || true, fillStyle: data.fillStyle || '#000000', text: data.text || "", font: { family: data.font?.family || "Arial", size: data.font?.size || 16, weight: data.font?.weight || types_1.FontWeight.Regular, }, multiline: { enabled: data.multiline?.enabled || false, spacing: data.multiline?.spacing || 1.1, }, size: { width: data.size?.width || "vw", height: data.size?.height || 0, }, align: data.align || types_1.TextAlign.Left, }; } } exports.TextLayer = TextLayer;