From be9fa9564aaf46c48aee4fa172142c297a980057 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Mon, 30 Oct 2023 18:45:48 -0400 Subject: [PATCH 01/10] update Label class with responsive styling --- examples/native/src/simple/index.ts | 5 +- src/index.ts | 6 +- src/renderers/webgl/node.ts | 32 ++-- src/renderers/webgl/objects/label.ts | 158 ------------------ src/renderers/webgl/objects/label/index.ts | 185 +++++++++++++++++++++ src/renderers/webgl/objects/label/utils.ts | 135 +++++++++++++++ src/renderers/webgl/utils.ts | 43 ----- 7 files changed, 340 insertions(+), 224 deletions(-) delete mode 100644 src/renderers/webgl/objects/label.ts create mode 100644 src/renderers/webgl/objects/label/index.ts create mode 100644 src/renderers/webgl/objects/label/utils.ts diff --git a/examples/native/src/simple/index.ts b/examples/native/src/simple/index.ts index b98abe1a..f8950a08 100644 --- a/examples/native/src/simple/index.ts +++ b/examples/native/src/simple/index.ts @@ -27,15 +27,14 @@ const NODE_STYLE: Graph.NodeStyle = { stroke: [{ width: 2, color: LIGHT_PURPLE }], icon: { type: 'textIcon', text: 'T', family: 'sans-serif', size: 14, color: '#fff', weight: '400' }, label: { - position: 'bottom', - color: LIGHT_PURPLE + position: 'bottom' } } const NODE_HOVER_STYLE: Graph.NodeStyle = { color: '#f66', stroke: [{ width: 2, color: '#fcc' }], - label: { position: 'bottom' }, + label: { position: 'bottom', color: '#fcc' }, icon: { type: 'textIcon', text: 'L', family: 'sans-serif', size: 14, color: '#fff', weight: '400' } } diff --git a/src/index.ts b/src/index.ts index 1adb9aa5..a35118c1 100644 --- a/src/index.ts +++ b/src/index.ts @@ -45,18 +45,16 @@ export type ImageIcon = { export type Stroke = { color: string; width: number } -export type LabelBackground = { color: string; opacity?: number } - export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' export type LabelStyle = Partial<{ color: string - fontFamily: string + fontFamily: string | string[] fontSize: number maxWidth: number stroke: Stroke position: LabelPosition - background: LabelBackground + background: string }> export type NodeStyle = { diff --git a/src/renderers/webgl/node.ts b/src/renderers/webgl/node.ts index 7ea9e17e..82b95bb6 100644 --- a/src/renderers/webgl/node.ts +++ b/src/renderers/webgl/node.ts @@ -2,7 +2,6 @@ import { FederatedPointerEvent } from 'pixi.js' import { MIN_LABEL_ZOOM, MIN_INTERACTION_ZOOM, MIN_NODE_STROKE_ZOOM, Renderer, MIN_NODE_ICON_ZOOM } from '.' import * as Graph from '../..' import { Label } from './objects/label' -import { positionNodeLabel } from './utils' import { NodeFill } from './objects/nodeFill' import { NodeStrokes } from './objects/nodeStrokes' import { Icon } from './objects/icon' @@ -43,17 +42,7 @@ export class NodeRenderer { } update(node: Graph.Node) { - if (this.label === undefined) { - if (node.label) { - this.label = new Label(this.renderer.labelsContainer, node.label) - } - } else { - if (node.label === undefined) { - this.renderer.labelObjectManager.delete(this.label) - this.labelMounted = false - this.label = undefined - } - } + this.setLabel(node) if (this.icon === undefined) { if (node.style?.icon) { @@ -502,10 +491,7 @@ export class NodeRenderer { this.fill.update(this.x, this.y, radius, node.style) this.strokes.update(this.x, this.y, radius, node.style) - if (this.label && node.label) { - const labelPosition = positionNodeLabel(this.x, this.y, node.label, this.strokes.radius, node.style?.label?.position) - this.label.update(node.label, labelPosition[0], labelPosition[1], node.style?.label) - } + this.setLabel(node) if (this.icon && node.style?.icon) { this.icon.update(this.x, this.y, node.style.icon) } @@ -521,4 +507,18 @@ export class NodeRenderer { this.y - this.strokes.radius <= this.renderer.maxY ) } + + private setLabel(node: Graph.Node) { + if (this.label === undefined) { + if (node.label !== undefined) { + this.label = new Label(this.renderer.labelsContainer, node.label, node.style?.label) + } + } else if (node.label === undefined) { + this.renderer.labelObjectManager.delete(this.label) + this.labelMounted = false + this.label = undefined + } else { + this.label.update(node.label, { x: this.x, y: this.y, offset: this.strokes.radius }, node.style?.label) + } + } } diff --git a/src/renderers/webgl/objects/label.ts b/src/renderers/webgl/objects/label.ts deleted file mode 100644 index dcc1e882..00000000 --- a/src/renderers/webgl/objects/label.ts +++ /dev/null @@ -1,158 +0,0 @@ -import { BitmapText, Container, Text, TextStyle, TextStyleAlign } from 'pixi.js' -import { isASCII } from '../utils' -import { LabelPosition, LabelStyle } from '../../..' - -const DEFAULT_FONT_SIZE = 10 -const DEFAULT_ORIENTATION = 'bottom' -const TEXT_OUTLINE_STYLE: Partial = { lineJoin: 'round', stroke: '#fff', strokeThickness: 2 } - -/** - * TODO - * - add support for background color, font color, font family - * - moving/scaling labels is slow. render ASCII text characters as sprites to partical container? - */ -export class Label { - mounted = false - - private container: Container - private text: BitmapText | Text - private label: string - private x?: number - private y?: number - private fontSize?: number - private position?: LabelPosition - - constructor(container: Container, label: string) { - this.container = container - this.label = label - this.text = isASCII(label) ? new BitmapText(label, { fontName: 'Label' }) : new Text(label, TEXT_OUTLINE_STYLE) - } - - update(label: string, x: number, y: number, style?: LabelStyle) { - if (label !== this.label) { - this.setText(label) - this.label = label - } - - if (x !== this.x) { - this.text.x = x - this.x = x - } - - if (y !== this.y) { - this.text.y = y - this.y = y - } - - const fontSize = style?.fontSize ?? DEFAULT_FONT_SIZE - - if (fontSize !== this.fontSize) { - this.setFontSize(fontSize) - this.fontSize = fontSize - } - - const position = style?.position ?? DEFAULT_ORIENTATION - - if (position !== this.position) { - switch (style?.position ?? DEFAULT_ORIENTATION) { - case 'bottom': - this.setAlign('center') - this.text.anchor.set(0.5, 0) - break - case 'left': - this.setAlign('left') - this.text.anchor.set(1, 0.5) - break - case 'top': - this.setAlign('center') - this.text.anchor.set(0.5, 1) - break - case 'right': - this.setAlign('right') - this.text.anchor.set(0, 0.5) - break - } - - this.position = position - } - - return this - } - - mount() { - if (!this.mounted) { - this.container.addChild(this.text) - this.mounted = true - } - - return this - } - - unmount() { - if (this.mounted) { - this.container.removeChild(this.text) - this.mounted = false - } - - return this - } - - delete() { - this.unmount() - this.text.destroy() - - return undefined - } - - private setText(label: string) { - if (this.text instanceof BitmapText) { - if (isASCII(label)) { - this.text.text = label - } else { - const isMounted = this.mounted - this.delete() - this.text = new Text(label, TEXT_OUTLINE_STYLE) - this.fontSize = undefined - this.position = undefined - this.x = undefined - this.y = undefined - if (isMounted) { - this.mount() - } - } - } else { - if (isASCII(label)) { - const isMounted = this.mounted - this.delete() - this.text = new BitmapText(label, { fontName: 'Label' }) - this.fontSize = undefined - this.position = undefined - this.x = undefined - this.y = undefined - if (isMounted) { - this.mount() - } - } else { - this.text.text = label - } - } - - this.label = label - } - - private setFontSize(fontSize: number) { - if (this.text instanceof BitmapText) { - this.text.fontSize = fontSize - } else { - this.text.style.fontSize = fontSize - } - } - - private setAlign(align: TextStyleAlign) { - if (this.text instanceof BitmapText) { - this.text.align = align - } else { - this.text.style.align = align - } - } -} diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts new file mode 100644 index 00000000..f4247379 --- /dev/null +++ b/src/renderers/webgl/objects/label/index.ts @@ -0,0 +1,185 @@ +import { BitmapText, Container, Text, TextStyleAlign } from 'pixi.js' +import { LabelPosition, LabelStyle, Stroke, equals } from '../../../..' +import { StyleWithDefaults, LabelAnchor } from './utils' +import * as utils from './utils' + +/** + * TODO + * - add support for background color + * - moving/scaling labels is slow. render ASCII text characters as sprites to partical container? + */ +export class Label { + mounted = false + + private container: Container + private text: BitmapText | Text + private label: string + private x?: number + private y?: number + private style: StyleWithDefaults + + constructor(container: Container, label: string, style?: LabelStyle) { + this.container = container + this.label = label + this.style = utils.mergeDefaults(style) + if (utils.renderAsBitmapText(label, this.style)) { + this.text = new BitmapText(label, utils.getBitmapStyle(this.style)) + } else { + this.text = new Text(label, utils.getTextStyle(this.style)) + } + this.position = this.style.position + } + + update(label: string, anchor: LabelAnchor, style?: LabelStyle) { + this.value = label + + if (this.text instanceof BitmapText ? !utils.renderAsBitmapText(label, style) : utils.renderAsBitmapText(label, style)) { + this.transformText() + } + + this.coordinates = anchor + this.color = style?.color + this.maxWidth = style?.maxWidth + this.fontFamily = style?.fontFamily + this.position = style?.position ?? utils.DEFAULT_ORIENTATION + this.fontSize = style?.fontSize ?? utils.DEFAULT_FONT_SIZE + this.stroke = style?.stroke ?? utils.DEFAULT_STROKE + return this + } + + mount() { + if (!this.mounted) { + this.container.addChild(this.text) + this.mounted = true + } + + return this + } + + unmount() { + if (this.mounted) { + this.container.removeChild(this.text) + this.mounted = false + } + + return this + } + + delete() { + this.unmount() + this.text.destroy() + + return undefined + } + + private transformText() { + if (this.text instanceof BitmapText) { + const isMounted = this.mounted + this.delete() + this.text = new Text(this.label, utils.getTextStyle(this.style)) + this.position = this.style.position + this.x = undefined + this.y = undefined + if (isMounted) { + this.mount() + } + } else { + const isMounted = this.mounted + this.delete() + this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) + this.position = this.style.position + this.x = undefined + this.y = undefined + if (isMounted) { + this.mount() + } + } + } + + private set value(label: string) { + if (label !== this.label) { + this.text.text = label + this.label = label + } + } + + private set coordinates(anchor: LabelAnchor) { + const [x, y] = utils.getLabelCoordinates(anchor, this.text instanceof BitmapText, this.style.position) + + if (x !== this.x) { + this.text.x = x + this.x = x + } + if (y !== this.y) { + this.text.y = y + this.y = y + } + } + + private set position(position: LabelPosition) { + if (position !== this.style.position) { + this.style.position = position + this.align = utils.getPositionAlign(position) + } + this.text.anchor.set(...utils.getPositionAnchor(position)) + } + + private set align(align: TextStyleAlign) { + if (this.text instanceof BitmapText) { + this.text.align = align + } else { + this.text.style.align = align + } + } + + private set fontSize(fontSize: number) { + if (fontSize !== this.style.fontSize) { + this.style.fontSize = fontSize + if (this.text instanceof BitmapText) { + this.text.fontSize = fontSize + } else { + this.text.style.fontSize = fontSize + } + } + } + + private set maxWidth(maxWidth: number | undefined) { + if (maxWidth !== this.style.maxWidth) { + this.style.maxWidth = maxWidth + if (this.text instanceof BitmapText) { + this.text.maxWidth = maxWidth ?? 0 + } else { + this.text.style.wordWrap = maxWidth !== undefined + this.text.style.wordWrapWidth = maxWidth ?? 0 + } + } + } + + private set color(color: string | undefined) { + if (color !== this.style.color) { + this.style.color = color + if (this.text instanceof Text) { + this.text.style.fill = color ?? utils.DEFAULT_COLOR + } + } + } + + private set stroke(stroke: Stroke) { + if (!equals(stroke, this.style.stroke)) { + this.style.stroke = stroke + if (this.text instanceof Text) { + this.text.style.stroke = stroke.color + this.text.style.strokeThickness = stroke.width + } + } + } + + private set fontFamily(fontFamily: string | string[] | undefined) { + if (!equals(fontFamily, this.style.fontFamily)) { + this.style.fontFamily = fontFamily + if (this.text instanceof Text) { + this.text.style.fontFamily = fontFamily ?? utils.DEFAULT_FONT_FAMILY + } + } + } +} diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts new file mode 100644 index 00000000..5cf8aee4 --- /dev/null +++ b/src/renderers/webgl/objects/label/utils.ts @@ -0,0 +1,135 @@ +import { Text, TextStyle, ITextStyle, IBitmapTextStyle, TextStyleAlign } from 'pixi.js' +import { LabelPosition, LabelStyle, Stroke } from '../../../..' + +export type LabelAnchor = { x: number; y: number; offset?: number } + +export type StyleWithDefaults = Omit & { + stroke: Stroke + fontSize: number + position: LabelPosition +} + +// defaults +export const DEFAULT_FONT_SIZE = 10 +export const DEFAULT_COLOR = '#000000' +export const DEFAULT_ORIENTATION = 'bottom' +export const DEFAULT_BITMAP_FONT = 'Label' +export const DEFAULT_FONT_FAMILY = ['Arial', 'sans-serif'] +export const DEFAULT_STROKE_COLOR = '#FFF' +export const DEFAULT_STROKE_WIDTH = 0 +export const DEFAULT_STROKE: Stroke = { color: '#FFF', width: 0 } +export const STYLE_DEFAULTS: StyleWithDefaults = { + position: DEFAULT_ORIENTATION, + fontSize: DEFAULT_FONT_SIZE, + stroke: DEFAULT_STROKE +} + +// install text defaults +Text.defaultResolution = 2 +Text.defaultAutoResolution = false +TextStyle.defaultStyle = { + ...TextStyle.defaultStyle, + lineJoin: 'round', + align: 'center', + wordWrap: false, + fill: DEFAULT_COLOR, + stroke: DEFAULT_STROKE.color, + fontSize: DEFAULT_FONT_SIZE, + fontFamily: DEFAULT_FONT_FAMILY +} + +// utils +export const mergeDefaults = (style: LabelStyle = {}): StyleWithDefaults => ({ + ...STYLE_DEFAULTS, + ...style +}) + +export const isASCII = (str: string) => { + for (const char of str) { + if (char.codePointAt(0)! > 126) { + return false + } + } + + return true +} + +export const renderAsBitmapText = (label: string, { color, fontFamily }: LabelStyle = {}) => + isASCII(label) && color === undefined && fontFamily === undefined + +export const getLabelCoordinates = ({ x, y, offset = 0 }: LabelAnchor, isBitmapText: boolean, position: LabelPosition) => { + if (isBitmapText) { + // BitmapText shifts text down 2px + switch (position) { + case 'left': + return [x - offset - 2, y - 2] + case 'top': + return [x, y - offset - 4] + case 'right': + return [x + offset + 2, y - 2] + default: + return [x, y + offset] + } + } else { + switch (position) { + case 'left': + return [x - offset - 2, y + 1] + case 'top': + return [x, y - offset + 2] + case 'right': + return [x + offset + 2, y + 1] + default: + return [x, y + offset] + } + } +} + +export const getPositionAlign = (position: LabelPosition): TextStyleAlign => { + return position === 'left' || position === 'right' ? position : 'center' +} + +export const getPositionAnchor = (position: LabelPosition): [x: number, y: number] => { + switch (position) { + case 'left': + return [1, 0.5] + case 'top': + return [0.5, 1] + case 'right': + return [0, 0.5] + default: + return [0.5, 0] + } +} + +export const getTextStyle = ({ color, fontFamily, fontSize, maxWidth, stroke, background, position }: StyleWithDefaults) => { + const style: Partial = {} + if (color !== undefined) { + style.fill = color + } + if (fontFamily !== undefined) { + style.fontFamily = fontFamily + } + if (fontSize !== undefined) { + style.fontSize = fontSize + } + if (maxWidth !== undefined) { + style.wordWrap = true + style.wordWrapWidth = maxWidth + } + if (stroke !== undefined) { + style.stroke = stroke.color + style.strokeThickness = stroke.width + } else if (background !== undefined) { + style.stroke = background + } + if (position !== undefined) { + style.align = getPositionAlign(position) + } + return new TextStyle(style) +} + +export const getBitmapStyle = (style: StyleWithDefaults): Partial => ({ + fontName: DEFAULT_BITMAP_FONT, + fontSize: style.fontSize, + align: getPositionAlign(style.position) +}) diff --git a/src/renderers/webgl/utils.ts b/src/renderers/webgl/utils.ts index 67d8e1df..4e609763 100644 --- a/src/renderers/webgl/utils.ts +++ b/src/renderers/webgl/utils.ts @@ -25,46 +25,3 @@ export const movePoint = (x: number, y: number, angle: number, distance: number) export const midPoint = (x0: number, y0: number, x1: number, y1: number) => [(x0 + x1) / 2, (y0 + y1) / 2] as const export const length = (x0: number, y0: number, x1: number, y1: number) => Math.hypot(x1 - x0, y1 - y0) - -export const positionNodeLabel = ( - x: number, - y: number, - label: string, - radius: number, - position: Graph.LabelPosition = 'bottom' -): [x: number, y: number] => { - if (isASCII(label)) { - // BitmapText shifts text down 2px - switch (position) { - case 'bottom': - return [x, y + radius] - case 'left': - return [x - radius - 2, y - 2] - case 'top': - return [x, y - radius - 4] - case 'right': - return [x + radius + 2, y - 2] - } - } else { - switch (position) { - case 'bottom': - return [x, y + radius] - case 'left': - return [x - radius - 2, y + 1] - case 'top': - return [x, y - radius + 2] - case 'right': - return [x + radius + 2, y + 1] - } - } -} - -export const isASCII = (str: string) => { - for (const char of str) { - if (char.codePointAt(0)! > 126) { - return false - } - } - - return true -} From 635bcc6078ea6787072c022ebfa2770268e6cf3b Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Mon, 30 Oct 2023 22:15:44 -0400 Subject: [PATCH 02/10] load bitmap fonts to enable dynamic styles --- examples/native/src/simple/index.ts | 6 +- src/index.ts | 1 + src/renderers/webgl/objects/label/index.ts | 83 ++++++++++++++++------ src/renderers/webgl/objects/label/utils.ts | 27 ++++--- 4 files changed, 86 insertions(+), 31 deletions(-) diff --git a/examples/native/src/simple/index.ts b/examples/native/src/simple/index.ts index f8950a08..be88be48 100644 --- a/examples/native/src/simple/index.ts +++ b/examples/native/src/simple/index.ts @@ -21,13 +21,17 @@ const sampleCoordinatePlane = function* (count: number, step: number, sample: nu const PURPLE = '#7A5DC5' const LIGHT_PURPLE = '#CAD' +const ARIAL_PINK = 'ArialPink' const NODE_STYLE: Graph.NodeStyle = { color: PURPLE, stroke: [{ width: 2, color: LIGHT_PURPLE }], icon: { type: 'textIcon', text: 'T', family: 'sans-serif', size: 14, color: '#fff', weight: '400' }, label: { - position: 'bottom' + position: 'bottom', + fontName: ARIAL_PINK, + fontFamily: ['Arial', 'sans-serif'], + color: LIGHT_PURPLE } } diff --git a/src/index.ts b/src/index.ts index a35118c1..a447877c 100644 --- a/src/index.ts +++ b/src/index.ts @@ -49,6 +49,7 @@ export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' export type LabelStyle = Partial<{ color: string + fontName: string fontFamily: string | string[] fontSize: number maxWidth: number diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts index f4247379..43e505e6 100644 --- a/src/renderers/webgl/objects/label/index.ts +++ b/src/renderers/webgl/objects/label/index.ts @@ -17,33 +17,51 @@ export class Label { private x?: number private y?: number private style: StyleWithDefaults + private dirty = false - constructor(container: Container, label: string, style?: LabelStyle) { + constructor(container: Container, label: string, labelStyle?: LabelStyle) { this.container = container this.label = label - this.style = utils.mergeDefaults(style) - if (utils.renderAsBitmapText(label, this.style)) { - this.text = new BitmapText(label, utils.getBitmapStyle(this.style)) + const style = utils.mergeDefaults(labelStyle) + const textStyle = utils.getTextStyle(style) + if (utils.isASCII(label)) { + utils.loadFont(style) + this.text = new BitmapText(label, utils.getBitmapStyle(style)) } else { - this.text = new Text(label, utils.getTextStyle(this.style)) + this.text = new Text(label, textStyle) } - this.position = this.style.position + this.style = style + this.text.anchor.set(...utils.getPositionAnchor(style.position)) } update(label: string, anchor: LabelAnchor, style?: LabelStyle) { this.value = label + const isBitmapText = this.isBitmapText() + const isASCII = utils.isASCII(label) - if (this.text instanceof BitmapText ? !utils.renderAsBitmapText(label, style) : utils.renderAsBitmapText(label, style)) { + if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { this.transformText() } this.coordinates = anchor - this.color = style?.color this.maxWidth = style?.maxWidth - this.fontFamily = style?.fontFamily + this.color = style?.color ?? utils.DEFAULT_COLOR this.position = style?.position ?? utils.DEFAULT_ORIENTATION this.fontSize = style?.fontSize ?? utils.DEFAULT_FONT_SIZE this.stroke = style?.stroke ?? utils.DEFAULT_STROKE + this.fontFamily = style?.fontFamily ?? utils.DEFAULT_FONT_FAMILY + this.fontName = style?.fontName ?? utils.DEFAULT_FONT_NAME + + if (this.dirty) { + this.dirty = false + if (this.isBitmapText(this.text)) { + utils.loadFont(this.style) + this.text.updateText() + } else { + this.text.updateText(true) + } + } + return this } @@ -72,8 +90,12 @@ export class Label { return undefined } + private isBitmapText(text: Text | BitmapText = this.text): text is BitmapText { + return text instanceof BitmapText + } + private transformText() { - if (this.text instanceof BitmapText) { + if (this.isBitmapText()) { const isMounted = this.mounted this.delete() this.text = new Text(this.label, utils.getTextStyle(this.style)) @@ -86,6 +108,7 @@ export class Label { } else { const isMounted = this.mounted this.delete() + utils.loadFont(this.style) this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) this.position = this.style.position this.x = undefined @@ -117,15 +140,16 @@ export class Label { } private set position(position: LabelPosition) { + this.align = utils.getPositionAlign(position) + this.text.anchor.set(...utils.getPositionAnchor(position)) if (position !== this.style.position) { + this.dirty = true this.style.position = position - this.align = utils.getPositionAlign(position) } - this.text.anchor.set(...utils.getPositionAnchor(position)) } private set align(align: TextStyleAlign) { - if (this.text instanceof BitmapText) { + if (this.isBitmapText(this.text)) { this.text.align = align } else { this.text.style.align = align @@ -134,8 +158,9 @@ export class Label { private set fontSize(fontSize: number) { if (fontSize !== this.style.fontSize) { + this.dirty = true this.style.fontSize = fontSize - if (this.text instanceof BitmapText) { + if (this.isBitmapText(this.text)) { this.text.fontSize = fontSize } else { this.text.style.fontSize = fontSize @@ -145,8 +170,9 @@ export class Label { private set maxWidth(maxWidth: number | undefined) { if (maxWidth !== this.style.maxWidth) { + this.dirty = true this.style.maxWidth = maxWidth - if (this.text instanceof BitmapText) { + if (this.isBitmapText(this.text)) { this.text.maxWidth = maxWidth ?? 0 } else { this.text.style.wordWrap = maxWidth !== undefined @@ -155,11 +181,12 @@ export class Label { } } - private set color(color: string | undefined) { + private set color(color: string) { if (color !== this.style.color) { this.style.color = color - if (this.text instanceof Text) { - this.text.style.fill = color ?? utils.DEFAULT_COLOR + if (!this.isBitmapText(this.text)) { + this.dirty = true + this.text.style.fill = color } } } @@ -167,18 +194,30 @@ export class Label { private set stroke(stroke: Stroke) { if (!equals(stroke, this.style.stroke)) { this.style.stroke = stroke - if (this.text instanceof Text) { + if (!this.isBitmapText(this.text)) { + this.dirty = true this.text.style.stroke = stroke.color this.text.style.strokeThickness = stroke.width } } } - private set fontFamily(fontFamily: string | string[] | undefined) { + private set fontFamily(fontFamily: string | string[]) { if (!equals(fontFamily, this.style.fontFamily)) { this.style.fontFamily = fontFamily - if (this.text instanceof Text) { - this.text.style.fontFamily = fontFamily ?? utils.DEFAULT_FONT_FAMILY + if (!this.isBitmapText(this.text)) { + this.dirty = true + this.text.style.fontFamily = fontFamily + } + } + } + + private set fontName(fontName: string) { + if (fontName !== this.style.fontName) { + this.style.fontName = fontName + if (this.isBitmapText(this.text)) { + this.dirty = true + this.text.fontName = fontName } } } diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index 5cf8aee4..c7bd8114 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,19 +1,21 @@ -import { Text, TextStyle, ITextStyle, IBitmapTextStyle, TextStyleAlign } from 'pixi.js' +import { Text, TextStyle, ITextStyle, IBitmapTextStyle, TextStyleAlign, BitmapFont } from 'pixi.js' import { LabelPosition, LabelStyle, Stroke } from '../../../..' export type LabelAnchor = { x: number; y: number; offset?: number } -export type StyleWithDefaults = Omit & { +export type StyleWithDefaults = Omit & { stroke: Stroke fontSize: number position: LabelPosition + fontFamily: string | string[] + fontName: string } // defaults export const DEFAULT_FONT_SIZE = 10 export const DEFAULT_COLOR = '#000000' export const DEFAULT_ORIENTATION = 'bottom' -export const DEFAULT_BITMAP_FONT = 'Label' +export const DEFAULT_FONT_NAME = 'Label' export const DEFAULT_FONT_FAMILY = ['Arial', 'sans-serif'] export const DEFAULT_STROKE_COLOR = '#FFF' export const DEFAULT_STROKE_WIDTH = 0 @@ -21,7 +23,9 @@ export const DEFAULT_STROKE: Stroke = { color: '#FFF', width: 0 } export const STYLE_DEFAULTS: StyleWithDefaults = { position: DEFAULT_ORIENTATION, fontSize: DEFAULT_FONT_SIZE, - stroke: DEFAULT_STROKE + stroke: DEFAULT_STROKE, + fontFamily: DEFAULT_FONT_FAMILY, + fontName: DEFAULT_FONT_NAME } // install text defaults @@ -54,9 +58,6 @@ export const isASCII = (str: string) => { return true } -export const renderAsBitmapText = (label: string, { color, fontFamily }: LabelStyle = {}) => - isASCII(label) && color === undefined && fontFamily === undefined - export const getLabelCoordinates = ({ x, y, offset = 0 }: LabelAnchor, isBitmapText: boolean, position: LabelPosition) => { if (isBitmapText) { // BitmapText shifts text down 2px @@ -129,7 +130,17 @@ export const getTextStyle = ({ color, fontFamily, fontSize, maxWidth, stroke, ba } export const getBitmapStyle = (style: StyleWithDefaults): Partial => ({ - fontName: DEFAULT_BITMAP_FONT, + fontName: style.fontName, fontSize: style.fontSize, align: getPositionAlign(style.position) }) + +export const bitmapFontIsAvailable = (fontName: string) => { + return BitmapFont.available[fontName] !== undefined +} + +export const loadFont = (style: StyleWithDefaults) => { + if (!bitmapFontIsAvailable(style.fontName)) { + BitmapFont.from(style.fontName, getTextStyle(style)) + } +} From 10b73742ef865eba730508bc3c5eb510a8b9f0a6 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Tue, 31 Oct 2023 11:21:50 -0400 Subject: [PATCH 03/10] colocate Label types and utils + file cleanup --- src/index.ts | 21 +- src/renderers/webgl/objects/label/Label.ts | 245 +++++++++++++++++++++ src/renderers/webgl/objects/label/index.ts | 226 +------------------ src/renderers/webgl/objects/label/utils.ts | 131 ++++++----- src/types.ts | 5 + 5 files changed, 340 insertions(+), 288 deletions(-) create mode 100644 src/renderers/webgl/objects/label/Label.ts diff --git a/src/index.ts b/src/index.ts index a447877c..62207e2d 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,4 +1,6 @@ import { TWO_PI } from './renderers/webgl/utils' +import type { LabelStyle } from './renderers/webgl/objects/label' +import type { Stroke } from './types' export type Node = { id: string @@ -43,21 +45,6 @@ export type ImageIcon = { offsetY?: number } -export type Stroke = { color: string; width: number } - -export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' - -export type LabelStyle = Partial<{ - color: string - fontName: string - fontFamily: string | string[] - fontSize: number - maxWidth: number - stroke: Stroke - position: LabelPosition - background: string -}> - export type NodeStyle = { color?: string icon?: TextIcon | ImageIcon @@ -391,3 +378,7 @@ export const angle = (x0: number, y0: number, x1: number, y1: number) => { const angle = Math.atan2(y0 - y1, x0 - x1) return angle < 0 ? angle + TWO_PI : angle } + +// exports +export type { Stroke } from './types' +export type { LabelStyle, LabelCoords, LabelPosition } from './renderers/webgl/objects/label' diff --git a/src/renderers/webgl/objects/label/Label.ts b/src/renderers/webgl/objects/label/Label.ts new file mode 100644 index 00000000..322b42ad --- /dev/null +++ b/src/renderers/webgl/objects/label/Label.ts @@ -0,0 +1,245 @@ +import type { StyleWithDefaults, LabelCoords, LabelPosition, LabelStyle } from './utils' +import type { Stroke } from '../../../../types' +import { BitmapText, Container, Text, TextStyleAlign } from 'pixi.js' +import { equals } from '../../../..' +import utils, { STYLE_DEFAULTS } from './utils' + +/** + * TODO + * - add support for background color, custom font loading + * - add support for loading custom fonts as asset bundles + * - moving/scaling labels is slow. render ASCII text characters as sprites to partical container? + */ +export class Label { + mounted = false + + private container: Container + private text: BitmapText | Text + private label: string + private x?: number + private y?: number + private style: StyleWithDefaults + private dirty = false + + constructor(container: Container, label: string, labelStyle?: LabelStyle) { + this.container = container + this.label = label + this.style = utils.mergeDefaults(labelStyle) + + if (utils.isASCII(label)) { + utils.loadFont(this.style) + this.text = new BitmapText(label, utils.getBitmapStyle(this.style)) + } else { + this.text = new Text(label, utils.getTextStyle(this.style)) + } + + this.position = this.style.position + } + + update(label: string, coords: LabelCoords, style?: LabelStyle) { + this.value = label + this.coordinates = coords + this.wordWrap = style?.wordWrap + this.color = style?.color + this.stroke = style?.stroke + this.position = style?.position ?? STYLE_DEFAULTS.POSITION + this.fontSize = style?.fontSize ?? STYLE_DEFAULTS.FONT_SIZE + this.fontFamily = style?.fontFamily ?? STYLE_DEFAULTS.FONT_FAMILY + this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME + + const isBitmapText = this.isBitmapText() + const isASCII = utils.isASCII(label) + + if (isASCII) { + // conditionally load font if BitmapFont is unavailable + utils.loadFont(this.style) + } + + if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { + // if the text type has changed, regenerate a new text object + this.dirty = false + this.transformText() + } + + if (this.dirty) { + this.dirty = false + this.updateText() + } + + return this + } + + mount() { + if (!this.mounted) { + this.container.addChild(this.text) + this.mounted = true + } + + return this + } + + unmount() { + if (this.mounted) { + this.container.removeChild(this.text) + this.mounted = false + } + + return this + } + + delete() { + this.unmount() + this.text.destroy() + + return undefined + } + + private isBitmapText(text: Text | BitmapText = this.text): text is BitmapText { + return text instanceof BitmapText + } + + private updateText() { + if (this.isBitmapText(this.text)) { + this.text.updateText() + } else { + this.text.updateText(true) + } + } + + private transformText() { + if (this.isBitmapText()) { + const isMounted = this.mounted + this.delete() + this.text = new Text(this.label, utils.getTextStyle(this.style)) + this.position = this.style.position + this.x = undefined + this.y = undefined + if (isMounted) { + this.mount() + } + } else { + const isMounted = this.mounted + this.delete() + this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) + this.position = this.style.position + this.x = undefined + this.y = undefined + if (isMounted) { + this.mount() + } + } + } + + private set value(label: string) { + if (label !== this.label) { + this.text.text = label + this.label = label + } + } + + private set coordinates(coords: LabelCoords) { + const [x, y] = utils.getLabelCoordinates(coords, this.text instanceof BitmapText, this.style.position) + + if (x !== this.x) { + this.text.x = x + this.x = x + } + if (y !== this.y) { + this.text.y = y + this.y = y + } + } + + private set position(position: LabelPosition) { + this.align = utils.getPositionAlign(position) + this.anchor = utils.getPositionAnchor(position) + if (position !== this.style.position) { + this.dirty = true + this.style.position = position + } + } + + private set align(align: TextStyleAlign) { + if (this.isBitmapText(this.text)) { + if (this.text.align !== align) { + this.dirty = true + this.text.align = align + } + } else { + if (this.text.style.align !== align) { + this.dirty = true + this.text.style.align = align + } + } + } + + private set anchor([x, y]: [x: number, y: number]) { + if (!this.text.anchor.equals({ x, y })) { + this.text.anchor.set(x, y) + } + } + + private set fontSize(fontSize: number) { + if (fontSize !== this.style.fontSize) { + this.dirty = true + this.style.fontSize = fontSize + if (this.isBitmapText(this.text)) { + this.text.fontSize = fontSize + } else { + this.text.style.fontSize = fontSize + } + } + } + + private set wordWrap(wordWrap: number | undefined) { + if (wordWrap !== this.style.wordWrap) { + this.style.wordWrap = wordWrap + if (!this.isBitmapText(this.text)) { + this.dirty = true + this.text.style.wordWrap = wordWrap !== undefined + this.text.style.wordWrapWidth = wordWrap ?? 0 + } + } + } + + private set color(color: string | undefined) { + if (color !== this.style.color) { + this.style.color = color + if (!this.isBitmapText(this.text)) { + this.dirty = true + this.text.style.fill = color ?? STYLE_DEFAULTS.COLOR + } + } + } + + private set stroke(stroke: Stroke | undefined) { + if (!equals(stroke, this.style.stroke)) { + this.style.stroke = stroke + if (!this.isBitmapText(this.text)) { + this.dirty = true + this.text.style.stroke = stroke?.color ?? STYLE_DEFAULTS.STROKE + this.text.style.strokeThickness = stroke?.width ?? STYLE_DEFAULTS.STROKE_THICKNESS + } + } + } + + private set fontFamily(fontFamily: string | string[]) { + if (!equals(fontFamily, this.style.fontFamily)) { + this.dirty = true + this.style.fontFamily = fontFamily + if (!this.isBitmapText(this.text)) { + this.text.style.fontFamily = fontFamily + } + } + } + + private set fontName(fontName: string) { + if (fontName !== this.style.fontName) { + this.style.fontName = fontName + if (this.isBitmapText(this.text)) { + this.dirty = true + this.text.fontName = fontName + } + } + } +} diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts index 43e505e6..eae3ee68 100644 --- a/src/renderers/webgl/objects/label/index.ts +++ b/src/renderers/webgl/objects/label/index.ts @@ -1,224 +1,2 @@ -import { BitmapText, Container, Text, TextStyleAlign } from 'pixi.js' -import { LabelPosition, LabelStyle, Stroke, equals } from '../../../..' -import { StyleWithDefaults, LabelAnchor } from './utils' -import * as utils from './utils' - -/** - * TODO - * - add support for background color - * - moving/scaling labels is slow. render ASCII text characters as sprites to partical container? - */ -export class Label { - mounted = false - - private container: Container - private text: BitmapText | Text - private label: string - private x?: number - private y?: number - private style: StyleWithDefaults - private dirty = false - - constructor(container: Container, label: string, labelStyle?: LabelStyle) { - this.container = container - this.label = label - const style = utils.mergeDefaults(labelStyle) - const textStyle = utils.getTextStyle(style) - if (utils.isASCII(label)) { - utils.loadFont(style) - this.text = new BitmapText(label, utils.getBitmapStyle(style)) - } else { - this.text = new Text(label, textStyle) - } - this.style = style - this.text.anchor.set(...utils.getPositionAnchor(style.position)) - } - - update(label: string, anchor: LabelAnchor, style?: LabelStyle) { - this.value = label - const isBitmapText = this.isBitmapText() - const isASCII = utils.isASCII(label) - - if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { - this.transformText() - } - - this.coordinates = anchor - this.maxWidth = style?.maxWidth - this.color = style?.color ?? utils.DEFAULT_COLOR - this.position = style?.position ?? utils.DEFAULT_ORIENTATION - this.fontSize = style?.fontSize ?? utils.DEFAULT_FONT_SIZE - this.stroke = style?.stroke ?? utils.DEFAULT_STROKE - this.fontFamily = style?.fontFamily ?? utils.DEFAULT_FONT_FAMILY - this.fontName = style?.fontName ?? utils.DEFAULT_FONT_NAME - - if (this.dirty) { - this.dirty = false - if (this.isBitmapText(this.text)) { - utils.loadFont(this.style) - this.text.updateText() - } else { - this.text.updateText(true) - } - } - - return this - } - - mount() { - if (!this.mounted) { - this.container.addChild(this.text) - this.mounted = true - } - - return this - } - - unmount() { - if (this.mounted) { - this.container.removeChild(this.text) - this.mounted = false - } - - return this - } - - delete() { - this.unmount() - this.text.destroy() - - return undefined - } - - private isBitmapText(text: Text | BitmapText = this.text): text is BitmapText { - return text instanceof BitmapText - } - - private transformText() { - if (this.isBitmapText()) { - const isMounted = this.mounted - this.delete() - this.text = new Text(this.label, utils.getTextStyle(this.style)) - this.position = this.style.position - this.x = undefined - this.y = undefined - if (isMounted) { - this.mount() - } - } else { - const isMounted = this.mounted - this.delete() - utils.loadFont(this.style) - this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) - this.position = this.style.position - this.x = undefined - this.y = undefined - if (isMounted) { - this.mount() - } - } - } - - private set value(label: string) { - if (label !== this.label) { - this.text.text = label - this.label = label - } - } - - private set coordinates(anchor: LabelAnchor) { - const [x, y] = utils.getLabelCoordinates(anchor, this.text instanceof BitmapText, this.style.position) - - if (x !== this.x) { - this.text.x = x - this.x = x - } - if (y !== this.y) { - this.text.y = y - this.y = y - } - } - - private set position(position: LabelPosition) { - this.align = utils.getPositionAlign(position) - this.text.anchor.set(...utils.getPositionAnchor(position)) - if (position !== this.style.position) { - this.dirty = true - this.style.position = position - } - } - - private set align(align: TextStyleAlign) { - if (this.isBitmapText(this.text)) { - this.text.align = align - } else { - this.text.style.align = align - } - } - - private set fontSize(fontSize: number) { - if (fontSize !== this.style.fontSize) { - this.dirty = true - this.style.fontSize = fontSize - if (this.isBitmapText(this.text)) { - this.text.fontSize = fontSize - } else { - this.text.style.fontSize = fontSize - } - } - } - - private set maxWidth(maxWidth: number | undefined) { - if (maxWidth !== this.style.maxWidth) { - this.dirty = true - this.style.maxWidth = maxWidth - if (this.isBitmapText(this.text)) { - this.text.maxWidth = maxWidth ?? 0 - } else { - this.text.style.wordWrap = maxWidth !== undefined - this.text.style.wordWrapWidth = maxWidth ?? 0 - } - } - } - - private set color(color: string) { - if (color !== this.style.color) { - this.style.color = color - if (!this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.fill = color - } - } - } - - private set stroke(stroke: Stroke) { - if (!equals(stroke, this.style.stroke)) { - this.style.stroke = stroke - if (!this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.stroke = stroke.color - this.text.style.strokeThickness = stroke.width - } - } - } - - private set fontFamily(fontFamily: string | string[]) { - if (!equals(fontFamily, this.style.fontFamily)) { - this.style.fontFamily = fontFamily - if (!this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.fontFamily = fontFamily - } - } - } - - private set fontName(fontName: string) { - if (fontName !== this.style.fontName) { - this.style.fontName = fontName - if (this.isBitmapText(this.text)) { - this.dirty = true - this.text.fontName = fontName - } - } - } -} +export { Label } from './Label' +export type { LabelStyle, LabelCoords, LabelPosition } from './utils' diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index c7bd8114..da4642da 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,31 +1,42 @@ +import type { Stroke } from '../../../../types' import { Text, TextStyle, ITextStyle, IBitmapTextStyle, TextStyleAlign, BitmapFont } from 'pixi.js' -import { LabelPosition, LabelStyle, Stroke } from '../../../..' -export type LabelAnchor = { x: number; y: number; offset?: number } +export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' -export type StyleWithDefaults = Omit & { +export type LabelStyle = Partial<{ + color: string + fontName: string + fontSize: number + wordWrap: number + background: string + fontFamily: string | string[] + position: LabelPosition + letterSpacing: number stroke: Stroke +}> + +type _StyleDefaults = 'fontSize' | 'position' | 'fontFamily' | 'fontName' +export type StyleWithDefaults = Omit & { fontSize: number position: LabelPosition fontFamily: string | string[] fontName: string } -// defaults -export const DEFAULT_FONT_SIZE = 10 -export const DEFAULT_COLOR = '#000000' -export const DEFAULT_ORIENTATION = 'bottom' -export const DEFAULT_FONT_NAME = 'Label' -export const DEFAULT_FONT_FAMILY = ['Arial', 'sans-serif'] -export const DEFAULT_STROKE_COLOR = '#FFF' -export const DEFAULT_STROKE_WIDTH = 0 -export const DEFAULT_STROKE: Stroke = { color: '#FFF', width: 0 } -export const STYLE_DEFAULTS: StyleWithDefaults = { - position: DEFAULT_ORIENTATION, - fontSize: DEFAULT_FONT_SIZE, - stroke: DEFAULT_STROKE, - fontFamily: DEFAULT_FONT_FAMILY, - fontName: DEFAULT_FONT_NAME +export type LabelCoords = { x: number; y: number; offset?: number } + +export const STYLE_DEFAULTS = { + FONT_SIZE: 10, + LETTER_SPACING: 1, + STROKE_THICKNESS: 0, + WORD_WRAP: false, + STROKE: '#FFF', + FONT_NAME: 'Label', + COLOR: '#000000', + ALIGN: 'center' as const, + POSITION: 'bottom' as const, + LINE_JOIN: 'round' as const, + FONT_FAMILY: ['Arial', 'sans-serif'] } // install text defaults @@ -33,22 +44,33 @@ Text.defaultResolution = 2 Text.defaultAutoResolution = false TextStyle.defaultStyle = { ...TextStyle.defaultStyle, - lineJoin: 'round', - align: 'center', - wordWrap: false, - fill: DEFAULT_COLOR, - stroke: DEFAULT_STROKE.color, - fontSize: DEFAULT_FONT_SIZE, - fontFamily: DEFAULT_FONT_FAMILY + align: STYLE_DEFAULTS.ALIGN, + fill: STYLE_DEFAULTS.COLOR, + stroke: STYLE_DEFAULTS.STROKE, + lineJoin: STYLE_DEFAULTS.LINE_JOIN, + wordWrap: STYLE_DEFAULTS.WORD_WRAP, + fontSize: STYLE_DEFAULTS.FONT_SIZE, + fontFamily: STYLE_DEFAULTS.FONT_FAMILY, + letterSpacing: STYLE_DEFAULTS.LETTER_SPACING, + strokeThickness: STYLE_DEFAULTS.STROKE_THICKNESS } // utils -export const mergeDefaults = (style: LabelStyle = {}): StyleWithDefaults => ({ - ...STYLE_DEFAULTS, +const mergeDefaults = ({ + position = STYLE_DEFAULTS.POSITION, + fontSize = STYLE_DEFAULTS.FONT_SIZE, + fontFamily = STYLE_DEFAULTS.FONT_FAMILY, + fontName = STYLE_DEFAULTS.FONT_NAME, + ...style +}: LabelStyle = {}): StyleWithDefaults => ({ + position, + fontSize, + fontFamily, + fontName, ...style }) -export const isASCII = (str: string) => { +const isASCII = (str: string) => { for (const char of str) { if (char.codePointAt(0)! > 126) { return false @@ -58,51 +80,51 @@ export const isASCII = (str: string) => { return true } -export const getLabelCoordinates = ({ x, y, offset = 0 }: LabelAnchor, isBitmapText: boolean, position: LabelPosition) => { +const getLabelCoordinates = ({ x, y, offset = 0 }: LabelCoords, isBitmapText: boolean, position: LabelPosition) => { if (isBitmapText) { // BitmapText shifts text down 2px switch (position) { + case 'bottom': + return [x, y + offset] case 'left': return [x - offset - 2, y - 2] case 'top': return [x, y - offset - 4] case 'right': return [x + offset + 2, y - 2] - default: - return [x, y + offset] } } else { switch (position) { + case 'bottom': + return [x, y + offset] case 'left': return [x - offset - 2, y + 1] case 'top': return [x, y - offset + 2] case 'right': return [x + offset + 2, y + 1] - default: - return [x, y + offset] } } } -export const getPositionAlign = (position: LabelPosition): TextStyleAlign => { +const getPositionAlign = (position: LabelPosition): TextStyleAlign => { return position === 'left' || position === 'right' ? position : 'center' } -export const getPositionAnchor = (position: LabelPosition): [x: number, y: number] => { +const getPositionAnchor = (position: LabelPosition): [x: number, y: number] => { switch (position) { + case 'bottom': + return [0.5, 0] case 'left': return [1, 0.5] case 'top': return [0.5, 1] case 'right': return [0, 0.5] - default: - return [0.5, 0] } } -export const getTextStyle = ({ color, fontFamily, fontSize, maxWidth, stroke, background, position }: StyleWithDefaults) => { +const getTextStyle = ({ color, fontFamily, fontSize, wordWrap, stroke, position, letterSpacing }: StyleWithDefaults) => { const style: Partial = {} if (color !== undefined) { style.fill = color @@ -113,34 +135,45 @@ export const getTextStyle = ({ color, fontFamily, fontSize, maxWidth, stroke, ba if (fontSize !== undefined) { style.fontSize = fontSize } - if (maxWidth !== undefined) { + if (wordWrap !== undefined) { style.wordWrap = true - style.wordWrapWidth = maxWidth + style.wordWrapWidth = wordWrap } if (stroke !== undefined) { style.stroke = stroke.color style.strokeThickness = stroke.width - } else if (background !== undefined) { - style.stroke = background } - if (position !== undefined) { + if (position !== STYLE_DEFAULTS.POSITION) { style.align = getPositionAlign(position) } + if (letterSpacing !== undefined) { + style.letterSpacing = letterSpacing + } return new TextStyle(style) } -export const getBitmapStyle = (style: StyleWithDefaults): Partial => ({ +const getBitmapStyle = (style: StyleWithDefaults): Partial => ({ fontName: style.fontName, fontSize: style.fontSize, align: getPositionAlign(style.position) }) -export const bitmapFontIsAvailable = (fontName: string) => { - return BitmapFont.available[fontName] !== undefined -} +const bitmapFontIsAvailable = (fontName: string) => BitmapFont.available[fontName] !== undefined -export const loadFont = (style: StyleWithDefaults) => { +const loadFont = (style: StyleWithDefaults) => { if (!bitmapFontIsAvailable(style.fontName)) { - BitmapFont.from(style.fontName, getTextStyle(style)) + BitmapFont.from(style.fontName, getTextStyle(style), { resolution: 2 }) } } + +export default { + isASCII, + mergeDefaults, + getLabelCoordinates, + getPositionAlign, + getPositionAnchor, + getTextStyle, + getBitmapStyle, + bitmapFontIsAvailable, + loadFont +} diff --git a/src/types.ts b/src/types.ts index 6ea3977d..cbf92303 100644 --- a/src/types.ts +++ b/src/types.ts @@ -1,3 +1,8 @@ export type Extend = { [K in Exclude]: T[K] } & R + +export type Stroke = { + color: string + width: number +} From a5cc9e2345d13a2759e9c66e21c124da76928917 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Tue, 31 Oct 2023 15:05:45 -0400 Subject: [PATCH 04/10] enable custom label backgrounds with example --- examples/native/index.html | 5 +- examples/native/src/labels/index.ts | 120 +++++++++++++++ examples/native/src/labels/labels.html | 14 ++ examples/native/src/{simple => perf}/index.ts | 8 +- .../{simple/simple.html => perf/perf.html} | 2 +- src/index.ts | 2 +- src/renderers/webgl/objects/label/Label.ts | 79 ++++++++-- src/renderers/webgl/objects/label/index.ts | 2 +- src/renderers/webgl/objects/label/utils.ts | 145 +++++++++++++++--- 9 files changed, 334 insertions(+), 43 deletions(-) create mode 100644 examples/native/src/labels/index.ts create mode 100644 examples/native/src/labels/labels.html rename examples/native/src/{simple => perf}/index.ts (98%) rename examples/native/src/{simple/simple.html => perf/perf.html} (90%) diff --git a/examples/native/index.html b/examples/native/index.html index 35010e11..fae189c6 100644 --- a/examples/native/index.html +++ b/examples/native/index.html @@ -11,7 +11,10 @@

Native Examples

diff --git a/examples/native/src/labels/index.ts b/examples/native/src/labels/index.ts new file mode 100644 index 00000000..ce0cc733 --- /dev/null +++ b/examples/native/src/labels/index.ts @@ -0,0 +1,120 @@ +import * as Renderer from '@trellis/renderers/webgl' +import * as Graph from '@trellis/index' +import * as Collide from '@trellis/layout/collide' + +const GREEN = '#91AD49' +const GREEN_LIGHT = '#C6D336' +const DARK_GREEN = '#607330' + +const TEXT_ICON: Graph.TextIcon = { + type: 'textIcon', + family: 'sans-serif', + color: '#fff', + weight: '400', + text: '!', + size: 14 +} + +const NODE_STYLE: Graph.NodeStyle = { + color: GREEN, + icon: TEXT_ICON, + stroke: [{ width: 2, color: GREEN_LIGHT }], + label: { + position: 'top', + fontName: 'NodeLabel', + fontFamily: ['Arial', 'sans-serif'], + background: { color: GREEN_LIGHT }, + margin: 4 + } +} + +const NODE_HOVER_STYLE: Graph.NodeStyle = { + color: DARK_GREEN, + icon: TEXT_ICON, + stroke: [{ width: 2, color: GREEN_LIGHT }], + label: { + position: 'top', + fontName: 'NodeLabelHover', + fontFamily: ['Arial', 'sans-serif'], + background: { color: GREEN_LIGHT }, + color: DARK_GREEN, + margin: 4 + } +} + +const data = [ + 'Myriel', + 'Napoleon', + 'Mlle.Baptistine', + 'Mme.Magloire', + 'CountessdeLo', + 'Geborand', + 'Champtercier', + 'Cravatte', + 'Count', + 'OldMan', + 'Labarre', + 'Valjean' +] + +const collide = Collide.Layout() + +const edges: Graph.Edge[] = [] +let nodes = data.map((label, index) => ({ + radius: 10, + label: index % 2 === 0 ? label + ' 北京' : label, + id: `${index}-${label}`, + style: NODE_STYLE +})) + +const layout = collide({ nodes, edges, options: { nodePadding: 50 } }) +nodes = layout.nodes + +const size = { width: 1250, height: 650 } +const bounds = Graph.getSelectionBounds(nodes, 100) +const viewport = Graph.boundsToViewport(bounds, size) +const container = document.querySelector('#graph') as HTMLDivElement + +const options: Renderer.Options = { + ...viewport, + ...size, + minZoom: 0.025, + onViewportDrag: (event: Renderer.ViewportDragEvent | Renderer.ViewportDragDecelerateEvent) => { + // console.log('viewport drag', `x: ${event.dx}, y: ${event.dy}`) + options.x! += event.dx + options.y! += event.dy + renderer.update({ nodes, edges, options }) + }, + onViewportWheel: ({ dx, dy, dz }) => { + options.x! += dx + options.y! += dy + options.zoom! += dz + renderer.update({ nodes, edges, options }) + }, + onNodePointerEnter: (event: Renderer.NodePointerEvent) => { + // console.log('node pointer enter', `x: ${event.x}, y: ${event.y}`) + nodes = nodes.map((node) => (node.id === event.target.id ? { ...node, label: node.label + ' 北京', style: NODE_HOVER_STYLE } : node)) + renderer.update({ nodes, edges, options }) + }, + onNodeDrag: (event: Renderer.NodeDragEvent) => { + // console.log('node drag', `x: ${event.x}, y: ${event.y}`) + nodes = nodes.map((node) => + node.id === event.target.id ? { ...node, x: (node.x ?? 0) + event.dx, y: (node.y ?? 0) + event.dy } : node + ) + renderer.update({ nodes, edges, options }) + }, + onNodePointerLeave: (event: Renderer.NodePointerEvent) => { + // console.log('node pointer leave', `x: ${event.x}, y: ${event.y}`) + nodes = nodes.map((node) => + node.id === event.target.id ? { ...node, label: node.label?.slice(0, node.label.length - 3), style: NODE_STYLE } : node + ) + renderer.update({ nodes, edges, options }) + } +} + +const renderer = new Renderer.Renderer({ container, width: options.width, height: options.height, debug: true }).update({ + nodes, + edges, + options +}) +;(window as any).renderer = renderer diff --git a/examples/native/src/labels/labels.html b/examples/native/src/labels/labels.html new file mode 100644 index 00000000..822cc05c --- /dev/null +++ b/examples/native/src/labels/labels.html @@ -0,0 +1,14 @@ + + + + + + + + Trellis Example: Label Style + + +
+ + + diff --git a/examples/native/src/simple/index.ts b/examples/native/src/perf/index.ts similarity index 98% rename from examples/native/src/simple/index.ts rename to examples/native/src/perf/index.ts index be88be48..4d85199c 100644 --- a/examples/native/src/simple/index.ts +++ b/examples/native/src/perf/index.ts @@ -28,10 +28,14 @@ const NODE_STYLE: Graph.NodeStyle = { stroke: [{ width: 2, color: LIGHT_PURPLE }], icon: { type: 'textIcon', text: 'T', family: 'sans-serif', size: 14, color: '#fff', weight: '400' }, label: { - position: 'bottom', + position: 'top', fontName: ARIAL_PINK, fontFamily: ['Arial', 'sans-serif'], - color: LIGHT_PURPLE + margin: 2, + background: { + color: '#f66', + opacity: 0.5 + } } } diff --git a/examples/native/src/simple/simple.html b/examples/native/src/perf/perf.html similarity index 90% rename from examples/native/src/simple/simple.html rename to examples/native/src/perf/perf.html index 736ba429..ab99af39 100644 --- a/examples/native/src/simple/simple.html +++ b/examples/native/src/perf/perf.html @@ -5,7 +5,7 @@ - Simple Trellis Example + Trellis Example: Perf
diff --git a/src/index.ts b/src/index.ts index 62207e2d..332179cc 100644 --- a/src/index.ts +++ b/src/index.ts @@ -381,4 +381,4 @@ export const angle = (x0: number, y0: number, x1: number, y1: number) => { // exports export type { Stroke } from './types' -export type { LabelStyle, LabelCoords, LabelPosition } from './renderers/webgl/objects/label' +export type { LabelStyle, LabelBackgroundStyle, LabelPosition } from './renderers/webgl/objects/label' diff --git a/src/renderers/webgl/objects/label/Label.ts b/src/renderers/webgl/objects/label/Label.ts index 322b42ad..be0ed03a 100644 --- a/src/renderers/webgl/objects/label/Label.ts +++ b/src/renderers/webgl/objects/label/Label.ts @@ -1,6 +1,6 @@ -import type { StyleWithDefaults, LabelCoords, LabelPosition, LabelStyle } from './utils' +import type { StyleWithDefaults, LabelCoords, LabelPosition, LabelStyle, LabelBackgroundStyle } from './utils' import type { Stroke } from '../../../../types' -import { BitmapText, Container, Text, TextStyleAlign } from 'pixi.js' +import { BitmapText, Container, Sprite, Text, TextStyleAlign, TextStyleFill, TextStyleFontWeight } from 'pixi.js' import { equals } from '../../../..' import utils, { STYLE_DEFAULTS } from './utils' @@ -13,39 +13,40 @@ import utils, { STYLE_DEFAULTS } from './utils' export class Label { mounted = false + private dirty = false + private label: string private container: Container private text: BitmapText | Text - private label: string + private style: StyleWithDefaults + private backgroundSprite: Sprite | null = null private x?: number private y?: number - private style: StyleWithDefaults - private dirty = false constructor(container: Container, label: string, labelStyle?: LabelStyle) { + const style = utils.mergeDefaults(labelStyle) + const text = utils.createTextObject(label, style) this.container = container this.label = label - this.style = utils.mergeDefaults(labelStyle) - - if (utils.isASCII(label)) { - utils.loadFont(this.style) - this.text = new BitmapText(label, utils.getBitmapStyle(this.style)) - } else { - this.text = new Text(label, utils.getTextStyle(this.style)) + this.style = style + this.text = text + if (style.background !== undefined) { + this.backgroundSprite = utils.createBackgroundSprite(text, style.background) } - - this.position = this.style.position } update(label: string, coords: LabelCoords, style?: LabelStyle) { this.value = label + this.style.margin = style?.margin this.coordinates = coords this.wordWrap = style?.wordWrap this.color = style?.color this.stroke = style?.stroke + this.fontWeight = style?.fontWeight this.position = style?.position ?? STYLE_DEFAULTS.POSITION this.fontSize = style?.fontSize ?? STYLE_DEFAULTS.FONT_SIZE this.fontFamily = style?.fontFamily ?? STYLE_DEFAULTS.FONT_FAMILY this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME + this.background = style?.background const isBitmapText = this.isBitmapText() const isASCII = utils.isASCII(label) @@ -71,6 +72,10 @@ export class Label { mount() { if (!this.mounted) { + if (this.backgroundSprite) { + this.container.addChild(this.backgroundSprite) + } + this.container.addChild(this.text) this.mounted = true } @@ -80,6 +85,10 @@ export class Label { unmount() { if (this.mounted) { + if (this.backgroundSprite) { + this.container.removeChild(this.backgroundSprite) + } + this.container.removeChild(this.text) this.mounted = false } @@ -138,15 +147,22 @@ export class Label { } private set coordinates(coords: LabelCoords) { - const [x, y] = utils.getLabelCoordinates(coords, this.text instanceof BitmapText, this.style.position) + const isBitmapText = this.isBitmapText() + const [x, y] = utils.getLabelCoordinates(coords, isBitmapText, this.style) if (x !== this.x) { this.text.x = x this.x = x + if (this.backgroundSprite) { + this.backgroundSprite.x = utils.getBackgroundX(x, isBitmapText, this.style) + } } if (y !== this.y) { this.text.y = y this.y = y + if (this.backgroundSprite) { + this.backgroundSprite.y = utils.getBackgroundY(y, isBitmapText, this.style) + } } } @@ -202,8 +218,8 @@ export class Label { } } - private set color(color: string | undefined) { - if (color !== this.style.color) { + private set color(color: TextStyleFill | undefined) { + if (!equals(color, this.style.color)) { this.style.color = color if (!this.isBitmapText(this.text)) { this.dirty = true @@ -242,4 +258,33 @@ export class Label { } } } + + private set fontWeight(fontWeight: TextStyleFontWeight | undefined) { + if (fontWeight !== this.style.fontWeight) { + this.style.fontWeight = fontWeight + if (!this.isBitmapText(this.text)) { + this.dirty = true + this.text.style.fontWeight = fontWeight ?? STYLE_DEFAULTS.FONT_WEIGHT + } + } + } + + private set background(background: LabelBackgroundStyle | undefined) { + if (!equals(background, this.style.background)) { + this.style.background = background + if (this.backgroundSprite) { + if (background !== undefined) { + utils.setBackgroundStyle(this.backgroundSprite, this.text, background) + } else { + const sprite = this.backgroundSprite + this.backgroundSprite = null + this.mounted && this.container.removeChild(sprite) + sprite.destroy() + } + } else if (background !== undefined) { + this.backgroundSprite = utils.createBackgroundSprite(this.text, background) + this.mounted && this.container.addChild(this.backgroundSprite) + } + } + } } diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts index eae3ee68..19d944dc 100644 --- a/src/renderers/webgl/objects/label/index.ts +++ b/src/renderers/webgl/objects/label/index.ts @@ -1,2 +1,2 @@ export { Label } from './Label' -export type { LabelStyle, LabelCoords, LabelPosition } from './utils' +export type { LabelStyle, LabelBackgroundStyle, LabelPosition } from './utils' diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index da4642da..b1c696e8 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,18 +1,41 @@ import type { Stroke } from '../../../../types' -import { Text, TextStyle, ITextStyle, IBitmapTextStyle, TextStyleAlign, BitmapFont } from 'pixi.js' +import { + Text, + TextStyle, + TextStyleFill, + ITextStyle, + IBitmapTextStyle, + TextStyleAlign, + BitmapFont, + ColorSource, + Sprite, + Texture, + BitmapText, + TextStyleFontWeight +} from 'pixi.js' export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' +export type BackgroundPadding = number | [vertical: number, horizontal: number] + +export type LabelBackgroundStyle = { + color: ColorSource + opacity?: number + padding?: BackgroundPadding +} + export type LabelStyle = Partial<{ - color: string fontName: string fontSize: number wordWrap: number - background: string - fontFamily: string | string[] - position: LabelPosition + margin: number letterSpacing: number + fontFamily: string | string[] + fontWeight: TextStyleFontWeight stroke: Stroke + color: TextStyleFill + position: LabelPosition + background: LabelBackgroundStyle }> type _StyleDefaults = 'fontSize' | 'position' | 'fontFamily' | 'fontName' @@ -25,10 +48,11 @@ export type StyleWithDefaults = Omit & { export type LabelCoords = { x: number; y: number; offset?: number } +const RESOLUTION = 2 export const STYLE_DEFAULTS = { FONT_SIZE: 10, - LETTER_SPACING: 1, STROKE_THICKNESS: 0, + PADDING: [4, 8] as [number, number], WORD_WRAP: false, STROKE: '#FFF', FONT_NAME: 'Label', @@ -36,11 +60,12 @@ export const STYLE_DEFAULTS = { ALIGN: 'center' as const, POSITION: 'bottom' as const, LINE_JOIN: 'round' as const, + FONT_WEIGHT: 'normal' as const, FONT_FAMILY: ['Arial', 'sans-serif'] } // install text defaults -Text.defaultResolution = 2 +Text.defaultResolution = RESOLUTION Text.defaultAutoResolution = false TextStyle.defaultStyle = { ...TextStyle.defaultStyle, @@ -51,7 +76,6 @@ TextStyle.defaultStyle = { wordWrap: STYLE_DEFAULTS.WORD_WRAP, fontSize: STYLE_DEFAULTS.FONT_SIZE, fontFamily: STYLE_DEFAULTS.FONT_FAMILY, - letterSpacing: STYLE_DEFAULTS.LETTER_SPACING, strokeThickness: STYLE_DEFAULTS.STROKE_THICKNESS } @@ -80,29 +104,43 @@ const isASCII = (str: string) => { return true } -const getLabelCoordinates = ({ x, y, offset = 0 }: LabelCoords, isBitmapText: boolean, position: LabelPosition) => { +const getLabelCoordinates = ( + { x, y, offset = 0 }: LabelCoords, + isBitmapText: boolean, + { position, margin = 0, background }: StyleWithDefaults +) => { + let vertical = 0 + let horizontal = 0 + if (background !== undefined) { + const [v, h] = getBackgroundPadding(background.padding) + vertical += v / 2 + horizontal += h / 2 + } + + const shift = margin + offset + if (isBitmapText) { // BitmapText shifts text down 2px switch (position) { case 'bottom': - return [x, y + offset] + return [x, y + shift + vertical] case 'left': - return [x - offset - 2, y - 2] + return [x - shift - horizontal - 2, y - 2] case 'top': - return [x, y - offset - 4] + return [x, y - shift - vertical - 4] case 'right': - return [x + offset + 2, y - 2] + return [x + shift + horizontal + 2, y - 2] } } else { switch (position) { case 'bottom': - return [x, y + offset] + return [x, y + shift + vertical] case 'left': - return [x - offset - 2, y + 1] + return [x - shift - horizontal - 2, y + 1] case 'top': - return [x, y - offset + 2] + return [x, y - shift - vertical + 2] case 'right': - return [x + offset + 2, y + 1] + return [x + shift + horizontal + 2, y + 1] } } } @@ -124,7 +162,7 @@ const getPositionAnchor = (position: LabelPosition): [x: number, y: number] => { } } -const getTextStyle = ({ color, fontFamily, fontSize, wordWrap, stroke, position, letterSpacing }: StyleWithDefaults) => { +const getTextStyle = ({ color, fontFamily, fontSize, fontWeight, wordWrap, stroke, position, letterSpacing }: StyleWithDefaults) => { const style: Partial = {} if (color !== undefined) { style.fill = color @@ -135,6 +173,9 @@ const getTextStyle = ({ color, fontFamily, fontSize, wordWrap, stroke, position, if (fontSize !== undefined) { style.fontSize = fontSize } + if (fontWeight !== undefined) { + style.fontWeight = fontWeight + } if (wordWrap !== undefined) { style.wordWrap = true style.wordWrapWidth = wordWrap @@ -162,10 +203,68 @@ const bitmapFontIsAvailable = (fontName: string) => BitmapFont.available[fontNam const loadFont = (style: StyleWithDefaults) => { if (!bitmapFontIsAvailable(style.fontName)) { - BitmapFont.from(style.fontName, getTextStyle(style), { resolution: 2 }) + BitmapFont.from(style.fontName, { ...getTextStyle(style), letterSpacing: 1 }, { resolution: RESOLUTION, chars: BitmapFont.ASCII }) } } +const createTextObject = (label: string, style: StyleWithDefaults) => { + let text: BitmapText | Text + + if (isASCII(label)) { + loadFont(style) + text = new BitmapText(label, getBitmapStyle(style)) + } else { + text = new Text(label, getTextStyle(style)) + } + + text.anchor.set(...getPositionAnchor(style.position)) + return text +} + +const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { + return typeof padding === 'number' ? [padding, padding] : padding +} + +// TODO -> fix offset difference between text and bitmap text +const getBackgroundX = (x: number, isBitmapText: boolean, { position, background }: StyleWithDefaults) => { + const horizontal = getBackgroundPadding(background?.padding)[1] / 2 + switch (position) { + case 'right': + return x - horizontal - 2 + case 'left': + return x + horizontal - 2 + default: + return x + } +} + +const getBackgroundY = (y: number, isBitmapText: boolean, { position, background }: StyleWithDefaults) => { + const vertical = getBackgroundPadding(background?.padding)[0] / 2 + switch (position) { + case 'bottom': + return y - vertical + case 'top': + return y + vertical + default: + return y + } +} + +const setBackgroundStyle = (sprite: Sprite, text: Text | BitmapText, { color, opacity = 1, padding }: LabelBackgroundStyle) => { + const [vertical, horizontal] = getBackgroundPadding(padding) + sprite.anchor.set(text.anchor.x, text.anchor.y) + sprite.height = text.height + vertical + sprite.width = text.width + horizontal + sprite.alpha = opacity + sprite.tint = color + return sprite +} + +const createBackgroundSprite = (text: Text | BitmapText, style: LabelBackgroundStyle) => { + const sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) + return setBackgroundStyle(sprite, text, style) +} + export default { isASCII, mergeDefaults, @@ -175,5 +274,11 @@ export default { getTextStyle, getBitmapStyle, bitmapFontIsAvailable, - loadFont + loadFont, + createTextObject, + getBackgroundX, + getBackgroundY, + setBackgroundStyle, + createBackgroundSprite, + getBackgroundPadding } From 04d8f938339f5b055cd9d8b286c576864a5d1d6c Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Tue, 31 Oct 2023 15:50:17 -0400 Subject: [PATCH 05/10] correct label offset with background padding --- examples/native/src/labels/index.ts | 4 +- src/renderers/webgl/objects/label/Label.ts | 28 ++--- src/renderers/webgl/objects/label/utils.ts | 125 ++++++++------------- 3 files changed, 66 insertions(+), 91 deletions(-) diff --git a/examples/native/src/labels/index.ts b/examples/native/src/labels/index.ts index ce0cc733..87d02579 100644 --- a/examples/native/src/labels/index.ts +++ b/examples/native/src/labels/index.ts @@ -20,7 +20,7 @@ const NODE_STYLE: Graph.NodeStyle = { icon: TEXT_ICON, stroke: [{ width: 2, color: GREEN_LIGHT }], label: { - position: 'top', + position: 'bottom', fontName: 'NodeLabel', fontFamily: ['Arial', 'sans-serif'], background: { color: GREEN_LIGHT }, @@ -33,7 +33,7 @@ const NODE_HOVER_STYLE: Graph.NodeStyle = { icon: TEXT_ICON, stroke: [{ width: 2, color: GREEN_LIGHT }], label: { - position: 'top', + position: 'bottom', fontName: 'NodeLabelHover', fontFamily: ['Arial', 'sans-serif'], background: { color: GREEN_LIGHT }, diff --git a/src/renderers/webgl/objects/label/Label.ts b/src/renderers/webgl/objects/label/Label.ts index be0ed03a..9cd9998e 100644 --- a/src/renderers/webgl/objects/label/Label.ts +++ b/src/renderers/webgl/objects/label/Label.ts @@ -147,21 +147,23 @@ export class Label { } private set coordinates(coords: LabelCoords) { - const isBitmapText = this.isBitmapText() - const [x, y] = utils.getLabelCoordinates(coords, isBitmapText, this.style) + const { label, bg } = utils.getLabelCoordinates(coords, this.style, this.isBitmapText()) - if (x !== this.x) { - this.text.x = x - this.x = x - if (this.backgroundSprite) { - this.backgroundSprite.x = utils.getBackgroundX(x, isBitmapText, this.style) - } + if (label.x !== this.x) { + this.text.x = label.x + this.x = label.x } - if (y !== this.y) { - this.text.y = y - this.y = y - if (this.backgroundSprite) { - this.backgroundSprite.y = utils.getBackgroundY(y, isBitmapText, this.style) + if (label.y !== this.y) { + this.text.y = label.y + this.y = label.y + } + + if (this.backgroundSprite) { + if (this.backgroundSprite.x !== bg.x) { + this.backgroundSprite.x = bg.x + } + if (this.backgroundSprite.y !== bg.y) { + this.backgroundSprite.y = bg.y } } } diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index b1c696e8..4402d09f 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,3 +1,4 @@ +/* eslint-disable @typescript-eslint/no-unused-vars */ import type { Stroke } from '../../../../types' import { Text, @@ -104,47 +105,6 @@ const isASCII = (str: string) => { return true } -const getLabelCoordinates = ( - { x, y, offset = 0 }: LabelCoords, - isBitmapText: boolean, - { position, margin = 0, background }: StyleWithDefaults -) => { - let vertical = 0 - let horizontal = 0 - if (background !== undefined) { - const [v, h] = getBackgroundPadding(background.padding) - vertical += v / 2 - horizontal += h / 2 - } - - const shift = margin + offset - - if (isBitmapText) { - // BitmapText shifts text down 2px - switch (position) { - case 'bottom': - return [x, y + shift + vertical] - case 'left': - return [x - shift - horizontal - 2, y - 2] - case 'top': - return [x, y - shift - vertical - 4] - case 'right': - return [x + shift + horizontal + 2, y - 2] - } - } else { - switch (position) { - case 'bottom': - return [x, y + shift + vertical] - case 'left': - return [x - shift - horizontal - 2, y + 1] - case 'top': - return [x, y - shift - vertical + 2] - case 'right': - return [x + shift + horizontal + 2, y + 1] - } - } -} - const getPositionAlign = (position: LabelPosition): TextStyleAlign => { return position === 'left' || position === 'right' ? position : 'center' } @@ -207,6 +167,20 @@ const loadFont = (style: StyleWithDefaults) => { } } +const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { + return typeof padding === 'number' ? [padding, padding] : padding +} + +const setBackgroundStyle = (sprite: Sprite, text: Text | BitmapText, { color, opacity = 1, padding }: LabelBackgroundStyle) => { + const [vertical, horizontal] = getBackgroundPadding(padding) + sprite.anchor.set(text.anchor.x, text.anchor.y) + sprite.height = text.height + vertical + sprite.width = text.width + horizontal + sprite.alpha = opacity + sprite.tint = color + return sprite +} + const createTextObject = (label: string, style: StyleWithDefaults) => { let text: BitmapText | Text @@ -221,48 +195,49 @@ const createTextObject = (label: string, style: StyleWithDefaults) => { return text } -const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { - return typeof padding === 'number' ? [padding, padding] : padding +const createBackgroundSprite = (text: Text | BitmapText, style: LabelBackgroundStyle) => { + const sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) + return setBackgroundStyle(sprite, text, style) } -// TODO -> fix offset difference between text and bitmap text -const getBackgroundX = (x: number, isBitmapText: boolean, { position, background }: StyleWithDefaults) => { - const horizontal = getBackgroundPadding(background?.padding)[1] / 2 - switch (position) { - case 'right': - return x - horizontal - 2 - case 'left': - return x + horizontal - 2 - default: - return x +const getLabelCoordinates = ( + { x, y, offset = 0 }: LabelCoords, + { position, margin = 2, background }: StyleWithDefaults, + isBitmapText: boolean +) => { + const shift = margin + offset + // BitmapText shifts text down 2px + const label = { x, y: isBitmapText ? y - 1 : y + 1 } + const bg = { x, y: isBitmapText ? y - 1 : y + 1 } + + let vertical = 0 + let horizontal = 0 + if (background !== undefined) { + const [v, h] = getBackgroundPadding(background.padding) + vertical += v / 2 + horizontal += h / 2 } -} -const getBackgroundY = (y: number, isBitmapText: boolean, { position, background }: StyleWithDefaults) => { - const vertical = getBackgroundPadding(background?.padding)[0] / 2 switch (position) { case 'bottom': - return y - vertical + label.y += shift + vertical + bg.y += shift + break + case 'left': + label.x -= shift + horizontal + bg.x -= shift + break case 'top': - return y + vertical - default: - return y + label.y -= shift + vertical + bg.y -= shift + break + case 'right': + label.x += shift + horizontal + bg.x += shift + break } -} -const setBackgroundStyle = (sprite: Sprite, text: Text | BitmapText, { color, opacity = 1, padding }: LabelBackgroundStyle) => { - const [vertical, horizontal] = getBackgroundPadding(padding) - sprite.anchor.set(text.anchor.x, text.anchor.y) - sprite.height = text.height + vertical - sprite.width = text.width + horizontal - sprite.alpha = opacity - sprite.tint = color - return sprite -} - -const createBackgroundSprite = (text: Text | BitmapText, style: LabelBackgroundStyle) => { - const sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) - return setBackgroundStyle(sprite, text, style) + return { label, bg } } export default { @@ -276,8 +251,6 @@ export default { bitmapFontIsAvailable, loadFont, createTextObject, - getBackgroundX, - getBackgroundY, setBackgroundStyle, createBackgroundSprite, getBackgroundPadding From ab652c6cc227d0a3f2bb3df77729dd2aa6805dc0 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Tue, 31 Oct 2023 16:38:01 -0400 Subject: [PATCH 06/10] resolve issues with size changes to label and background --- examples/native/src/labels/index.ts | 8 +- src/renderers/webgl/objects/label/Label.ts | 97 +++++++++++----------- src/renderers/webgl/objects/label/utils.ts | 75 +++++++++++------ 3 files changed, 100 insertions(+), 80 deletions(-) diff --git a/examples/native/src/labels/index.ts b/examples/native/src/labels/index.ts index 87d02579..054c5701 100644 --- a/examples/native/src/labels/index.ts +++ b/examples/native/src/labels/index.ts @@ -20,7 +20,7 @@ const NODE_STYLE: Graph.NodeStyle = { icon: TEXT_ICON, stroke: [{ width: 2, color: GREEN_LIGHT }], label: { - position: 'bottom', + position: 'right', fontName: 'NodeLabel', fontFamily: ['Arial', 'sans-serif'], background: { color: GREEN_LIGHT }, @@ -33,11 +33,11 @@ const NODE_HOVER_STYLE: Graph.NodeStyle = { icon: TEXT_ICON, stroke: [{ width: 2, color: GREEN_LIGHT }], label: { - position: 'bottom', + position: 'right', fontName: 'NodeLabelHover', fontFamily: ['Arial', 'sans-serif'], - background: { color: GREEN_LIGHT }, - color: DARK_GREEN, + background: { color: DARK_GREEN }, + color: '#FFF', margin: 4 } } diff --git a/src/renderers/webgl/objects/label/Label.ts b/src/renderers/webgl/objects/label/Label.ts index 9cd9998e..580226ab 100644 --- a/src/renderers/webgl/objects/label/Label.ts +++ b/src/renderers/webgl/objects/label/Label.ts @@ -14,6 +14,7 @@ export class Label { mounted = false private dirty = false + private transformed = false private label: string private container: Container private text: BitmapText | Text @@ -35,7 +36,17 @@ export class Label { } update(label: string, coords: LabelCoords, style?: LabelStyle) { + const previous = this.text.getLocalBounds() + this.value = label + + const isBitmapText = this.isBitmapText() + const isASCII = utils.isASCII(label) + // if the text type has changed, regenerate a new text object + if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { + this.transformText(label, utils.mergeDefaults(style)) + } + this.style.margin = style?.margin this.coordinates = coords this.wordWrap = style?.wordWrap @@ -48,25 +59,19 @@ export class Label { this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME this.background = style?.background - const isBitmapText = this.isBitmapText() - const isASCII = utils.isASCII(label) - - if (isASCII) { - // conditionally load font if BitmapFont is unavailable - utils.loadFont(this.style) - } - - if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { - // if the text type has changed, regenerate a new text object - this.dirty = false - this.transformText() - } - if (this.dirty) { this.dirty = false this.updateText() } + if (this.backgroundSprite) { + const bounds = this.text.getLocalBounds() + if (previous.width !== bounds.width || previous.height !== bounds.height) { + utils.setBackgroundSize(this.backgroundSprite, bounds, this.style.background?.padding) + } + } + + this.transformed = false return this } @@ -115,27 +120,17 @@ export class Label { } } - private transformText() { - if (this.isBitmapText()) { - const isMounted = this.mounted - this.delete() - this.text = new Text(this.label, utils.getTextStyle(this.style)) - this.position = this.style.position - this.x = undefined - this.y = undefined - if (isMounted) { - this.mount() - } - } else { - const isMounted = this.mounted - this.delete() - this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) - this.position = this.style.position - this.x = undefined - this.y = undefined - if (isMounted) { - this.mount() - } + private transformText(label: string, style: StyleWithDefaults) { + const isMounted = this.mounted + + this.delete() + this.text = utils.createTextObject(label, style) + this.transformed = true + this.x = undefined + this.y = undefined + + if (isMounted) { + this.mount() } } @@ -172,7 +167,7 @@ export class Label { this.align = utils.getPositionAlign(position) this.anchor = utils.getPositionAnchor(position) if (position !== this.style.position) { - this.dirty = true + this.dirty = !this.transformed this.style.position = position } } @@ -199,12 +194,14 @@ export class Label { private set fontSize(fontSize: number) { if (fontSize !== this.style.fontSize) { - this.dirty = true this.style.fontSize = fontSize - if (this.isBitmapText(this.text)) { - this.text.fontSize = fontSize - } else { - this.text.style.fontSize = fontSize + if (!this.transformed) { + this.dirty = true + if (this.isBitmapText(this.text)) { + this.text.fontSize = fontSize + } else { + this.text.style.fontSize = fontSize + } } } } @@ -212,7 +209,7 @@ export class Label { private set wordWrap(wordWrap: number | undefined) { if (wordWrap !== this.style.wordWrap) { this.style.wordWrap = wordWrap - if (!this.isBitmapText(this.text)) { + if (!this.transformed && !this.isBitmapText(this.text)) { this.dirty = true this.text.style.wordWrap = wordWrap !== undefined this.text.style.wordWrapWidth = wordWrap ?? 0 @@ -223,7 +220,7 @@ export class Label { private set color(color: TextStyleFill | undefined) { if (!equals(color, this.style.color)) { this.style.color = color - if (!this.isBitmapText(this.text)) { + if (!this.transformed && !this.isBitmapText(this.text)) { this.dirty = true this.text.style.fill = color ?? STYLE_DEFAULTS.COLOR } @@ -233,7 +230,7 @@ export class Label { private set stroke(stroke: Stroke | undefined) { if (!equals(stroke, this.style.stroke)) { this.style.stroke = stroke - if (!this.isBitmapText(this.text)) { + if (!this.transformed && !this.isBitmapText(this.text)) { this.dirty = true this.text.style.stroke = stroke?.color ?? STYLE_DEFAULTS.STROKE this.text.style.strokeThickness = stroke?.width ?? STYLE_DEFAULTS.STROKE_THICKNESS @@ -243,10 +240,12 @@ export class Label { private set fontFamily(fontFamily: string | string[]) { if (!equals(fontFamily, this.style.fontFamily)) { - this.dirty = true this.style.fontFamily = fontFamily - if (!this.isBitmapText(this.text)) { - this.text.style.fontFamily = fontFamily + if (!this.transformed) { + this.dirty = true + if (!this.isBitmapText(this.text)) { + this.text.style.fontFamily = fontFamily + } } } } @@ -254,7 +253,7 @@ export class Label { private set fontName(fontName: string) { if (fontName !== this.style.fontName) { this.style.fontName = fontName - if (this.isBitmapText(this.text)) { + if (!this.transformed && this.isBitmapText(this.text)) { this.dirty = true this.text.fontName = fontName } @@ -264,7 +263,7 @@ export class Label { private set fontWeight(fontWeight: TextStyleFontWeight | undefined) { if (fontWeight !== this.style.fontWeight) { this.style.fontWeight = fontWeight - if (!this.isBitmapText(this.text)) { + if (!this.transformed && !this.isBitmapText(this.text)) { this.dirty = true this.text.style.fontWeight = fontWeight ?? STYLE_DEFAULTS.FONT_WEIGHT } diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index 4402d09f..ce0abcec 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,4 +1,3 @@ -/* eslint-disable @typescript-eslint/no-unused-vars */ import type { Stroke } from '../../../../types' import { Text, @@ -12,7 +11,8 @@ import { Sprite, Texture, BitmapText, - TextStyleFontWeight + TextStyleFontWeight, + Rectangle } from 'pixi.js' export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' @@ -53,8 +53,9 @@ const RESOLUTION = 2 export const STYLE_DEFAULTS = { FONT_SIZE: 10, STROKE_THICKNESS: 0, - PADDING: [4, 8] as [number, number], + LETTER_SPACING: 0.5, WORD_WRAP: false, + PADDING: [4, 8] as [number, number], STROKE: '#FFF', FONT_NAME: 'Label', COLOR: '#000000', @@ -77,7 +78,8 @@ TextStyle.defaultStyle = { wordWrap: STYLE_DEFAULTS.WORD_WRAP, fontSize: STYLE_DEFAULTS.FONT_SIZE, fontFamily: STYLE_DEFAULTS.FONT_FAMILY, - strokeThickness: STYLE_DEFAULTS.STROKE_THICKNESS + strokeThickness: STYLE_DEFAULTS.STROKE_THICKNESS, + letterSpacing: STYLE_DEFAULTS.LETTER_SPACING } // utils @@ -156,31 +158,19 @@ const getTextStyle = ({ color, fontFamily, fontSize, fontWeight, wordWrap, strok const getBitmapStyle = (style: StyleWithDefaults): Partial => ({ fontName: style.fontName, fontSize: style.fontSize, - align: getPositionAlign(style.position) + align: getPositionAlign(style.position), + letterSpacing: style.letterSpacing ?? STYLE_DEFAULTS.LETTER_SPACING }) -const bitmapFontIsAvailable = (fontName: string) => BitmapFont.available[fontName] !== undefined - const loadFont = (style: StyleWithDefaults) => { - if (!bitmapFontIsAvailable(style.fontName)) { - BitmapFont.from(style.fontName, { ...getTextStyle(style), letterSpacing: 1 }, { resolution: RESOLUTION, chars: BitmapFont.ASCII }) + if (BitmapFont.available[style.fontName] === undefined) { + BitmapFont.from(style.fontName, getTextStyle(style), { + resolution: RESOLUTION, + chars: BitmapFont.ASCII + }) } } -const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { - return typeof padding === 'number' ? [padding, padding] : padding -} - -const setBackgroundStyle = (sprite: Sprite, text: Text | BitmapText, { color, opacity = 1, padding }: LabelBackgroundStyle) => { - const [vertical, horizontal] = getBackgroundPadding(padding) - sprite.anchor.set(text.anchor.x, text.anchor.y) - sprite.height = text.height + vertical - sprite.width = text.width + horizontal - sprite.alpha = opacity - sprite.tint = color - return sprite -} - const createTextObject = (label: string, style: StyleWithDefaults) => { let text: BitmapText | Text @@ -195,6 +185,24 @@ const createTextObject = (label: string, style: StyleWithDefaults) => { return text } +const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { + return typeof padding === 'number' ? [padding, padding] : padding +} + +const setBackgroundSize = (sprite: Sprite, bounds: Rectangle, padding?: BackgroundPadding) => { + const [vertical, horizontal] = getBackgroundPadding(padding) + sprite.height = bounds.height + vertical + sprite.width = bounds.width + horizontal + return sprite +} + +const setBackgroundStyle = (sprite: Sprite, text: Text | BitmapText, { color, opacity = 1, padding }: LabelBackgroundStyle) => { + sprite.anchor.set(text.anchor.x, text.anchor.y) + sprite.alpha = opacity + sprite.tint = color + return setBackgroundSize(sprite, text.getLocalBounds(), padding) +} + const createBackgroundSprite = (text: Text | BitmapText, style: LabelBackgroundStyle) => { const sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) return setBackgroundStyle(sprite, text, style) @@ -206,9 +214,8 @@ const getLabelCoordinates = ( isBitmapText: boolean ) => { const shift = margin + offset - // BitmapText shifts text down 2px - const label = { x, y: isBitmapText ? y - 1 : y + 1 } - const bg = { x, y: isBitmapText ? y - 1 : y + 1 } + const label = { x, y } + const bg = { x, y } let vertical = 0 let horizontal = 0 @@ -222,18 +229,32 @@ const getLabelCoordinates = ( case 'bottom': label.y += shift + vertical bg.y += shift + break case 'left': label.x -= shift + horizontal bg.x -= shift + + if (isBitmapText) { + label.y -= 1 + bg.y -= 1 + } + break case 'top': label.y -= shift + vertical bg.y -= shift + break case 'right': label.x += shift + horizontal bg.x += shift + + if (isBitmapText) { + label.y -= 1 + bg.y -= 1 + } + break } @@ -248,9 +269,9 @@ export default { getPositionAnchor, getTextStyle, getBitmapStyle, - bitmapFontIsAvailable, loadFont, createTextObject, + setBackgroundSize, setBackgroundStyle, createBackgroundSprite, getBackgroundPadding From 8413a82f37dc91f69f4812fe3fc4aa06be9b5815 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Wed, 1 Nov 2023 11:36:07 -0400 Subject: [PATCH 07/10] break apart LabelBackground into its own object class --- src/renderers/webgl/node.ts | 30 +- src/renderers/webgl/objects/label/Label.ts | 291 ----------------- .../webgl/objects/label/background.ts | 126 ++++++++ src/renderers/webgl/objects/label/index.ts | 304 +++++++++++++++++- src/renderers/webgl/objects/label/utils.ts | 82 ++--- 5 files changed, 470 insertions(+), 363 deletions(-) delete mode 100644 src/renderers/webgl/objects/label/Label.ts create mode 100644 src/renderers/webgl/objects/label/background.ts diff --git a/src/renderers/webgl/node.ts b/src/renderers/webgl/node.ts index 82b95bb6..5af2a8bb 100644 --- a/src/renderers/webgl/node.ts +++ b/src/renderers/webgl/node.ts @@ -42,7 +42,17 @@ export class NodeRenderer { } update(node: Graph.Node) { - this.setLabel(node) + if (this.label === undefined) { + if (node.label !== undefined) { + this.label = new Label(this.renderer.labelsContainer, node.label, node.style?.label) + } + } else if (node.label === undefined || node.label.trim() === '') { + this.renderer.labelObjectManager.delete(this.label) + this.labelMounted = false + this.label = undefined + } else { + this.label.update(node.label, node.style?.label) + } if (this.icon === undefined) { if (node.style?.icon) { @@ -491,7 +501,9 @@ export class NodeRenderer { this.fill.update(this.x, this.y, radius, node.style) this.strokes.update(this.x, this.y, radius, node.style) - this.setLabel(node) + if (this.label !== undefined) { + this.label.moveTo(this.x, this.y, this.strokes.radius) + } if (this.icon && node.style?.icon) { this.icon.update(this.x, this.y, node.style.icon) } @@ -507,18 +519,4 @@ export class NodeRenderer { this.y - this.strokes.radius <= this.renderer.maxY ) } - - private setLabel(node: Graph.Node) { - if (this.label === undefined) { - if (node.label !== undefined) { - this.label = new Label(this.renderer.labelsContainer, node.label, node.style?.label) - } - } else if (node.label === undefined) { - this.renderer.labelObjectManager.delete(this.label) - this.labelMounted = false - this.label = undefined - } else { - this.label.update(node.label, { x: this.x, y: this.y, offset: this.strokes.radius }, node.style?.label) - } - } } diff --git a/src/renderers/webgl/objects/label/Label.ts b/src/renderers/webgl/objects/label/Label.ts deleted file mode 100644 index 580226ab..00000000 --- a/src/renderers/webgl/objects/label/Label.ts +++ /dev/null @@ -1,291 +0,0 @@ -import type { StyleWithDefaults, LabelCoords, LabelPosition, LabelStyle, LabelBackgroundStyle } from './utils' -import type { Stroke } from '../../../../types' -import { BitmapText, Container, Sprite, Text, TextStyleAlign, TextStyleFill, TextStyleFontWeight } from 'pixi.js' -import { equals } from '../../../..' -import utils, { STYLE_DEFAULTS } from './utils' - -/** - * TODO - * - add support for background color, custom font loading - * - add support for loading custom fonts as asset bundles - * - moving/scaling labels is slow. render ASCII text characters as sprites to partical container? - */ -export class Label { - mounted = false - - private dirty = false - private transformed = false - private label: string - private container: Container - private text: BitmapText | Text - private style: StyleWithDefaults - private backgroundSprite: Sprite | null = null - private x?: number - private y?: number - - constructor(container: Container, label: string, labelStyle?: LabelStyle) { - const style = utils.mergeDefaults(labelStyle) - const text = utils.createTextObject(label, style) - this.container = container - this.label = label - this.style = style - this.text = text - if (style.background !== undefined) { - this.backgroundSprite = utils.createBackgroundSprite(text, style.background) - } - } - - update(label: string, coords: LabelCoords, style?: LabelStyle) { - const previous = this.text.getLocalBounds() - - this.value = label - - const isBitmapText = this.isBitmapText() - const isASCII = utils.isASCII(label) - // if the text type has changed, regenerate a new text object - if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { - this.transformText(label, utils.mergeDefaults(style)) - } - - this.style.margin = style?.margin - this.coordinates = coords - this.wordWrap = style?.wordWrap - this.color = style?.color - this.stroke = style?.stroke - this.fontWeight = style?.fontWeight - this.position = style?.position ?? STYLE_DEFAULTS.POSITION - this.fontSize = style?.fontSize ?? STYLE_DEFAULTS.FONT_SIZE - this.fontFamily = style?.fontFamily ?? STYLE_DEFAULTS.FONT_FAMILY - this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME - this.background = style?.background - - if (this.dirty) { - this.dirty = false - this.updateText() - } - - if (this.backgroundSprite) { - const bounds = this.text.getLocalBounds() - if (previous.width !== bounds.width || previous.height !== bounds.height) { - utils.setBackgroundSize(this.backgroundSprite, bounds, this.style.background?.padding) - } - } - - this.transformed = false - return this - } - - mount() { - if (!this.mounted) { - if (this.backgroundSprite) { - this.container.addChild(this.backgroundSprite) - } - - this.container.addChild(this.text) - this.mounted = true - } - - return this - } - - unmount() { - if (this.mounted) { - if (this.backgroundSprite) { - this.container.removeChild(this.backgroundSprite) - } - - this.container.removeChild(this.text) - this.mounted = false - } - - return this - } - - delete() { - this.unmount() - this.text.destroy() - - return undefined - } - - private isBitmapText(text: Text | BitmapText = this.text): text is BitmapText { - return text instanceof BitmapText - } - - private updateText() { - if (this.isBitmapText(this.text)) { - this.text.updateText() - } else { - this.text.updateText(true) - } - } - - private transformText(label: string, style: StyleWithDefaults) { - const isMounted = this.mounted - - this.delete() - this.text = utils.createTextObject(label, style) - this.transformed = true - this.x = undefined - this.y = undefined - - if (isMounted) { - this.mount() - } - } - - private set value(label: string) { - if (label !== this.label) { - this.text.text = label - this.label = label - } - } - - private set coordinates(coords: LabelCoords) { - const { label, bg } = utils.getLabelCoordinates(coords, this.style, this.isBitmapText()) - - if (label.x !== this.x) { - this.text.x = label.x - this.x = label.x - } - if (label.y !== this.y) { - this.text.y = label.y - this.y = label.y - } - - if (this.backgroundSprite) { - if (this.backgroundSprite.x !== bg.x) { - this.backgroundSprite.x = bg.x - } - if (this.backgroundSprite.y !== bg.y) { - this.backgroundSprite.y = bg.y - } - } - } - - private set position(position: LabelPosition) { - this.align = utils.getPositionAlign(position) - this.anchor = utils.getPositionAnchor(position) - if (position !== this.style.position) { - this.dirty = !this.transformed - this.style.position = position - } - } - - private set align(align: TextStyleAlign) { - if (this.isBitmapText(this.text)) { - if (this.text.align !== align) { - this.dirty = true - this.text.align = align - } - } else { - if (this.text.style.align !== align) { - this.dirty = true - this.text.style.align = align - } - } - } - - private set anchor([x, y]: [x: number, y: number]) { - if (!this.text.anchor.equals({ x, y })) { - this.text.anchor.set(x, y) - } - } - - private set fontSize(fontSize: number) { - if (fontSize !== this.style.fontSize) { - this.style.fontSize = fontSize - if (!this.transformed) { - this.dirty = true - if (this.isBitmapText(this.text)) { - this.text.fontSize = fontSize - } else { - this.text.style.fontSize = fontSize - } - } - } - } - - private set wordWrap(wordWrap: number | undefined) { - if (wordWrap !== this.style.wordWrap) { - this.style.wordWrap = wordWrap - if (!this.transformed && !this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.wordWrap = wordWrap !== undefined - this.text.style.wordWrapWidth = wordWrap ?? 0 - } - } - } - - private set color(color: TextStyleFill | undefined) { - if (!equals(color, this.style.color)) { - this.style.color = color - if (!this.transformed && !this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.fill = color ?? STYLE_DEFAULTS.COLOR - } - } - } - - private set stroke(stroke: Stroke | undefined) { - if (!equals(stroke, this.style.stroke)) { - this.style.stroke = stroke - if (!this.transformed && !this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.stroke = stroke?.color ?? STYLE_DEFAULTS.STROKE - this.text.style.strokeThickness = stroke?.width ?? STYLE_DEFAULTS.STROKE_THICKNESS - } - } - } - - private set fontFamily(fontFamily: string | string[]) { - if (!equals(fontFamily, this.style.fontFamily)) { - this.style.fontFamily = fontFamily - if (!this.transformed) { - this.dirty = true - if (!this.isBitmapText(this.text)) { - this.text.style.fontFamily = fontFamily - } - } - } - } - - private set fontName(fontName: string) { - if (fontName !== this.style.fontName) { - this.style.fontName = fontName - if (!this.transformed && this.isBitmapText(this.text)) { - this.dirty = true - this.text.fontName = fontName - } - } - } - - private set fontWeight(fontWeight: TextStyleFontWeight | undefined) { - if (fontWeight !== this.style.fontWeight) { - this.style.fontWeight = fontWeight - if (!this.transformed && !this.isBitmapText(this.text)) { - this.dirty = true - this.text.style.fontWeight = fontWeight ?? STYLE_DEFAULTS.FONT_WEIGHT - } - } - } - - private set background(background: LabelBackgroundStyle | undefined) { - if (!equals(background, this.style.background)) { - this.style.background = background - if (this.backgroundSprite) { - if (background !== undefined) { - utils.setBackgroundStyle(this.backgroundSprite, this.text, background) - } else { - const sprite = this.backgroundSprite - this.backgroundSprite = null - this.mounted && this.container.removeChild(sprite) - sprite.destroy() - } - } else if (background !== undefined) { - this.backgroundSprite = utils.createBackgroundSprite(this.text, background) - this.mounted && this.container.addChild(this.backgroundSprite) - } - } - } -} diff --git a/src/renderers/webgl/objects/label/background.ts b/src/renderers/webgl/objects/label/background.ts new file mode 100644 index 00000000..f7b7e036 --- /dev/null +++ b/src/renderers/webgl/objects/label/background.ts @@ -0,0 +1,126 @@ +import utils, { RESOLUTION, STYLE_DEFAULTS } from './utils' +import type { BackgroundPadding, LabelBackgroundStyle } from './utils' +import { BitmapText, ColorSource, Container, Rectangle, Sprite, Text, Texture } from 'pixi.js' +import { equals } from '../../../..' + +export class LabelBackground { + mounted = false + + private x?: number + private y?: number + private dirty = false + private sprite: Sprite + private label: Text | BitmapText + private container: Container + private style: Required + private rect: Rectangle + + constructor(container: Container, label: Text | BitmapText, style: LabelBackgroundStyle) { + this.label = label + this.container = container + this.style = utils.mergeBackgroundDefaults(style) + this.sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) + this.sprite.anchor.set(this.label.anchor.x, this.label.anchor.y) + this.sprite.alpha = this.style.opacity + this.sprite.tint = this.style.color + this.rect = label.getLocalBounds() + this.resize() + } + + update(label: Text | BitmapText, style: LabelBackgroundStyle) { + this.label = label + this.color = style.color + this.bounds = label.getLocalBounds() + this.opacity = style.opacity ?? STYLE_DEFAULTS.OPACITY + this.padding = style.padding ?? STYLE_DEFAULTS.PADDING + this.sprite.anchor.set(this.label.anchor.x, this.label.anchor.y) + + if (this.dirty) { + this.dirty = false + this.resize() + } + + return this + } + + moveTo(x: number, y: number) { + if (this.x !== x) { + this.x = x + this.sprite.x = x + } + + if (this.y !== y) { + this.y = y + this.sprite.y = y + } + } + + mount() { + if (!this.mounted) { + this.container.addChild(this.sprite) + this.mounted = true + } + + return this + } + + unmount() { + if (this.mounted) { + this.container.removeChild(this.sprite) + this.mounted = false + } + + return this + } + + delete() { + this.unmount() + this.sprite.destroy() + + return undefined + } + + private resize() { + const [vertical, horizontal] = utils.getBackgroundPadding(this.style.padding) + + const height = this.rect.height + vertical + if (height !== this.sprite.height) { + this.sprite.height = height + } + + const width = this.rect.width + horizontal + if (width !== this.sprite.width) { + this.sprite.width = width + } + + return this + } + + private set color(color: ColorSource) { + if (color !== this.style.color) { + this.style.color = color + this.sprite.tint = color + } + } + + private set opacity(opacity: number) { + if (opacity !== this.style.opacity) { + this.style.opacity = opacity + this.sprite.alpha = opacity + } + } + + private set padding(padding: BackgroundPadding) { + if (!equals(padding, this.style.padding)) { + this.style.padding = padding + this.dirty = true + } + } + + private set bounds(bounds: Rectangle) { + if (this.rect.width !== bounds.width || this.rect.height !== bounds.height) { + this.rect = bounds + this.dirty = true + } + } +} diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts index 19d944dc..0f124800 100644 --- a/src/renderers/webgl/objects/label/index.ts +++ b/src/renderers/webgl/objects/label/index.ts @@ -1,2 +1,304 @@ -export { Label } from './Label' +import utils, { STYLE_DEFAULTS } from './utils' +import type { StyleWithDefaults, LabelPosition, LabelStyle, LabelBackgroundStyle } from './utils' +import type { Stroke } from '../../../../types' +import { BitmapText, Container, Text, TextStyleAlign, TextStyleFill, TextStyleFontWeight } from 'pixi.js' +import { LabelBackground } from './background' +import { equals } from '../../../..' + +/** + * TODO + * - add support for background color, custom font loading + * - add support for loading custom fonts as asset bundles + * - moving/scaling labels is slow. render ASCII text characters as sprites to partical container? + */ +export class Label { + mounted = false + + private x?: number + private y?: number + private dirty = false + private transformed = false + private label: string + private container: Container + private text: BitmapText | Text + private style: StyleWithDefaults + private labelBackground: LabelBackground | null = null + + constructor(container: Container, label: string, style: LabelStyle | undefined) { + this.label = label + this.container = container + this.style = utils.mergeDefaults(style) + + if (utils.isASCII(this.label)) { + utils.loadFont(this.style) + this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) + } else { + this.text = new Text(this.label, utils.getTextStyle(this.style)) + } + + this.text.anchor.set(...utils.getAnchorPoint(this.style.position)) + if (this.style.background !== undefined) { + this.labelBackground = new LabelBackground(container, this.text, this.style.background) + } + } + + update(label: string, style: LabelStyle | undefined) { + this.value = label + + const isBitmapText = this.isBitmapText() + const isASCII = utils.isASCII(this.label) + // if the text type has changed, regenerate a new text object + if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { + this.transformText(this.label, utils.mergeDefaults(style)) + } + + this.wordWrap = style?.wordWrap + this.color = style?.color + this.stroke = style?.stroke + this.fontWeight = style?.fontWeight + this.letterSpacing = style?.letterSpacing + this.position = style?.position ?? STYLE_DEFAULTS.POSITION + this.fontSize = style?.fontSize ?? STYLE_DEFAULTS.FONT_SIZE + this.fontFamily = style?.fontFamily ?? STYLE_DEFAULTS.FONT_FAMILY + this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME + this.background = style?.background + + if (this.dirty) { + this.dirty = false + this.updateText() + } + + this.transformed = false + return this + } + + moveTo(x: number, y: number, offset = 0) { + const { label, bg } = utils.getLabelCoordinates(x, y, offset, this.isBitmapText(), this.style) + + this.labelBackground?.moveTo(bg.x, bg.y) + + if (label.x !== this.x) { + this.text.x = label.x + this.x = label.x + } + if (label.y !== this.y) { + this.text.y = label.y + this.y = label.y + } + } + + mount() { + if (!this.mounted) { + this.labelBackground?.mount() + this.container.addChild(this.text) + this.mounted = true + } + + return this + } + + unmount() { + if (this.mounted) { + this.labelBackground?.unmount() + this.container.removeChild(this.text) + this.mounted = false + } + + return this + } + + delete() { + this.unmount() + this.text.destroy() + if (!this.transformed) { + this.labelBackground?.delete() + } + + return undefined + } + + private isBitmapText(text: Text | BitmapText = this.text): text is BitmapText { + return text instanceof BitmapText + } + + private updateText() { + if (this.isBitmapText(this.text)) { + this.text.updateText() + } else { + this.text.updateText(true) + } + } + + private transformText(label: string, style: StyleWithDefaults) { + this.transformed = true + const isMounted = this.mounted + + this.delete() + + if (utils.isASCII(label)) { + utils.loadFont(style) + this.text = new BitmapText(label, utils.getBitmapStyle(style)) + } else { + this.text = new Text(label, utils.getTextStyle(style)) + } + + this.text.anchor.set(...utils.getAnchorPoint(style.position)) + this.text.x = this.x ?? 0 + this.text.y = this.y ?? 0 + + if (isMounted) { + this.mount() + } + } + + private set value(label: string) { + if (label !== this.label) { + this.text.text = label + this.label = label + } + } + + private set position(position: LabelPosition) { + this.align = utils.getTextAlign(position) + this.anchor = utils.getAnchorPoint(position) + this.style.position = position + } + + private set align(align: TextStyleAlign) { + if (this.isBitmapText(this.text)) { + if (this.text.align !== align) { + this.dirty = true + this.text.align = align + } + } else { + if (this.text.style.align !== align) { + this.dirty = true + this.text.style.align = align + } + } + } + + private set anchor([x, y]: [x: number, y: number]) { + if (!this.text.anchor.equals({ x, y })) { + this.text.anchor.set(x, y) + } + } + + private set fontSize(fontSize: number) { + if (fontSize !== this.style.fontSize) { + this.style.fontSize = fontSize + if (this.isBitmapText(this.text)) { + if (this.text.fontSize !== fontSize) { + this.dirty = true + this.text.fontSize = fontSize + } + } else { + if (this.text.style.fontSize !== fontSize) { + this.dirty = true + this.text.style.fontSize = fontSize + } + } + } + } + + private set wordWrap(wordWrap: number | undefined) { + if (wordWrap !== this.style.wordWrap) { + this.style.wordWrap = wordWrap + const wordWrapWidth = wordWrap ?? 0 + if (!this.isBitmapText(this.text) && this.text.style.wordWrapWidth !== wordWrapWidth) { + this.dirty = true + this.text.style.wordWrap = wordWrap !== undefined + this.text.style.wordWrapWidth = wordWrapWidth + } + } + } + + private set color(value: TextStyleFill | undefined) { + if (!equals(value, this.style.color)) { + this.style.color = value + const color = value ?? STYLE_DEFAULTS.COLOR + if (!this.isBitmapText(this.text) && this.text.style.fill !== color) { + this.dirty = true + this.text.style.fill = color + } + } + } + + private set stroke(value: Stroke | undefined) { + if (!equals(value, this.style.stroke)) { + this.style.stroke = value + const stroke = value?.color ?? STYLE_DEFAULTS.STROKE + const strokeThickness = value?.width ?? STYLE_DEFAULTS.STROKE_THICKNESS + if (!this.isBitmapText(this.text) && (this.text.style.stroke !== stroke || this.text.style.strokeThickness !== strokeThickness)) { + this.dirty = true + this.text.style.stroke = stroke + this.text.style.strokeThickness = strokeThickness + } + } + } + + private set fontFamily(fontFamily: string | string[]) { + if (!equals(fontFamily, this.style.fontFamily)) { + this.style.fontFamily = fontFamily + if (!this.isBitmapText(this.text) && !equals(this.text.style.fontFamily, fontFamily)) { + this.dirty = true + this.text.style.fontFamily = fontFamily + } + } + } + + private set fontName(fontName: string) { + if (fontName !== this.style.fontName) { + this.style.fontName = fontName + if (this.isBitmapText(this.text) && this.text.fontName !== fontName) { + this.dirty = true + this.text.fontName = fontName + } + } + } + + private set fontWeight(value: TextStyleFontWeight | undefined) { + if (value !== this.style.fontWeight) { + this.style.fontWeight = value + const fontWeight = value ?? STYLE_DEFAULTS.FONT_WEIGHT + if (!this.isBitmapText(this.text) && this.text.style.fontWeight !== fontWeight) { + this.dirty = true + this.text.style.fontWeight = fontWeight + } + } + } + + private set background(background: LabelBackgroundStyle | undefined) { + this.style.background = background + + if (this.labelBackground === null && background !== undefined) { + this.labelBackground = new LabelBackground(this.container, this.text, background) + } else if (this.labelBackground && background !== undefined) { + this.labelBackground.update(this.text, background) + } else if (this.labelBackground && background === undefined) { + this.labelBackground.delete() + this.labelBackground = null + } + } + + private set letterSpacing(value: number | undefined) { + if (value !== this.style.letterSpacing) { + this.style.letterSpacing = value + const letterSpacing = value ?? STYLE_DEFAULTS.LETTER_SPACING + if (!this.isBitmapText(this.text)) { + if (this.text.style.letterSpacing !== letterSpacing) { + this.dirty = true + this.text.style.letterSpacing = letterSpacing + } + } else { + if (this.text.letterSpacing !== letterSpacing) { + this.dirty = true + this.text.letterSpacing = letterSpacing + } + } + } + } +} + +export { LabelBackground } from './background' export type { LabelStyle, LabelBackgroundStyle, LabelPosition } from './utils' diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index ce0abcec..11458342 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -8,11 +8,7 @@ import { TextStyleAlign, BitmapFont, ColorSource, - Sprite, - Texture, - BitmapText, - TextStyleFontWeight, - Rectangle + TextStyleFontWeight } from 'pixi.js' export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' @@ -28,8 +24,8 @@ export type LabelBackgroundStyle = { export type LabelStyle = Partial<{ fontName: string fontSize: number - wordWrap: number margin: number + wordWrap: number letterSpacing: number fontFamily: string | string[] fontWeight: TextStyleFontWeight @@ -47,14 +43,14 @@ export type StyleWithDefaults = Omit & { fontName: string } -export type LabelCoords = { x: number; y: number; offset?: number } - -const RESOLUTION = 2 +export const RESOLUTION = 2 export const STYLE_DEFAULTS = { FONT_SIZE: 10, STROKE_THICKNESS: 0, LETTER_SPACING: 0.5, WORD_WRAP: false, + MARGIN: 2, + OPACITY: 1, PADDING: [4, 8] as [number, number], STROKE: '#FFF', FONT_NAME: 'Label', @@ -97,6 +93,16 @@ const mergeDefaults = ({ ...style }) +const mergeBackgroundDefaults = ({ + color, + opacity = STYLE_DEFAULTS.OPACITY, + padding = STYLE_DEFAULTS.PADDING +}: LabelBackgroundStyle): Required => ({ + color, + opacity, + padding +}) + const isASCII = (str: string) => { for (const char of str) { if (char.codePointAt(0)! > 126) { @@ -107,11 +113,11 @@ const isASCII = (str: string) => { return true } -const getPositionAlign = (position: LabelPosition): TextStyleAlign => { +const getTextAlign = (position: LabelPosition): TextStyleAlign => { return position === 'left' || position === 'right' ? position : 'center' } -const getPositionAnchor = (position: LabelPosition): [x: number, y: number] => { +const getAnchorPoint = (position: LabelPosition): [x: number, y: number] => { switch (position) { case 'bottom': return [0.5, 0] @@ -147,7 +153,7 @@ const getTextStyle = ({ color, fontFamily, fontSize, fontWeight, wordWrap, strok style.strokeThickness = stroke.width } if (position !== STYLE_DEFAULTS.POSITION) { - style.align = getPositionAlign(position) + style.align = getTextAlign(position) } if (letterSpacing !== undefined) { style.letterSpacing = letterSpacing @@ -158,7 +164,7 @@ const getTextStyle = ({ color, fontFamily, fontSize, fontWeight, wordWrap, strok const getBitmapStyle = (style: StyleWithDefaults): Partial => ({ fontName: style.fontName, fontSize: style.fontSize, - align: getPositionAlign(style.position), + align: getTextAlign(style.position), letterSpacing: style.letterSpacing ?? STYLE_DEFAULTS.LETTER_SPACING }) @@ -171,47 +177,16 @@ const loadFont = (style: StyleWithDefaults) => { } } -const createTextObject = (label: string, style: StyleWithDefaults) => { - let text: BitmapText | Text - - if (isASCII(label)) { - loadFont(style) - text = new BitmapText(label, getBitmapStyle(style)) - } else { - text = new Text(label, getTextStyle(style)) - } - - text.anchor.set(...getPositionAnchor(style.position)) - return text -} - const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { return typeof padding === 'number' ? [padding, padding] : padding } -const setBackgroundSize = (sprite: Sprite, bounds: Rectangle, padding?: BackgroundPadding) => { - const [vertical, horizontal] = getBackgroundPadding(padding) - sprite.height = bounds.height + vertical - sprite.width = bounds.width + horizontal - return sprite -} - -const setBackgroundStyle = (sprite: Sprite, text: Text | BitmapText, { color, opacity = 1, padding }: LabelBackgroundStyle) => { - sprite.anchor.set(text.anchor.x, text.anchor.y) - sprite.alpha = opacity - sprite.tint = color - return setBackgroundSize(sprite, text.getLocalBounds(), padding) -} - -const createBackgroundSprite = (text: Text | BitmapText, style: LabelBackgroundStyle) => { - const sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) - return setBackgroundStyle(sprite, text, style) -} - const getLabelCoordinates = ( - { x, y, offset = 0 }: LabelCoords, - { position, margin = 2, background }: StyleWithDefaults, - isBitmapText: boolean + x: number, + y: number, + offset: number, + isBitmapText: boolean, + { position, background, margin = STYLE_DEFAULTS.MARGIN }: StyleWithDefaults ) => { const shift = margin + offset const label = { x, y } @@ -264,15 +239,12 @@ const getLabelCoordinates = ( export default { isASCII, mergeDefaults, + mergeBackgroundDefaults, getLabelCoordinates, - getPositionAlign, - getPositionAnchor, + getTextAlign, + getAnchorPoint, getTextStyle, getBitmapStyle, loadFont, - createTextObject, - setBackgroundSize, - setBackgroundStyle, - createBackgroundSprite, getBackgroundPadding } From bc587391b82569397e490b87457a837206ed5cbb Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Thu, 2 Nov 2023 11:18:39 -0400 Subject: [PATCH 08/10] add in some comparison perf gains for label updates --- examples/native/src/labels/index.ts | 4 +- src/renderers/webgl/node.ts | 2 +- .../webgl/objects/label/background.ts | 74 ++++--- src/renderers/webgl/objects/label/index.ts | 189 ++++++++---------- src/renderers/webgl/objects/label/utils.ts | 73 ++++--- 5 files changed, 170 insertions(+), 172 deletions(-) diff --git a/examples/native/src/labels/index.ts b/examples/native/src/labels/index.ts index 054c5701..feca6d38 100644 --- a/examples/native/src/labels/index.ts +++ b/examples/native/src/labels/index.ts @@ -22,7 +22,7 @@ const NODE_STYLE: Graph.NodeStyle = { label: { position: 'right', fontName: 'NodeLabel', - fontFamily: ['Arial', 'sans-serif'], + fontFamily: 'Arial, sans-serif', background: { color: GREEN_LIGHT }, margin: 4 } @@ -35,7 +35,7 @@ const NODE_HOVER_STYLE: Graph.NodeStyle = { label: { position: 'right', fontName: 'NodeLabelHover', - fontFamily: ['Arial', 'sans-serif'], + fontFamily: 'Arial, sans-serif', background: { color: DARK_GREEN }, color: '#FFF', margin: 4 diff --git a/src/renderers/webgl/node.ts b/src/renderers/webgl/node.ts index 5af2a8bb..cbffdc25 100644 --- a/src/renderers/webgl/node.ts +++ b/src/renderers/webgl/node.ts @@ -50,7 +50,7 @@ export class NodeRenderer { this.renderer.labelObjectManager.delete(this.label) this.labelMounted = false this.label = undefined - } else { + } else if (!this.label.equals(node.label, node.style?.label)) { this.label.update(node.label, node.style?.label) } diff --git a/src/renderers/webgl/objects/label/background.ts b/src/renderers/webgl/objects/label/background.ts index f7b7e036..6bae1b0d 100644 --- a/src/renderers/webgl/objects/label/background.ts +++ b/src/renderers/webgl/objects/label/background.ts @@ -1,6 +1,6 @@ -import utils, { RESOLUTION, STYLE_DEFAULTS } from './utils' -import type { BackgroundPadding, LabelBackgroundStyle } from './utils' -import { BitmapText, ColorSource, Container, Rectangle, Sprite, Text, Texture } from 'pixi.js' +import utils, { STYLE_DEFAULTS } from './utils' +import type { LabelBackgroundStyle } from './utils' +import { BitmapText, ColorSource, Container, Point, Rectangle, Sprite, Text, Texture } from 'pixi.js' import { equals } from '../../../..' export class LabelBackground { @@ -12,28 +12,39 @@ export class LabelBackground { private sprite: Sprite private label: Text | BitmapText private container: Container - private style: Required private rect: Rectangle + private _style: LabelBackgroundStyle constructor(container: Container, label: Text | BitmapText, style: LabelBackgroundStyle) { this.label = label this.container = container - this.style = utils.mergeBackgroundDefaults(style) - this.sprite = Sprite.from(Texture.WHITE, { resolution: RESOLUTION }) + this._style = style + this.rect = this.label.getLocalBounds() + + const { width, height } = this.size + + this.sprite = Sprite.from(Texture.WHITE) + this.sprite.height = height + this.sprite.width = width this.sprite.anchor.set(this.label.anchor.x, this.label.anchor.y) this.sprite.alpha = this.style.opacity this.sprite.tint = this.style.color - this.rect = label.getLocalBounds() - this.resize() } update(label: Text | BitmapText, style: LabelBackgroundStyle) { - this.label = label - this.color = style.color + this.dirty = !equals(style.padding, this._style.padding) this.bounds = label.getLocalBounds() - this.opacity = style.opacity ?? STYLE_DEFAULTS.OPACITY - this.padding = style.padding ?? STYLE_DEFAULTS.PADDING - this.sprite.anchor.set(this.label.anchor.x, this.label.anchor.y) + this.anchor = label.anchor.clone() + + if (this.label !== label) { + this.label = label + } + + if (this._style !== style) { + this._style = style + this.color = style.color + this.opacity = style.opacity ?? STYLE_DEFAULTS.OPACITY + } if (this.dirty) { this.dirty = false @@ -81,42 +92,45 @@ export class LabelBackground { } private resize() { - const [vertical, horizontal] = utils.getBackgroundPadding(this.style.padding) - - const height = this.rect.height + vertical + const { height, width } = this.size if (height !== this.sprite.height) { this.sprite.height = height } - - const width = this.rect.width + horizontal if (width !== this.sprite.width) { this.sprite.width = width } - return this } + private get style() { + return utils.mergeBackgroundDefaults(this._style) + } + + private get size() { + const [top, right, bottom, left] = utils.getBackgroundPadding(this._style.padding) + const height = this.rect.height + top + bottom + const width = this.rect.width + right + left + return { width, height } + } + + private set anchor(anchor: Point) { + if (!this.sprite.anchor.equals(anchor)) { + this.sprite.anchor.copyFrom(anchor) + } + } + private set color(color: ColorSource) { - if (color !== this.style.color) { - this.style.color = color + if (this.sprite.tint !== color) { this.sprite.tint = color } } private set opacity(opacity: number) { - if (opacity !== this.style.opacity) { - this.style.opacity = opacity + if (this.sprite.alpha !== opacity) { this.sprite.alpha = opacity } } - private set padding(padding: BackgroundPadding) { - if (!equals(padding, this.style.padding)) { - this.style.padding = padding - this.dirty = true - } - } - private set bounds(bounds: Rectangle) { if (this.rect.width !== bounds.width || this.rect.height !== bounds.height) { this.rect = bounds diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts index 0f124800..a41334d9 100644 --- a/src/renderers/webgl/objects/label/index.ts +++ b/src/renderers/webgl/objects/label/index.ts @@ -1,9 +1,8 @@ import utils, { STYLE_DEFAULTS } from './utils' -import type { StyleWithDefaults, LabelPosition, LabelStyle, LabelBackgroundStyle } from './utils' +import type { LabelPosition, LabelStyle, LabelBackgroundStyle } from './utils' import type { Stroke } from '../../../../types' import { BitmapText, Container, Text, TextStyleAlign, TextStyleFill, TextStyleFontWeight } from 'pixi.js' import { LabelBackground } from './background' -import { equals } from '../../../..' /** * TODO @@ -21,17 +20,18 @@ export class Label { private label: string private container: Container private text: BitmapText | Text - private style: StyleWithDefaults private labelBackground: LabelBackground | null = null + private _style: LabelStyle | undefined constructor(container: Container, label: string, style: LabelStyle | undefined) { - this.label = label this.container = container - this.style = utils.mergeDefaults(style) + this.label = label + this._style = style if (utils.isASCII(this.label)) { utils.loadFont(this.style) this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) + this.text.resolution = 2 } else { this.text = new Text(this.label, utils.getTextStyle(this.style)) } @@ -43,32 +43,45 @@ export class Label { } update(label: string, style: LabelStyle | undefined) { - this.value = label + const labelHasChanged = this.label !== label + const styleHasChanged = this._style !== style + + this._style = style - const isBitmapText = this.isBitmapText() - const isASCII = utils.isASCII(this.label) - // if the text type has changed, regenerate a new text object - if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { - this.transformText(this.label, utils.mergeDefaults(style)) + if (labelHasChanged) { + this.label = label + this.text.text = label + + const isBitmapText = this.isBitmapText() + const isASCII = utils.isASCII(label) + // if the text type has changed, regenerate a new text object + if ((isBitmapText && !isASCII) || (!isBitmapText && isASCII)) { + this.transformText() + } } - this.wordWrap = style?.wordWrap - this.color = style?.color - this.stroke = style?.stroke - this.fontWeight = style?.fontWeight - this.letterSpacing = style?.letterSpacing - this.position = style?.position ?? STYLE_DEFAULTS.POSITION - this.fontSize = style?.fontSize ?? STYLE_DEFAULTS.FONT_SIZE - this.fontFamily = style?.fontFamily ?? STYLE_DEFAULTS.FONT_FAMILY - this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME - this.background = style?.background + if (styleHasChanged) { + this.stroke = style?.stroke + this.wordWrap = style?.wordWrap + this.color = style?.color ?? STYLE_DEFAULTS.COLOR + this.fontWeight = style?.fontWeight ?? STYLE_DEFAULTS.FONT_WEIGHT + this.letterSpacing = style?.letterSpacing ?? STYLE_DEFAULTS.LETTER_SPACING + this.position = style?.position ?? STYLE_DEFAULTS.POSITION + this.fontSize = style?.fontSize ?? STYLE_DEFAULTS.FONT_SIZE + this.fontFamily = style?.fontFamily ?? STYLE_DEFAULTS.FONT_FAMILY + this.fontName = style?.fontName ?? STYLE_DEFAULTS.FONT_NAME + } if (this.dirty) { this.dirty = false this.updateText() } - this.transformed = false + if (labelHasChanged || styleHasChanged) { + this.transformed = false + this.background = style?.background + } + return this } @@ -117,6 +130,10 @@ export class Label { return undefined } + equals(label: string, style: LabelStyle | undefined) { + return this.label === label && this._style === style + } + private isBitmapText(text: Text | BitmapText = this.text): text is BitmapText { return text instanceof BitmapText } @@ -129,20 +146,21 @@ export class Label { } } - private transformText(label: string, style: StyleWithDefaults) { + private transformText() { this.transformed = true const isMounted = this.mounted this.delete() - if (utils.isASCII(label)) { - utils.loadFont(style) - this.text = new BitmapText(label, utils.getBitmapStyle(style)) + if (utils.isASCII(this.label)) { + utils.loadFont(this.style) + this.text = new BitmapText(this.label, utils.getBitmapStyle(this.style)) + this.text.resolution = 2 } else { - this.text = new Text(label, utils.getTextStyle(style)) + this.text = new Text(this.label, utils.getTextStyle(this.style)) } - this.text.anchor.set(...utils.getAnchorPoint(style.position)) + this.anchor = utils.getAnchorPoint(this.style.position) this.text.x = this.x ?? 0 this.text.y = this.y ?? 0 @@ -151,17 +169,13 @@ export class Label { } } - private set value(label: string) { - if (label !== this.label) { - this.text.text = label - this.label = label - } + private get style() { + return utils.mergeDefaults(this._style) } private set position(position: LabelPosition) { this.align = utils.getTextAlign(position) this.anchor = utils.getAnchorPoint(position) - this.style.position = position } private set align(align: TextStyleAlign) { @@ -185,51 +199,40 @@ export class Label { } private set fontSize(fontSize: number) { - if (fontSize !== this.style.fontSize) { - this.style.fontSize = fontSize - if (this.isBitmapText(this.text)) { - if (this.text.fontSize !== fontSize) { - this.dirty = true - this.text.fontSize = fontSize - } - } else { - if (this.text.style.fontSize !== fontSize) { - this.dirty = true - this.text.style.fontSize = fontSize - } + if (this.isBitmapText(this.text)) { + if (this.text.fontSize !== fontSize) { + this.dirty = true + this.text.fontSize = fontSize + } + } else { + if (this.text.style.fontSize !== fontSize) { + this.dirty = true + this.text.style.fontSize = fontSize } } } private set wordWrap(wordWrap: number | undefined) { - if (wordWrap !== this.style.wordWrap) { - this.style.wordWrap = wordWrap - const wordWrapWidth = wordWrap ?? 0 - if (!this.isBitmapText(this.text) && this.text.style.wordWrapWidth !== wordWrapWidth) { - this.dirty = true - this.text.style.wordWrap = wordWrap !== undefined - this.text.style.wordWrapWidth = wordWrapWidth - } + const wordWrapWidth = wordWrap ?? 0 + if (!this.isBitmapText(this.text) && this.text.style.wordWrapWidth !== wordWrapWidth) { + this.dirty = true + this.text.style.wordWrap = wordWrap !== undefined + this.text.style.wordWrapWidth = wordWrapWidth } } - private set color(value: TextStyleFill | undefined) { - if (!equals(value, this.style.color)) { - this.style.color = value - const color = value ?? STYLE_DEFAULTS.COLOR - if (!this.isBitmapText(this.text) && this.text.style.fill !== color) { - this.dirty = true - this.text.style.fill = color - } + private set color(color: TextStyleFill) { + if (!this.isBitmapText(this.text) && this.text.style.fill !== color) { + this.dirty = true + this.text.style.fill = color } } private set stroke(value: Stroke | undefined) { - if (!equals(value, this.style.stroke)) { - this.style.stroke = value + if (!this.isBitmapText(this.text)) { const stroke = value?.color ?? STYLE_DEFAULTS.STROKE const strokeThickness = value?.width ?? STYLE_DEFAULTS.STROKE_THICKNESS - if (!this.isBitmapText(this.text) && (this.text.style.stroke !== stroke || this.text.style.strokeThickness !== strokeThickness)) { + if (this.text.style.stroke !== stroke || this.text.style.strokeThickness !== strokeThickness) { this.dirty = true this.text.style.stroke = stroke this.text.style.strokeThickness = strokeThickness @@ -237,40 +240,28 @@ export class Label { } } - private set fontFamily(fontFamily: string | string[]) { - if (!equals(fontFamily, this.style.fontFamily)) { - this.style.fontFamily = fontFamily - if (!this.isBitmapText(this.text) && !equals(this.text.style.fontFamily, fontFamily)) { - this.dirty = true - this.text.style.fontFamily = fontFamily - } + private set fontFamily(fontFamily: string) { + if (!this.isBitmapText(this.text) && fontFamily !== this.text.style.fontFamily) { + this.dirty = true + this.text.style.fontFamily = fontFamily } } private set fontName(fontName: string) { - if (fontName !== this.style.fontName) { - this.style.fontName = fontName - if (this.isBitmapText(this.text) && this.text.fontName !== fontName) { - this.dirty = true - this.text.fontName = fontName - } + if (this.isBitmapText(this.text) && this.text.fontName !== fontName) { + this.dirty = true + this.text.fontName = fontName } } - private set fontWeight(value: TextStyleFontWeight | undefined) { - if (value !== this.style.fontWeight) { - this.style.fontWeight = value - const fontWeight = value ?? STYLE_DEFAULTS.FONT_WEIGHT - if (!this.isBitmapText(this.text) && this.text.style.fontWeight !== fontWeight) { - this.dirty = true - this.text.style.fontWeight = fontWeight - } + private set fontWeight(fontWeight: TextStyleFontWeight) { + if (!this.isBitmapText(this.text) && this.text.style.fontWeight !== fontWeight) { + this.dirty = true + this.text.style.fontWeight = fontWeight } } private set background(background: LabelBackgroundStyle | undefined) { - this.style.background = background - if (this.labelBackground === null && background !== undefined) { this.labelBackground = new LabelBackground(this.container, this.text, background) } else if (this.labelBackground && background !== undefined) { @@ -281,21 +272,15 @@ export class Label { } } - private set letterSpacing(value: number | undefined) { - if (value !== this.style.letterSpacing) { - this.style.letterSpacing = value - const letterSpacing = value ?? STYLE_DEFAULTS.LETTER_SPACING - if (!this.isBitmapText(this.text)) { - if (this.text.style.letterSpacing !== letterSpacing) { - this.dirty = true - this.text.style.letterSpacing = letterSpacing - } - } else { - if (this.text.letterSpacing !== letterSpacing) { - this.dirty = true - this.text.letterSpacing = letterSpacing - } + private set letterSpacing(letterSpacing: number) { + if (!this.isBitmapText(this.text)) { + if (this.text.style.letterSpacing !== letterSpacing) { + this.dirty = true + this.text.style.letterSpacing = letterSpacing } + } else if (this.text.letterSpacing !== letterSpacing) { + this.dirty = true + this.text.letterSpacing = letterSpacing } } } diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index 11458342..c00d99aa 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,3 +1,4 @@ +import { MIN_ZOOM } from '../..' import type { Stroke } from '../../../../types' import { Text, @@ -8,17 +9,16 @@ import { TextStyleAlign, BitmapFont, ColorSource, - TextStyleFontWeight + TextStyleFontWeight, + LINE_JOIN } from 'pixi.js' export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' -export type BackgroundPadding = number | [vertical: number, horizontal: number] - export type LabelBackgroundStyle = { color: ColorSource opacity?: number - padding?: BackgroundPadding + padding?: number | number[] } export type LabelStyle = Partial<{ @@ -27,7 +27,7 @@ export type LabelStyle = Partial<{ margin: number wordWrap: number letterSpacing: number - fontFamily: string | string[] + fontFamily: string fontWeight: TextStyleFontWeight stroke: Stroke color: TextStyleFill @@ -35,12 +35,13 @@ export type LabelStyle = Partial<{ background: LabelBackgroundStyle }> -type _StyleDefaults = 'fontSize' | 'position' | 'fontFamily' | 'fontName' +type _StyleDefaults = 'fontSize' | 'position' | 'fontFamily' | 'fontName' | 'margin' export type StyleWithDefaults = Omit & { fontSize: number position: LabelPosition fontFamily: string | string[] fontName: string + margin: number } export const RESOLUTION = 2 @@ -57,9 +58,8 @@ export const STYLE_DEFAULTS = { COLOR: '#000000', ALIGN: 'center' as const, POSITION: 'bottom' as const, - LINE_JOIN: 'round' as const, FONT_WEIGHT: 'normal' as const, - FONT_FAMILY: ['Arial', 'sans-serif'] + FONT_FAMILY: 'Arial, sans-serif' } // install text defaults @@ -70,7 +70,7 @@ TextStyle.defaultStyle = { align: STYLE_DEFAULTS.ALIGN, fill: STYLE_DEFAULTS.COLOR, stroke: STYLE_DEFAULTS.STROKE, - lineJoin: STYLE_DEFAULTS.LINE_JOIN, + lineJoin: LINE_JOIN.ROUND, wordWrap: STYLE_DEFAULTS.WORD_WRAP, fontSize: STYLE_DEFAULTS.FONT_SIZE, fontFamily: STYLE_DEFAULTS.FONT_FAMILY, @@ -84,12 +84,14 @@ const mergeDefaults = ({ fontSize = STYLE_DEFAULTS.FONT_SIZE, fontFamily = STYLE_DEFAULTS.FONT_FAMILY, fontName = STYLE_DEFAULTS.FONT_NAME, + margin = STYLE_DEFAULTS.MARGIN, ...style }: LabelStyle = {}): StyleWithDefaults => ({ position, fontSize, fontFamily, fontName, + margin, ...style }) @@ -170,15 +172,17 @@ const getBitmapStyle = (style: StyleWithDefaults): Partial => const loadFont = (style: StyleWithDefaults) => { if (BitmapFont.available[style.fontName] === undefined) { - BitmapFont.from(style.fontName, getTextStyle(style), { - resolution: RESOLUTION, + BitmapFont.from(style.fontName, getTextStyle({ ...style, fontSize: style.fontSize * RESOLUTION * MIN_ZOOM }), { chars: BitmapFont.ASCII }) } } -const getBackgroundPadding = (padding: BackgroundPadding = STYLE_DEFAULTS.PADDING): [vertical: number, horizontal: number] => { - return typeof padding === 'number' ? [padding, padding] : padding +const getBackgroundPadding = ( + padding: number | number[] = STYLE_DEFAULTS.PADDING +): [top: number, right: number, bottom: number, left: number] => { + const [top, right = top, bottom = top, left = right]: number[] = typeof padding === 'number' ? [padding] : padding + return [top, right, bottom, left] } const getLabelCoordinates = ( @@ -186,50 +190,45 @@ const getLabelCoordinates = ( y: number, offset: number, isBitmapText: boolean, - { position, background, margin = STYLE_DEFAULTS.MARGIN }: StyleWithDefaults + { position, background, margin }: StyleWithDefaults ) => { const shift = margin + offset const label = { x, y } const bg = { x, y } - let vertical = 0 - let horizontal = 0 + let top = 0 + let right = 0 + let bottom = 0 + let left = 0 if (background !== undefined) { - const [v, h] = getBackgroundPadding(background.padding) - vertical += v / 2 - horizontal += h / 2 + const [t, r, b, l] = getBackgroundPadding(background.padding) + top += t + right += r + bottom += b + left += l + } + + if (isBitmapText && (position === 'left' || position === 'right')) { + label.y -= 1 + bg.y -= 1 } switch (position) { case 'bottom': - label.y += shift + vertical + label.y += shift + top bg.y += shift - break case 'left': - label.x -= shift + horizontal + label.x -= shift + right bg.x -= shift - - if (isBitmapText) { - label.y -= 1 - bg.y -= 1 - } - break case 'top': - label.y -= shift + vertical + label.y -= shift + bottom bg.y -= shift - break case 'right': - label.x += shift + horizontal + label.x += shift + left bg.x += shift - - if (isBitmapText) { - label.y -= 1 - bg.y -= 1 - } - break } From 8d0520e3c5fea42c94cb67da838f3de45f7b2e34 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Mon, 6 Nov 2023 15:42:57 -0500 Subject: [PATCH 09/10] remove PIXI types from trellis types --- src/index.ts | 2 +- src/renderers/webgl/objects/label/index.ts | 12 +++++------ src/renderers/webgl/objects/label/utils.ts | 23 +++++++--------------- 3 files changed, 14 insertions(+), 23 deletions(-) diff --git a/src/index.ts b/src/index.ts index 332179cc..8d07f884 100644 --- a/src/index.ts +++ b/src/index.ts @@ -381,4 +381,4 @@ export const angle = (x0: number, y0: number, x1: number, y1: number) => { // exports export type { Stroke } from './types' -export type { LabelStyle, LabelBackgroundStyle, LabelPosition } from './renderers/webgl/objects/label' +export type { LabelStyle, LabelBackgroundStyle, LabelPosition, FontWeight, TextAlign } from './renderers/webgl/objects/label' diff --git a/src/renderers/webgl/objects/label/index.ts b/src/renderers/webgl/objects/label/index.ts index a41334d9..7b229fe6 100644 --- a/src/renderers/webgl/objects/label/index.ts +++ b/src/renderers/webgl/objects/label/index.ts @@ -1,7 +1,7 @@ import utils, { STYLE_DEFAULTS } from './utils' -import type { LabelPosition, LabelStyle, LabelBackgroundStyle } from './utils' +import type { LabelPosition, LabelStyle, LabelBackgroundStyle, TextAlign, FontWeight } from './utils' import type { Stroke } from '../../../../types' -import { BitmapText, Container, Text, TextStyleAlign, TextStyleFill, TextStyleFontWeight } from 'pixi.js' +import { BitmapText, Container, Text } from 'pixi.js' import { LabelBackground } from './background' /** @@ -178,7 +178,7 @@ export class Label { this.anchor = utils.getAnchorPoint(position) } - private set align(align: TextStyleAlign) { + private set align(align: TextAlign) { if (this.isBitmapText(this.text)) { if (this.text.align !== align) { this.dirty = true @@ -221,7 +221,7 @@ export class Label { } } - private set color(color: TextStyleFill) { + private set color(color: string) { if (!this.isBitmapText(this.text) && this.text.style.fill !== color) { this.dirty = true this.text.style.fill = color @@ -254,7 +254,7 @@ export class Label { } } - private set fontWeight(fontWeight: TextStyleFontWeight) { + private set fontWeight(fontWeight: FontWeight) { if (!this.isBitmapText(this.text) && this.text.style.fontWeight !== fontWeight) { this.dirty = true this.text.style.fontWeight = fontWeight @@ -286,4 +286,4 @@ export class Label { } export { LabelBackground } from './background' -export type { LabelStyle, LabelBackgroundStyle, LabelPosition } from './utils' +export type { LabelStyle, LabelBackgroundStyle, LabelPosition, FontWeight, TextAlign } from './utils' diff --git a/src/renderers/webgl/objects/label/utils.ts b/src/renderers/webgl/objects/label/utils.ts index c00d99aa..ee8ce8a4 100644 --- a/src/renderers/webgl/objects/label/utils.ts +++ b/src/renderers/webgl/objects/label/utils.ts @@ -1,22 +1,13 @@ import { MIN_ZOOM } from '../..' import type { Stroke } from '../../../../types' -import { - Text, - TextStyle, - TextStyleFill, - ITextStyle, - IBitmapTextStyle, - TextStyleAlign, - BitmapFont, - ColorSource, - TextStyleFontWeight, - LINE_JOIN -} from 'pixi.js' +import { Text, TextStyle, ITextStyle, IBitmapTextStyle, BitmapFont, LINE_JOIN } from 'pixi.js' +export type TextAlign = 'left' | 'center' | 'right' | 'justify' +export type FontWeight = 'normal' | 'bold' | 'bolder' | 'lighter' | '100' | '200' | '300' | '400' | '500' | '600' | '700' | '800' | '900' export type LabelPosition = 'bottom' | 'left' | 'top' | 'right' export type LabelBackgroundStyle = { - color: ColorSource + color: string opacity?: number padding?: number | number[] } @@ -28,9 +19,9 @@ export type LabelStyle = Partial<{ wordWrap: number letterSpacing: number fontFamily: string - fontWeight: TextStyleFontWeight + fontWeight: FontWeight stroke: Stroke - color: TextStyleFill + color: string position: LabelPosition background: LabelBackgroundStyle }> @@ -115,7 +106,7 @@ const isASCII = (str: string) => { return true } -const getTextAlign = (position: LabelPosition): TextStyleAlign => { +const getTextAlign = (position: LabelPosition): TextAlign => { return position === 'left' || position === 'right' ? position : 'center' } From 83d44c97fbd7359e15e6c00e04bcfd3f7fbbf455 Mon Sep 17 00:00:00 2001 From: Mikey Gower Date: Mon, 6 Nov 2023 15:43:50 -0500 Subject: [PATCH 10/10] resolve type error in react button --- src/bindings/react/button.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/bindings/react/button.ts b/src/bindings/react/button.ts index 51bae3f0..84f5bb02 100644 --- a/src/bindings/react/button.ts +++ b/src/bindings/react/button.ts @@ -6,6 +6,7 @@ export type Props = { group?: 'top' | 'middle' | 'bottom' title?: string onClick?: () => void + children?: React.ReactNode } const STYLE = {