chat-layout 1.1.0 → 1.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (3) hide show
  1. package/index.mjs +79 -10
  2. package/index.mjs.map +1 -1
  3. package/package.json +1 -1
package/index.mjs CHANGED
@@ -1,5 +1,5 @@
1
1
  import { layoutNextLine, layoutWithLines, measureLineStats, measureNaturalWidth, prepareWithSegments } from "@chenglou/pretext";
2
- import { layoutNextRichInlineLineRange, materializeRichInlineLineRange, measureRichInlineStats, prepareRichInline, walkRichInlineLineRanges } from "@chenglou/pretext/rich-inline";
2
+ import { layoutNextRichInlineLineRange, materializeRichInlineLineRange, measureRichInlineStats, prepareRichInline } from "@chenglou/pretext/rich-inline";
3
3
  //#region src/internal/node-registry.ts
4
4
  const registry = /* @__PURE__ */ new WeakMap();
5
5
  const revisions = /* @__PURE__ */ new WeakMap();
@@ -1231,6 +1231,8 @@ function layoutTextWithOverflow(ctx, text, maxWidth, options = {}) {
1231
1231
  //#endregion
1232
1232
  //#region src/text/rich.ts
1233
1233
  const RICH_PREPARED_CACHE_CAPACITY = 256;
1234
+ const LEADING_COLLAPSIBLE_BOUNDARY_RE = /^[ \t\n\f\r]+/;
1235
+ const TRAILING_COLLAPSIBLE_BOUNDARY_RE = /[ \t\n\f\r]+$/;
1234
1236
  const richPreparedCache = /* @__PURE__ */ new Map();
1235
1237
  function withFont(ctx, font, cb) {
1236
1238
  const previousFont = ctx.graphics.font;
@@ -1248,18 +1250,85 @@ function readRichPrepared(spans, defaultFont) {
1248
1250
  const key = getRichPreparedCacheKey(spans, defaultFont);
1249
1251
  const cached = readLruValue(richPreparedCache, key);
1250
1252
  if (cached != null) return cached;
1251
- return writeLruValue(richPreparedCache, key, prepareRichInline(spans.map((span) => ({
1253
+ const items = spans.map((span) => ({
1252
1254
  text: span.text,
1253
1255
  font: span.font ?? defaultFont,
1254
1256
  break: span.break,
1255
1257
  extraWidth: span.extraWidth
1256
- }))), RICH_PREPARED_CACHE_CAPACITY);
1258
+ }));
1259
+ const preparedItemIndexBySourceItemIndex = buildPreparedItemIndexBySourceItemIndex(spans);
1260
+ return writeLruValue(richPreparedCache, key, {
1261
+ prepared: prepareRichInline(items),
1262
+ preparedItemIndexBySourceItemIndex
1263
+ }, RICH_PREPARED_CACHE_CAPACITY);
1264
+ }
1265
+ function trimRichInlineBoundaryWhitespace(text) {
1266
+ return text.replace(LEADING_COLLAPSIBLE_BOUNDARY_RE, "").replace(TRAILING_COLLAPSIBLE_BOUNDARY_RE, "");
1267
+ }
1268
+ function buildPreparedItemIndexBySourceItemIndex(spans) {
1269
+ const preparedItemIndexBySourceItemIndex = Array.from({ length: spans.length });
1270
+ let preparedItemIndex = 0;
1271
+ for (let index = 0; index < spans.length; index += 1) {
1272
+ if (trimRichInlineBoundaryWhitespace(spans[index].text).length === 0) continue;
1273
+ preparedItemIndexBySourceItemIndex[index] = preparedItemIndex;
1274
+ preparedItemIndex += 1;
1275
+ }
1276
+ return preparedItemIndexBySourceItemIndex;
1277
+ }
1278
+ function getRichFragmentStartCursor(prepared, fragment) {
1279
+ const itemIndex = prepared.preparedItemIndexBySourceItemIndex[fragment.itemIndex];
1280
+ if (itemIndex == null) return null;
1281
+ return {
1282
+ itemIndex,
1283
+ segmentIndex: fragment.start.segmentIndex,
1284
+ graphemeIndex: fragment.start.graphemeIndex
1285
+ };
1286
+ }
1287
+ function splitOverflowingRichLineRange(prepared, lineRange, maxWidth) {
1288
+ if (lineRange.width <= maxWidth || lineRange.fragments.length <= 1) return lineRange;
1289
+ const trailingFragment = lineRange.fragments[lineRange.fragments.length - 1];
1290
+ const splitCursor = getRichFragmentStartCursor(prepared, trailingFragment);
1291
+ if (splitCursor == null) return lineRange;
1292
+ const fragments = lineRange.fragments.slice(0, -1);
1293
+ return {
1294
+ fragments,
1295
+ width: fragments.reduce((total, fragment) => total + fragment.gapBefore + fragment.occupiedWidth, 0),
1296
+ end: splitCursor
1297
+ };
1298
+ }
1299
+ function layoutNextConstrainedRichInlineLineRange(prepared, maxWidth, start) {
1300
+ const lineRange = layoutNextRichInlineLineRange(prepared.prepared, maxWidth, start);
1301
+ if (lineRange == null) return null;
1302
+ return splitOverflowingRichLineRange(prepared, lineRange, maxWidth);
1303
+ }
1304
+ function walkConstrainedRichInlineLineRanges(prepared, maxWidth, onLine) {
1305
+ let lineCount = 0;
1306
+ let cursor;
1307
+ while (true) {
1308
+ const lineRange = layoutNextConstrainedRichInlineLineRange(prepared, maxWidth, cursor);
1309
+ if (lineRange == null) return lineCount;
1310
+ onLine(lineRange);
1311
+ lineCount += 1;
1312
+ cursor = lineRange.end;
1313
+ }
1314
+ }
1315
+ function measureConstrainedRichInlineStats(prepared, maxWidth) {
1316
+ let lineCount = 0;
1317
+ let maxLineWidth = 0;
1318
+ walkConstrainedRichInlineLineRanges(prepared, maxWidth, (lineRange) => {
1319
+ lineCount += 1;
1320
+ if (lineRange.width > maxLineWidth) maxLineWidth = lineRange.width;
1321
+ });
1322
+ return {
1323
+ lineCount,
1324
+ maxLineWidth
1325
+ };
1257
1326
  }
1258
1327
  function measureRichFragmentShift(ctx, font) {
1259
1328
  return withFont(ctx, font, () => measureFontShift(ctx));
1260
1329
  }
1261
1330
  function materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, overflowed) {
1262
- const richLine = materializeRichInlineLineRange(readRichPrepared(spans, defaultFont), lineRange);
1331
+ const richLine = materializeRichInlineLineRange(readRichPrepared(spans, defaultFont).prepared, lineRange);
1263
1332
  const fragments = richLine.fragments.map((fragment) => {
1264
1333
  const span = spans[fragment.itemIndex];
1265
1334
  const font = span?.font ?? defaultFont;
@@ -1412,7 +1481,7 @@ function layoutRichFirstLineIntrinsic(ctx, spans, defaultFont, defaultColor) {
1412
1481
  fragments: [],
1413
1482
  overflowed: false
1414
1483
  };
1415
- const lineRange = layoutNextRichInlineLineRange(readRichPrepared(spans, defaultFont), INTRINSIC_MAX_WIDTH);
1484
+ const lineRange = layoutNextRichInlineLineRange(readRichPrepared(spans, defaultFont).prepared, INTRINSIC_MAX_WIDTH);
1416
1485
  if (lineRange == null) return {
1417
1486
  width: 0,
1418
1487
  fragments: [],
@@ -1427,7 +1496,7 @@ function layoutRichFirstLine(ctx, spans, maxWidth, defaultFont, defaultColor) {
1427
1496
  fragments: [],
1428
1497
  overflowed: false
1429
1498
  };
1430
- const lineRange = layoutNextRichInlineLineRange(readRichPrepared(spans, defaultFont), clampedMaxWidth);
1499
+ const lineRange = layoutNextConstrainedRichInlineLineRange(readRichPrepared(spans, defaultFont), clampedMaxWidth);
1431
1500
  if (lineRange == null) return {
1432
1501
  width: 0,
1433
1502
  fragments: [],
@@ -1454,7 +1523,7 @@ function measureRichText(_ctx, spans, maxWidth, defaultFont) {
1454
1523
  width: 0,
1455
1524
  lineCount: 0
1456
1525
  };
1457
- const { maxLineWidth: width, lineCount } = measureRichInlineStats(readRichPrepared(spans, defaultFont), maxWidth);
1526
+ const { maxLineWidth: width, lineCount } = measureConstrainedRichInlineStats(readRichPrepared(spans, defaultFont), maxWidth);
1458
1527
  return {
1459
1528
  width,
1460
1529
  lineCount
@@ -1465,7 +1534,7 @@ function measureRichTextIntrinsic(_ctx, spans, defaultFont) {
1465
1534
  width: 0,
1466
1535
  lineCount: 0
1467
1536
  };
1468
- const { maxLineWidth: width, lineCount } = measureRichInlineStats(readRichPrepared(spans, defaultFont), INTRINSIC_MAX_WIDTH);
1537
+ const { maxLineWidth: width, lineCount } = measureRichInlineStats(readRichPrepared(spans, defaultFont).prepared, INTRINSIC_MAX_WIDTH);
1469
1538
  return {
1470
1539
  width,
1471
1540
  lineCount
@@ -1487,7 +1556,7 @@ function measureRichTextMinContent(_ctx, spans, defaultFont, overflowWrap = "bre
1487
1556
  width: 0,
1488
1557
  lineCount: 0
1489
1558
  };
1490
- const { lineCount } = measureRichInlineStats(readRichPrepared(spans, defaultFont), Math.max(maxWidth, MIN_CONTENT_WIDTH_EPSILON));
1559
+ const { lineCount } = measureConstrainedRichInlineStats(readRichPrepared(spans, defaultFont), Math.max(maxWidth, MIN_CONTENT_WIDTH_EPSILON));
1491
1560
  return {
1492
1561
  width: maxWidth,
1493
1562
  lineCount
@@ -1501,7 +1570,7 @@ function layoutRichText(ctx, spans, maxWidth, defaultFont, defaultColor) {
1501
1570
  };
1502
1571
  const prepared = readRichPrepared(spans, defaultFont);
1503
1572
  const lineRanges = [];
1504
- walkRichInlineLineRanges(prepared, maxWidth, (lineRange) => lineRanges.push(lineRange));
1573
+ walkConstrainedRichInlineLineRanges(prepared, maxWidth, (lineRange) => lineRanges.push(lineRange));
1505
1574
  if (lineRanges.length === 0) return {
1506
1575
  width: 0,
1507
1576
  lines: [],
package/index.mjs.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","names":["#children","#inner","clampToConstraints","#top","#bottom","#left","#right","LINE_START_CURSOR","#ctx","#lastWidth","#cache","#layoutCache","#textLayoutCache","#clearAllCaches","#syncCachesToViewportWidth","#items","clamp","#unsubscribeListState","#handleListStateChange","#cancelJumpAnimation","#jumpAnimation","#controlledState","#prepareReplacementAnimations","#readReplacementAnimation","#sampleReplacementHeight","#sampleLayerAlpha","#drawReplacementLayers","#nextReplacementLayerKey","#replacementAnimations","#isLayerComplete","#handleReplace","#createReplacementLayer","clamp","clamp","#resolveVisibleWindow","#resolveVisibleWindow"],"sources":["../src/internal/node-registry.ts","../src/utils.ts","../src/nodes/base.ts","../src/layout.ts","../src/nodes/shared.ts","../src/nodes/box.ts","../src/nodes/flex.ts","../src/nodes/place.ts","../src/text/core.ts","../src/text/plain-core.ts","../src/text/plain.ts","../src/text/rich.ts","../src/nodes/text.ts","../src/renderer/base.ts","../src/renderer/list-state.ts","../src/renderer/memo.ts","../src/renderer/virtualized/base.ts","../src/renderer/virtualized/solver.ts","../src/renderer/virtualized/chat.ts","../src/renderer/virtualized/timeline.ts"],"sourcesContent":["import type { Node } from \"../types\";\n\nconst registry = new WeakMap<Node<any>, Node<any>>();\nconst revisions = new WeakMap<Node<any>, number>();\n\nfunction getOwnershipError(): Error {\n return new Error(\"A node can only be attached to one parent. Shared nodes are not supported.\");\n}\n\nfunction getDetachOwnershipError(): Error {\n return new Error(\"Cannot detach or replace a node from a parent that does not own it.\");\n}\n\nfunction bumpRevision(node: Node<any>): void {\n revisions.set(node, (revisions.get(node) ?? 0) + 1);\n}\n\nexport function getNodeRevision<C extends CanvasRenderingContext2D>(node: Node<C>): number {\n return revisions.get(node) ?? 0;\n}\n\nexport function attachNodeToParent<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n parent: Node<C>,\n): void {\n if (registry.has(node)) {\n throw getOwnershipError();\n }\n registry.set(node, parent);\n bumpRevision(parent);\n}\n\nexport function attachNodesToParent<C extends CanvasRenderingContext2D>(\n nodes: Iterable<Node<C>>,\n parent: Node<C>,\n): void {\n for (const node of nodes) {\n attachNodeToParent(node, parent);\n }\n}\n\nexport function detachNodeFromParent<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n parent?: Node<C>,\n): void {\n const currentParent = registry.get(node);\n if (currentParent == null) {\n return;\n }\n if (parent != null && currentParent !== parent) {\n throw getDetachOwnershipError();\n }\n registry.delete(node);\n bumpRevision(currentParent);\n}\n\nexport function replaceNodeParent<C extends CanvasRenderingContext2D>(\n previousNode: Node<C>,\n nextNode: Node<C>,\n parent: Node<C>,\n): void {\n if (previousNode === nextNode) {\n return;\n }\n const currentParent = registry.get(previousNode);\n if (currentParent !== parent) {\n throw getDetachOwnershipError();\n }\n if (registry.has(nextNode)) {\n throw getOwnershipError();\n }\n registry.delete(previousNode);\n registry.set(nextNode, parent);\n bumpRevision(parent);\n}\n\nexport function replaceNodesParent<C extends CanvasRenderingContext2D>(\n previousNodes: Iterable<Node<C>>,\n nextNodes: Iterable<Node<C>>,\n parent: Node<C>,\n): void {\n const previousSnapshot = Array.from(previousNodes);\n const nextSnapshot = Array.from(nextNodes);\n if (\n previousSnapshot.length === nextSnapshot.length &&\n previousSnapshot.every((node, index) => node === nextSnapshot[index])\n ) {\n return;\n }\n\n const previousSet = new Set(previousSnapshot);\n const nextSet = new Set<Node<C>>();\n for (const node of nextSnapshot) {\n if (nextSet.has(node)) {\n throw getOwnershipError();\n }\n nextSet.add(node);\n const currentParent = registry.get(node);\n if (currentParent != null && currentParent !== parent) {\n throw getOwnershipError();\n }\n }\n\n for (const node of previousSnapshot) {\n if (!nextSet.has(node)) {\n const currentParent = registry.get(node);\n if (currentParent !== parent) {\n throw getDetachOwnershipError();\n }\n registry.delete(node);\n }\n }\n\n for (const node of nextSnapshot) {\n if (!previousSet.has(node)) {\n registry.set(node, parent);\n }\n }\n\n bumpRevision(parent);\n}\n\nexport function forEachNodeAncestor<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n visitor: (ancestor: Node<C>) => void,\n): void {\n let current: Node<C> | undefined = node;\n while ((current = registry.get(current))) {\n visitor(current);\n }\n}\n\nexport function registerNodeParent<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n parent: Node<C>,\n): void {\n attachNodeToParent(node, parent);\n}\n\nexport function unregisterNodeParent<C extends CanvasRenderingContext2D>(node: Node<C>, parent?: Node<C>): void {\n detachNodeFromParent(node, parent);\n}\n\nexport function getNodeParent<C extends CanvasRenderingContext2D>(node: Node<C>): Node<C> | undefined {\n return registry.get(node);\n}\n","export function shallow<T extends object>(object: T): T {\n return Object.create(object) as T;\n}\n","import { attachNodeToParent, replaceNodeParent, replaceNodesParent } from \"../internal/node-registry\";\nimport type { Box, Context, HitTest, LayoutConstraints, Node } from \"../types\";\nimport { shallow } from \"../utils\";\n\nexport function withNodeConstraints<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n constraints: LayoutConstraints | undefined,\n): Context<C> {\n if (constraints === ctx.constraints) {\n return ctx;\n }\n const next = shallow(ctx);\n next.constraints = constraints;\n return next;\n}\n\nexport function measureNodeMinContent<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n node: Node<C>,\n constraints: LayoutConstraints | undefined = ctx.constraints,\n): Box {\n const nextCtx = withNodeConstraints(ctx, constraints);\n if (node.measureMinContent != null) {\n return node.measureMinContent(nextCtx);\n }\n return node.measure(nextCtx);\n}\n\n/**\n * A node that owns an ordered list of child nodes.\n */\nexport abstract class Group<C extends CanvasRenderingContext2D> implements Node<C> {\n #children: Node<C>[];\n\n /**\n * @param children Initial child nodes, in layout order.\n */\n constructor(children: Node<C>[]) {\n this.#children = [...children];\n replaceNodesParent([], this.#children, this);\n }\n\n /** Child nodes managed by this group. */\n get children(): readonly Node<C>[] {\n return this.#children;\n }\n\n /**\n * Replaces the full child list while updating parent links.\n */\n replaceChildren(nextChildren: Node<C>[]): void {\n const nextSnapshot = [...nextChildren];\n replaceNodesParent(this.#children, nextSnapshot, this);\n this.#children = nextSnapshot;\n }\n\n abstract measure(ctx: Context<C>): Box;\n abstract draw(ctx: Context<C>, x: number, y: number): boolean;\n abstract hittest(ctx: Context<C>, test: HitTest): boolean;\n}\n\n/**\n * A node that forwards layout and drawing to a single inner node.\n */\nexport class Wrapper<C extends CanvasRenderingContext2D> implements Node<C> {\n #inner: Node<C>;\n\n /**\n * @param inner Wrapped child node.\n */\n constructor(inner: Node<C>) {\n this.#inner = inner;\n attachNodeToParent(this.#inner, this);\n }\n\n /** The wrapped child node. */\n get inner(): Node<C> {\n return this.#inner;\n }\n\n /** Replaces the wrapped child node. */\n set inner(newNode: Node<C>) {\n if (newNode === this.#inner) {\n return;\n }\n replaceNodeParent(this.#inner, newNode, this);\n this.#inner = newNode;\n }\n\n measure(ctx: Context<C>): Box {\n return this.inner.measure(ctx);\n }\n\n measureMinContent(ctx: Context<C>): Box {\n return measureNodeMinContent(ctx, this.inner);\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n return this.inner.draw(ctx, x, y);\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n return this.inner.hittest(ctx, test);\n }\n}\n","import type { Box, ChildLayoutResult, FlexLayoutResult, LayoutConstraints, LayoutRect } from \"./types\";\n\n/**\n * 创建 LayoutRect 的辅助函数\n */\nexport function createRect(x: number, y: number, width: number, height: number): LayoutRect {\n return { x, y, width, height };\n}\n\n/**\n * 合并多个 rect 得到包含所有 rect 的最小外接矩形\n */\nexport function mergeRects(rects: LayoutRect[]): LayoutRect {\n if (rects.length === 0) {\n return createRect(0, 0, 0, 0);\n }\n\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const rect of rects) {\n minX = Math.min(minX, rect.x);\n minY = Math.min(minY, rect.y);\n maxX = Math.max(maxX, rect.x + rect.width);\n maxY = Math.max(maxY, rect.y + rect.height);\n }\n\n return createRect(minX, minY, maxX - minX, maxY - minY);\n}\n\n/**\n * 从子节点布局结果计算容器的 contentBox\n */\nexport function computeContentBox<C extends CanvasRenderingContext2D>(\n children: ChildLayoutResult<C>[],\n): LayoutRect {\n return mergeRects(children.map((child) => child.contentBox));\n}\n\n/**\n * 根据约束和实际内容计算最终的 containerBox\n */\nexport function computeContainerBox(\n contentBox: LayoutRect,\n constraints?: LayoutConstraints,\n): LayoutRect {\n let width = contentBox.width;\n let height = contentBox.height;\n\n if (constraints?.minWidth != null) {\n width = Math.max(width, constraints.minWidth);\n }\n if (constraints?.maxWidth != null) {\n width = Math.min(width, constraints.maxWidth);\n }\n if (constraints?.minHeight != null) {\n height = Math.max(height, constraints.minHeight);\n }\n if (constraints?.maxHeight != null) {\n height = Math.min(height, constraints.maxHeight);\n }\n\n return createRect(contentBox.x, contentBox.y, width, height);\n}\n\n/**\n * 将 Box 转换为 LayoutRect(位置为 0,0)\n */\nexport function boxToRect(box: Box): LayoutRect {\n return createRect(0, 0, box.width, box.height);\n}\n\n/**\n * 检查点是否在 rect 内\n */\nexport function pointInRect(x: number, y: number, rect: LayoutRect): boolean {\n return x >= rect.x && x < rect.x + rect.width && y >= rect.y && y < rect.y + rect.height;\n}\n\n/**\n * 平移 rect 的位置\n */\nexport function offsetRect(rect: LayoutRect, dx: number, dy: number): LayoutRect {\n return createRect(rect.x + dx, rect.y + dy, rect.width, rect.height);\n}\n\n/**\n * 读取单子节点布局结果中的唯一 child。\n */\nexport function getSingleChildLayout<C extends CanvasRenderingContext2D>(\n layout: FlexLayoutResult<C>,\n): ChildLayoutResult<C> | undefined {\n return layout.children[0];\n}\n\n/**\n * 在布局结果中按指定盒模型查找命中的 child,并返回局部坐标。\n */\nexport function findChildAtPoint<C extends CanvasRenderingContext2D>(\n children: ChildLayoutResult<C>[],\n x: number,\n y: number,\n box: \"rect\" | \"contentBox\" = \"contentBox\",\n):\n | {\n child: ChildLayoutResult<C>;\n localX: number;\n localY: number;\n }\n | undefined {\n for (let i = children.length - 1; i >= 0; i -= 1) {\n const child = children[i]!;\n const target = box === \"rect\" ? child.rect : child.contentBox;\n if (!pointInRect(x, y, target)) {\n continue;\n }\n return {\n child,\n localX: x - target.x,\n localY: y - target.y,\n };\n }\n return undefined;\n}\n","import { findChildAtPoint } from \"../layout\";\nimport type { Context, FlexLayoutResult, HitTest, LayoutConstraints, Node } from \"../types\";\nimport { shallow } from \"../utils\";\n\ntype LayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints): FlexLayoutResult<C> | undefined;\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints): void;\n};\n\ntype LayoutContext<C extends CanvasRenderingContext2D> = Context<C> & LayoutCacheAccess<C>;\n\nexport function withConstraints<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n constraints: LayoutConstraints | undefined,\n): Context<C> {\n const next = shallow(ctx);\n next.constraints = constraints;\n return next;\n}\n\nfunction getLayoutContext<C extends CanvasRenderingContext2D>(ctx: Context<C>): LayoutContext<C> {\n return ctx as LayoutContext<C>;\n}\n\nexport function readLayoutResult<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n): FlexLayoutResult<C> | undefined {\n return getLayoutContext(ctx).getLayoutResult(node, ctx.constraints);\n}\n\nexport function writeLayoutResult<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n result: FlexLayoutResult<C>,\n): void {\n getLayoutContext(ctx).setLayoutResult(node, result, ctx.constraints);\n}\n\nfunction ensureLayoutResult<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n): FlexLayoutResult<C> | undefined {\n return readLayoutResult(node, ctx);\n}\n\nexport function drawLayoutChildren<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n x: number,\n y: number,\n): boolean {\n const layoutResult = ensureLayoutResult(node, ctx);\n if (!layoutResult) {\n return false;\n }\n\n let result = false;\n for (const childResult of layoutResult.children) {\n const childDrawn = childResult.node.draw(\n withConstraints(ctx, childResult.constraints),\n x + childResult.contentBox.x,\n y + childResult.contentBox.y,\n );\n result = childDrawn || result;\n }\n return result;\n}\n\nexport function hittestLayoutChildren<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n test: HitTest,\n box: \"rect\" | \"contentBox\" = \"contentBox\",\n): boolean {\n const layoutResult = ensureLayoutResult(node, ctx);\n if (!layoutResult) {\n return false;\n }\n\n const hit = findChildAtPoint(layoutResult.children, test.x, test.y, box);\n if (!hit) {\n return false;\n }\n\n return hit.child.node.hittest(\n withConstraints(ctx, hit.child.constraints),\n {\n ...test,\n x: hit.localX,\n y: hit.localY,\n },\n );\n}\n","import { createRect, findChildAtPoint, getSingleChildLayout } from \"../layout\";\nimport type { Box, Context, HitTest, Node } from \"../types\";\nimport { measureNodeMinContent } from \"./base\";\nimport { Wrapper } from \"./base\";\nimport { readLayoutResult, withConstraints, writeLayoutResult } from \"./shared\";\n\nfunction clampToConstraints(value: number, min?: number, max?: number): number {\n let result = value;\n if (min != null) {\n result = Math.max(result, min);\n }\n if (max != null) {\n result = Math.min(result, max);\n }\n return result;\n}\n\nfunction shrinkConstraint(value: number | undefined, padding: number): number | undefined {\n if (value == null) {\n return undefined;\n }\n return Math.max(0, value - padding);\n}\n\n/**\n * Adds padding around a single child node.\n */\nexport class PaddingBox<C extends CanvasRenderingContext2D> extends Wrapper<C> {\n /**\n * @param inner Wrapped child node.\n * @param padding Padding in CSS pixels on each side.\n */\n constructor(\n inner: Node<C>,\n readonly padding: {\n top?: number;\n bottom?: number;\n left?: number;\n right?: number;\n } = {},\n ) {\n super(inner);\n }\n\n get #top(): number {\n return this.padding.top ?? 0;\n }\n\n get #bottom(): number {\n return this.padding.bottom ?? 0;\n }\n\n get #left(): number {\n return this.padding.left ?? 0;\n }\n\n get #right(): number {\n return this.padding.right ?? 0;\n }\n\n measure(ctx: Context<C>): Box {\n const paddingLeft = this.#left;\n const paddingRight = this.#right;\n const paddingTop = this.#top;\n const paddingBottom = this.#bottom;\n const horizontalPadding = paddingLeft + paddingRight;\n const verticalPadding = paddingTop + paddingBottom;\n const childConstraints = ctx.constraints\n ? {\n ...ctx.constraints,\n minWidth: shrinkConstraint(ctx.constraints.minWidth, horizontalPadding),\n maxWidth: shrinkConstraint(ctx.constraints.maxWidth, horizontalPadding),\n minHeight: shrinkConstraint(ctx.constraints.minHeight, verticalPadding),\n maxHeight: shrinkConstraint(ctx.constraints.maxHeight, verticalPadding),\n }\n : undefined;\n const { width, height } = ctx.measureNode(this.inner, childConstraints);\n const containerBox = createRect(\n 0,\n 0,\n clampToConstraints(width + horizontalPadding, ctx.constraints?.minWidth, ctx.constraints?.maxWidth),\n clampToConstraints(height + verticalPadding, ctx.constraints?.minHeight, ctx.constraints?.maxHeight),\n );\n const childRect = createRect(paddingLeft, paddingTop, width, height);\n writeLayoutResult(this, ctx, {\n containerBox,\n contentBox: childRect,\n children: [\n {\n node: this.inner,\n rect: childRect,\n contentBox: childRect,\n constraints: childConstraints,\n },\n ],\n constraints: ctx.constraints,\n });\n return {\n width: containerBox.width,\n height: containerBox.height,\n };\n }\n\n measureMinContent(ctx: Context<C>): Box {\n const paddingLeft = this.#left;\n const paddingRight = this.#right;\n const paddingTop = this.#top;\n const paddingBottom = this.#bottom;\n const horizontalPadding = paddingLeft + paddingRight;\n const verticalPadding = paddingTop + paddingBottom;\n const childConstraints = ctx.constraints\n ? {\n ...ctx.constraints,\n minWidth: shrinkConstraint(ctx.constraints.minWidth, horizontalPadding),\n maxWidth: shrinkConstraint(ctx.constraints.maxWidth, horizontalPadding),\n minHeight: shrinkConstraint(ctx.constraints.minHeight, verticalPadding),\n maxHeight: shrinkConstraint(ctx.constraints.maxHeight, verticalPadding),\n }\n : undefined;\n const { width, height } = measureNodeMinContent(ctx, this.inner, childConstraints);\n return {\n width: width + horizontalPadding,\n height: height + verticalPadding,\n };\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return this.inner.draw(ctx, x + this.#left, y + this.#top);\n }\n\n const childResult = getSingleChildLayout(layoutResult);\n if (!childResult) {\n return false;\n }\n\n return childResult.node.draw(\n withConstraints(ctx, childResult.constraints),\n x + childResult.rect.x,\n y + childResult.rect.y,\n );\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return false;\n }\n\n const hit = findChildAtPoint(layoutResult.children, test.x, test.y, \"rect\");\n if (!hit) {\n return false;\n }\n\n return hit.child.node.hittest(\n withConstraints(ctx, hit.child.constraints),\n {\n ...test,\n x: hit.localX,\n y: hit.localY,\n },\n );\n }\n}\n\n/**\n * A leaf node with a fixed size and no drawing behavior.\n */\nexport class Fixed<C extends CanvasRenderingContext2D> implements Node<C> {\n /**\n * @param width Fixed width in CSS pixels.\n * @param height Fixed height in CSS pixels.\n */\n constructor(\n readonly width: number,\n readonly height: number,\n ) {}\n\n measure(_ctx: Context<C>): Box {\n return { width: this.width, height: this.height };\n }\n\n measureMinContent(_ctx: Context<C>): Box {\n return { width: this.width, height: this.height };\n }\n\n draw(_ctx: Context<C>, _x: number, _y: number): boolean {\n return false;\n }\n\n hittest(_ctx: Context<C>, _test: HitTest): boolean {\n return false;\n }\n}\n","import { computeContentBox, createRect } from \"../layout\";\nimport type {\n Axis,\n Box,\n ChildLayoutResult,\n Context,\n CrossAxisAlignment,\n FlexContainerOptions,\n FlexItemOptions,\n HitTest,\n LayoutConstraints,\n Node,\n} from \"../types\";\nimport { Group, measureNodeMinContent, Wrapper } from \"./base\";\nimport { drawLayoutChildren, hittestLayoutChildren, writeLayoutResult } from \"./shared\";\n\nfunction getMainSize(axis: Axis, box: Box): number {\n return axis === \"row\" ? box.width : box.height;\n}\n\nfunction getCrossSize(axis: Axis, box: Box): number {\n return axis === \"row\" ? box.height : box.width;\n}\n\nfunction getMinMain(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.minWidth : constraints?.minHeight;\n}\n\nfunction getMaxMain(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.maxWidth : constraints?.maxHeight;\n}\n\nfunction getMinCross(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.minHeight : constraints?.minWidth;\n}\n\nfunction getMaxCross(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.maxHeight : constraints?.maxWidth;\n}\n\nfunction createAxisConstraints(\n axis: Axis,\n constraints: LayoutConstraints | undefined,\n main: { min?: number; max?: number },\n cross: { min?: number; max?: number } = {},\n): LayoutConstraints | undefined {\n if (\n constraints == null &&\n main.min == null &&\n main.max == null &&\n cross.min == null &&\n cross.max == null\n ) {\n return undefined;\n }\n\n const next: LayoutConstraints = {\n ...constraints,\n };\n if (axis === \"row\") {\n next.minWidth = main.min;\n next.maxWidth = main.max;\n next.minHeight = cross.min;\n next.maxHeight = cross.max;\n } else {\n next.minHeight = main.min;\n next.maxHeight = main.max;\n next.minWidth = cross.min;\n next.maxWidth = cross.max;\n }\n return next;\n}\n\nfunction clampToConstraints(value: number, min?: number, max?: number): number {\n let result = value;\n if (min != null) {\n result = Math.max(result, min);\n }\n if (max != null) {\n result = Math.min(result, max);\n }\n return result;\n}\n\nfunction constraintsEqual(left: LayoutConstraints | undefined, right: LayoutConstraints | undefined): boolean {\n if (left === right) {\n return true;\n }\n if (left == null || right == null) {\n return left == null && right == null;\n }\n return left.minWidth === right.minWidth\n && left.maxWidth === right.maxWidth\n && left.minHeight === right.minHeight\n && left.maxHeight === right.maxHeight;\n}\n\nfunction getCrossAlignment(alignSelf: CrossAxisAlignment | \"auto\" | undefined, alignItems: CrossAxisAlignment): CrossAxisAlignment {\n if (alignSelf == null || alignSelf === \"auto\") {\n return alignItems;\n }\n return alignSelf;\n}\n\nfunction getJustifySpacing(\n justifyContent: NonNullable<FlexContainerOptions[\"justifyContent\"]>,\n freeSpace: number,\n itemCount: number,\n gap: number,\n): { leading: number; between: number } {\n switch (justifyContent) {\n case \"center\":\n return { leading: freeSpace / 2, between: gap };\n case \"end\":\n return { leading: freeSpace, between: gap };\n case \"space-between\":\n return {\n leading: 0,\n between: itemCount > 1 ? gap + freeSpace / (itemCount - 1) : gap,\n };\n case \"space-around\":\n return {\n leading: itemCount > 0 ? freeSpace / itemCount / 2 : 0,\n between: itemCount > 0 ? gap + freeSpace / itemCount : gap,\n };\n case \"space-evenly\":\n return {\n leading: itemCount > 0 ? freeSpace / (itemCount + 1) : 0,\n between: itemCount > 0 ? gap + freeSpace / (itemCount + 1) : gap,\n };\n case \"start\":\n default:\n return { leading: 0, between: gap };\n }\n}\n\nfunction getCrossOffset(align: CrossAxisAlignment, frameCross: number, contentCross: number): number {\n switch (align) {\n case \"center\":\n return (frameCross - contentCross) / 2;\n case \"end\":\n return frameCross - contentCross;\n case \"stretch\":\n case \"start\":\n default:\n return 0;\n }\n}\n\nfunction createRectFromAxis(axis: Axis, main: number, cross: number, mainSize: number, crossSize: number) {\n return axis === \"row\"\n ? createRect(main, cross, mainSize, crossSize)\n : createRect(cross, main, crossSize, mainSize);\n}\n\nconst SHRINK_EPSILON = 1e-6;\n\ntype FlexMeasurement<C extends CanvasRenderingContext2D> = {\n child: Node<C>;\n item: FlexItemOptions;\n basisMeasured: Box;\n measured: Box;\n basisConstraints?: LayoutConstraints;\n initialConstraints?: LayoutConstraints;\n finalConstraints?: LayoutConstraints;\n allocatedMain?: number;\n grow: number;\n shrink: number;\n effectiveAlign: CrossAxisAlignment;\n stretch: boolean;\n basis: number;\n minContentMain: number;\n finalMain: number;\n frozen: boolean;\n frameMain: number;\n frameCross: number;\n};\n\ntype MeasuredLayout<C extends CanvasRenderingContext2D> = {\n box: Box;\n layout: {\n containerBox: ChildLayoutResult<C>[\"rect\"];\n contentBox: ChildLayoutResult<C>[\"contentBox\"];\n children: ChildLayoutResult<C>[];\n constraints?: LayoutConstraints;\n };\n};\n\nfunction readFlexItemOptions<C extends CanvasRenderingContext2D>(child: Node<C>): FlexItemOptions {\n if (child instanceof FlexItem) {\n return child.item;\n }\n return {};\n}\n\nexport function computeFlexLayout<C extends CanvasRenderingContext2D>(\n children: readonly Node<C>[],\n options: FlexContainerOptions,\n constraints: LayoutConstraints | undefined,\n measureChild: (node: Node<C>, constraints?: LayoutConstraints) => Box,\n measureChildMinContent: (node: Node<C>, constraints?: LayoutConstraints) => Box,\n): MeasuredLayout<C> {\n const axis = options.direction ?? \"row\";\n const gap = options.gap ?? 0;\n const justifyContent = options.justifyContent ?? \"start\";\n const alignItems = options.alignItems ?? \"start\";\n const reverse = options.reverse ?? false;\n const mainAxisSize = options.mainAxisSize ?? \"fill\";\n const orderedChildren = reverse ? [...children].reverse() : children;\n const maxMain = getMaxMain(axis, constraints);\n const minMain = getMinMain(axis, constraints);\n const maxCross = getMaxCross(axis, constraints);\n const minCross = getMinCross(axis, constraints);\n const gapTotal = orderedChildren.length > 1 ? gap * (orderedChildren.length - 1) : 0;\n const finiteMain = maxMain != null;\n const finiteCross = maxCross != null;\n const availableMain = finiteMain ? Math.max(0, maxMain - gapTotal) : undefined;\n let totalGrow = 0;\n let totalBasis = 0;\n let nonGrowBasis = 0;\n const measurements = new Map<Node<C>, FlexMeasurement<C>>();\n const basisConstraints = createAxisConstraints(\n axis,\n constraints,\n {\n min: undefined,\n max: undefined,\n },\n {\n min: undefined,\n max: maxCross,\n },\n );\n\n for (const child of orderedChildren) {\n const item = readFlexItemOptions(child);\n const grow = item.grow ?? 0;\n const shrink = item.shrink ?? 0;\n totalGrow += grow;\n const effectiveAlign = getCrossAlignment(item.alignSelf, alignItems);\n const stretch = effectiveAlign === \"stretch\";\n const basisMeasured = measureChild(child, basisConstraints);\n const basis = getMainSize(axis, basisMeasured);\n\n totalBasis += basis;\n if (grow <= 0) {\n nonGrowBasis += basis;\n }\n\n measurements.set(child, {\n child,\n item,\n basisMeasured,\n measured: basisMeasured,\n basisConstraints,\n initialConstraints: basisConstraints,\n finalConstraints: basisConstraints,\n allocatedMain: undefined,\n grow,\n shrink,\n effectiveAlign,\n stretch,\n basis,\n minContentMain: basis,\n finalMain: basis,\n frozen: false,\n frameMain: basis,\n frameCross: getCrossSize(axis, basisMeasured),\n });\n }\n\n const entersShrinkPath = finiteMain && availableMain != null && totalBasis - availableMain > SHRINK_EPSILON;\n\n if (entersShrinkPath) {\n const totalDeficit = totalBasis - availableMain!;\n let remainingDeficit = totalDeficit;\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n const minContentMeasured = measureChildMinContent(child, measurement.basisConstraints);\n measurement.minContentMain = Math.min(measurement.basis, getMainSize(axis, minContentMeasured));\n measurement.finalMain = measurement.basis;\n measurement.frozen = measurement.shrink <= 0 || measurement.basis - measurement.minContentMain <= SHRINK_EPSILON;\n }\n\n while (remainingDeficit > SHRINK_EPSILON) {\n const active = orderedChildren\n .map((child) => measurements.get(child)!)\n .filter((measurement) => !measurement.frozen && measurement.shrink > 0);\n const totalScaled = active.reduce((sum, measurement) => sum + measurement.shrink * measurement.basis, 0);\n\n if (active.length === 0 || totalScaled <= SHRINK_EPSILON) {\n break;\n }\n\n let frozeAny = false;\n for (const measurement of active) {\n const tentative = measurement.basis - remainingDeficit * ((measurement.shrink * measurement.basis) / totalScaled);\n if (tentative <= measurement.minContentMain + SHRINK_EPSILON) {\n measurement.finalMain = measurement.minContentMain;\n measurement.frozen = true;\n frozeAny = true;\n } else {\n measurement.finalMain = tentative;\n }\n }\n\n if (!frozeAny) {\n remainingDeficit = 0;\n break;\n }\n\n let absorbedDeficit = 0;\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (measurement.frozen) {\n absorbedDeficit += Math.max(0, measurement.basis - measurement.finalMain);\n }\n }\n remainingDeficit = Math.max(0, totalDeficit - absorbedDeficit);\n }\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n measurement.measured = measurement.basisMeasured;\n measurement.initialConstraints = measurement.basisConstraints;\n measurement.finalConstraints = createAxisConstraints(\n axis,\n constraints,\n {\n min: undefined,\n max: measurement.finalMain,\n },\n {\n min: undefined,\n max: maxCross,\n },\n );\n measurement.allocatedMain = undefined;\n measurement.frameMain = measurement.finalMain;\n measurement.frameCross = getCrossSize(axis, measurement.measured);\n }\n } else {\n const remainingMain = finiteMain && availableMain != null ? Math.max(0, availableMain - nonGrowBasis) : undefined;\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (!(measurement.grow > 0 && finiteMain && remainingMain != null && totalGrow > 0)) {\n measurement.measured = measurement.basisMeasured;\n measurement.initialConstraints = measurement.basisConstraints;\n measurement.finalConstraints = finiteMain\n ? createAxisConstraints(\n axis,\n constraints,\n {\n min: undefined,\n max: measurement.finalMain,\n },\n {\n min: undefined,\n max: maxCross,\n },\n )\n : measurement.basisConstraints;\n measurement.allocatedMain = undefined;\n measurement.finalMain = measurement.basis;\n measurement.frameMain = measurement.basis;\n measurement.frameCross = getCrossSize(axis, measurement.measured);\n continue;\n }\n\n const allocatedMain = (remainingMain * measurement.grow) / totalGrow;\n const childConstraints = createAxisConstraints(\n axis,\n constraints,\n {\n max: allocatedMain,\n },\n {\n min: undefined,\n max: maxCross,\n },\n );\n const measured = measureChild(child, childConstraints);\n measurement.measured = measured;\n measurement.initialConstraints = childConstraints;\n measurement.finalConstraints = childConstraints;\n measurement.allocatedMain = allocatedMain;\n measurement.finalMain = allocatedMain;\n measurement.frameMain = allocatedMain;\n measurement.frameCross = getCrossSize(axis, measured);\n }\n }\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (!constraintsEqual(measurement.initialConstraints, measurement.finalConstraints)) {\n measurement.measured = measureChild(child, measurement.finalConstraints);\n }\n measurement.frameMain = measurement.finalMain;\n measurement.frameCross = getCrossSize(axis, measurement.measured);\n }\n\n let contentMain = gapTotal;\n let contentCross = 0;\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n contentMain += measurement.frameMain;\n contentCross = Math.max(contentCross, measurement.frameCross);\n }\n\n const containerMain = finiteMain && mainAxisSize === \"fill\"\n ? Math.max(maxMain!, contentMain)\n : clampToConstraints(contentMain, minMain, maxMain);\n const containerCross = clampToConstraints(contentCross, minCross, maxCross);\n if (finiteCross) {\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (!measurement.stretch) {\n continue;\n }\n\n const finalConstraints = createAxisConstraints(\n axis,\n measurement.finalConstraints,\n {\n min: getMinMain(axis, measurement.finalConstraints),\n max: getMaxMain(axis, measurement.finalConstraints),\n },\n {\n min: containerCross,\n max: containerCross,\n },\n );\n const remeasured = measureChild(child, finalConstraints);\n measurement.measured = remeasured;\n measurement.finalConstraints = finalConstraints;\n measurement.frameCross = containerCross;\n measurement.frameMain = measurement.allocatedMain ?? getMainSize(axis, remeasured);\n }\n\n contentMain = gapTotal;\n contentCross = 0;\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n contentMain += measurement.frameMain;\n contentCross = Math.max(contentCross, getCrossSize(axis, measurement.measured));\n }\n }\n\n const finalContainerMain = finiteMain && mainAxisSize === \"fill\"\n ? Math.max(maxMain!, contentMain)\n : clampToConstraints(contentMain, minMain, maxMain);\n const freeSpace = Math.max(0, finalContainerMain - contentMain);\n const spacing = getJustifySpacing(justifyContent, freeSpace, orderedChildren.length, gap);\n const childResults: ChildLayoutResult<C>[] = [];\n let cursor = spacing.leading;\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n const frameCross = measurement.stretch && finiteCross ? containerCross : measurement.frameCross;\n const contentMainSize = getMainSize(axis, measurement.measured);\n const contentCrossSize = getCrossSize(axis, measurement.measured);\n const rectCross = measurement.stretch ? 0 : getCrossOffset(measurement.effectiveAlign, containerCross, frameCross);\n const contentCrossOffset = rectCross + getCrossOffset(measurement.effectiveAlign, frameCross, contentCrossSize);\n const rect = createRectFromAxis(axis, cursor, rectCross, measurement.frameMain, frameCross);\n const contentBox = createRectFromAxis(axis, cursor, contentCrossOffset, contentMainSize, contentCrossSize);\n\n childResults.push({\n node: child,\n rect,\n contentBox,\n constraints: measurement.finalConstraints,\n });\n cursor += measurement.frameMain + spacing.between;\n }\n\n const containerBox = axis === \"row\"\n ? createRect(0, 0, finalContainerMain, containerCross)\n : createRect(0, 0, containerCross, finalContainerMain);\n const finalContentBox = childResults.length > 0\n ? computeContentBox(childResults)\n : createRect(0, 0, 0, 0);\n\n return {\n box: {\n width: containerBox.width,\n height: containerBox.height,\n },\n layout: {\n containerBox,\n contentBox: finalContentBox,\n children: childResults,\n constraints,\n },\n };\n}\n\n/**\n * Wraps a child node with per-item flex options.\n */\nexport class FlexItem<C extends CanvasRenderingContext2D> extends Wrapper<C> {\n /**\n * @param inner Wrapped child node.\n * @param item Flex behavior overrides for the child.\n */\n constructor(\n inner: Node<C>,\n readonly item: FlexItemOptions = {},\n ) {\n super(inner);\n }\n}\n\n/**\n * Lays out children in a single flex row or column.\n */\nexport class Flex<C extends CanvasRenderingContext2D> extends Group<C> {\n /**\n * @param children Child nodes in visual order.\n * @param options Flex container configuration.\n */\n constructor(\n children: Node<C>[],\n readonly options: FlexContainerOptions = {},\n ) {\n super(children);\n }\n\n measure(ctx: Context<C>): Box {\n const result = computeFlexLayout(\n this.children,\n this.options,\n ctx.constraints,\n (node, constraints) => ctx.measureNode(node, constraints),\n (node, constraints) => measureNodeMinContent(ctx, node, constraints),\n );\n writeLayoutResult(this, ctx, result.layout);\n return result.box;\n }\n\n measureMinContent(ctx: Context<C>): Box {\n const axis = this.options.direction ?? \"row\";\n const gap = this.options.gap ?? 0;\n const orderedChildren = this.options.reverse ? [...this.children].reverse() : this.children;\n const gapTotal = orderedChildren.length > 1 ? gap * (orderedChildren.length - 1) : 0;\n const childConstraints = createAxisConstraints(\n axis,\n ctx.constraints,\n {\n min: undefined,\n max: undefined,\n },\n {\n min: undefined,\n max: getMaxCross(axis, ctx.constraints),\n },\n );\n\n let width = axis === \"row\" ? gapTotal : 0;\n let height = axis === \"column\" ? gapTotal : 0;\n\n for (const child of orderedChildren) {\n const measured = measureNodeMinContent(ctx, child, childConstraints);\n if (axis === \"row\") {\n width += measured.width;\n height = Math.max(height, measured.height);\n } else {\n width = Math.max(width, measured.width);\n height += measured.height;\n }\n }\n\n return { width, height };\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n return drawLayoutChildren(this, ctx, x, y);\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n return hittestLayoutChildren(this, ctx, test, \"contentBox\");\n }\n}\n","import { createRect, findChildAtPoint, getSingleChildLayout } from \"../layout\";\nimport type { Box, Context, HitTest, Node, TextAlign } from \"../types\";\nimport { measureNodeMinContent, Wrapper } from \"./base\";\nimport { readLayoutResult, withConstraints, writeLayoutResult } from \"./shared\";\n\nfunction resolveHorizontalOffset(align: TextAlign, availableWidth: number, childWidth: number): number {\n switch (align) {\n case \"center\":\n return (availableWidth - childWidth) / 2;\n case \"end\":\n return availableWidth - childWidth;\n case \"start\":\n return 0;\n }\n}\n\n/**\n * Aligns a single child horizontally within the available width.\n */\nexport class Place<C extends CanvasRenderingContext2D> extends Wrapper<C> {\n /**\n * @param inner Wrapped child node.\n * @param options Alignment behavior for the child.\n */\n constructor(\n inner: Node<C>,\n readonly options: {\n align?: TextAlign;\n expand?: boolean;\n } = {},\n ) {\n super(inner);\n }\n\n measure(ctx: Context<C>): Box {\n const availableWidth = ctx.constraints?.maxWidth;\n const expand = this.options.expand ?? true;\n const childConstraints = ctx.constraints\n ? {\n ...ctx.constraints,\n }\n : undefined;\n const childBox = ctx.measureNode(this.inner, childConstraints);\n let width = expand && availableWidth != null ? availableWidth : childBox.width;\n if (ctx.constraints?.minWidth != null) {\n width = Math.max(width, ctx.constraints.minWidth);\n }\n if (ctx.constraints?.maxWidth != null) {\n width = Math.min(width, ctx.constraints.maxWidth);\n }\n\n const align = this.options.align ?? \"start\";\n const childRect = createRect(resolveHorizontalOffset(align, width, childBox.width), 0, childBox.width, childBox.height);\n\n writeLayoutResult(this, ctx, {\n containerBox: createRect(0, 0, width, childBox.height),\n contentBox: childRect,\n children: [\n {\n node: this.inner,\n rect: childRect,\n contentBox: createRect(0, 0, childBox.width, childBox.height),\n constraints: childConstraints,\n },\n ],\n constraints: ctx.constraints,\n });\n\n return {\n width,\n height: childBox.height,\n };\n }\n\n measureMinContent(ctx: Context<C>): Box {\n return measureNodeMinContent(ctx, this.inner);\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return this.inner.draw(ctx, x, y);\n }\n\n const childResult = getSingleChildLayout(layoutResult);\n if (!childResult) {\n return false;\n }\n const childCtx = withConstraints(ctx, childResult.constraints);\n return childResult.node.draw(childCtx, x + childResult.rect.x, y + childResult.rect.y);\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return false;\n }\n\n const hit = findChildAtPoint(layoutResult.children, test.x, test.y, \"rect\");\n if (!hit) {\n return false;\n }\n\n return hit.child.node.hittest(\n withConstraints(ctx, hit.child.constraints),\n {\n ...test,\n x: hit.localX,\n y: hit.localY,\n },\n );\n }\n}\n","import type { Context, TextEllipsisPosition } from \"../types\";\n\nexport const FONT_SHIFT_PROBE = \"M\";\nexport const ELLIPSIS_GLYPH = \"…\";\nexport const INTRINSIC_MAX_WIDTH = Number.POSITIVE_INFINITY;\nexport const MIN_CONTENT_WIDTH_EPSILON = 0.001;\nexport const FONT_SHIFT_CACHE_CAPACITY = 64;\nexport const ELLIPSIS_WIDTH_CACHE_CAPACITY = 64;\n\nconst fontShiftCache = new Map<string, number>();\nconst ellipsisWidthCache = new Map<string, number>();\n\nlet sharedGraphemeSegmenter: Intl.Segmenter | null | undefined;\n\nexport function readLruValue<T>(cache: Map<string, T>, key: string): T | undefined {\n const cached = cache.get(key);\n if (cached == null) {\n return undefined;\n }\n cache.delete(key);\n cache.set(key, cached);\n return cached;\n}\n\nexport function writeLruValue<T>(cache: Map<string, T>, key: string, value: T, capacity: number): T {\n if (cache.has(key)) {\n cache.delete(key);\n } else if (cache.size >= capacity) {\n const firstKey = cache.keys().next().value;\n if (firstKey != null) {\n cache.delete(firstKey);\n }\n }\n cache.set(key, value);\n return value;\n}\n\nexport function measureFontShift<C extends CanvasRenderingContext2D>(ctx: Context<C>): number {\n const font = ctx.graphics.font;\n const cached = readLruValue(fontShiftCache, font);\n if (cached != null) {\n return cached;\n }\n const {\n fontBoundingBoxAscent: ascent = 0,\n fontBoundingBoxDescent: descent = 0,\n } = ctx.graphics.measureText(FONT_SHIFT_PROBE);\n return writeLruValue(fontShiftCache, font, ascent - descent, FONT_SHIFT_CACHE_CAPACITY);\n}\n\nexport function measureEllipsisWidth<C extends CanvasRenderingContext2D>(ctx: Context<C>): number {\n const font = ctx.graphics.font;\n const cached = readLruValue(ellipsisWidthCache, font);\n if (cached != null) {\n return cached;\n }\n return writeLruValue(\n ellipsisWidthCache,\n font,\n ctx.graphics.measureText(ELLIPSIS_GLYPH).width,\n ELLIPSIS_WIDTH_CACHE_CAPACITY,\n );\n}\n\nfunction getGraphemeSegmenter(): Intl.Segmenter | null {\n if (sharedGraphemeSegmenter !== undefined) {\n return sharedGraphemeSegmenter;\n }\n sharedGraphemeSegmenter = typeof Intl.Segmenter === \"function\"\n ? new Intl.Segmenter(undefined, { granularity: \"grapheme\" })\n : null;\n return sharedGraphemeSegmenter;\n}\n\nexport function splitGraphemes(text: string): string[] {\n const segmenter = getGraphemeSegmenter();\n if (segmenter == null) {\n return Array.from(text);\n }\n const graphemes: string[] = [];\n for (const part of segmenter.segment(text)) {\n graphemes.push(part.segment);\n }\n return graphemes;\n}\n\nexport function buildPrefixWidths(widths: readonly number[]): number[] {\n const cumulativeWidths = [0];\n let total = 0;\n for (const width of widths) {\n total += width;\n cumulativeWidths.push(total);\n }\n return cumulativeWidths;\n}\n\nexport function buildSuffixWidths(widths: readonly number[]): number[] {\n const cumulativeWidths = [0];\n let total = 0;\n for (let index = widths.length - 1; index >= 0; index -= 1) {\n total += widths[index] ?? 0;\n cumulativeWidths.push(total);\n }\n return cumulativeWidths;\n}\n\nexport function findMaxFittingCount(cumulativeWidths: readonly number[], maxWidth: number): number {\n if (maxWidth <= 0) {\n return 0;\n }\n let low = 0;\n let high = cumulativeWidths.length - 1;\n while (low < high) {\n const mid = Math.floor((low + high + 1) / 2);\n if ((cumulativeWidths[mid] ?? 0) <= maxWidth) {\n low = mid;\n } else {\n high = mid - 1;\n }\n }\n return low;\n}\n\nexport function normalizeMaxLines(maxLines: number | undefined): number | undefined {\n if (maxLines == null || !Number.isFinite(maxLines)) {\n return undefined;\n }\n return Math.max(1, Math.trunc(maxLines));\n}\n\nexport function selectEllipsisUnitCounts({\n position,\n prefixWidths,\n suffixWidths,\n unitCount,\n availableWidth,\n getMaxSuffixCount = (prefixCount) => unitCount - prefixCount,\n}: {\n position: TextEllipsisPosition;\n prefixWidths: readonly number[];\n suffixWidths: readonly number[];\n unitCount: number;\n availableWidth: number;\n getMaxSuffixCount?: (prefixCount: number) => number;\n}): { prefixCount: number; suffixCount: number } {\n let prefixCount = 0;\n let suffixCount = 0;\n\n switch (position) {\n case \"start\":\n suffixCount = Math.min(unitCount, findMaxFittingCount(suffixWidths, availableWidth));\n break;\n case \"middle\": {\n let bestVisibleUnits = -1;\n let bestBalanceScore = Number.NEGATIVE_INFINITY;\n for (let nextPrefixCount = 0; nextPrefixCount <= unitCount; nextPrefixCount += 1) {\n const prefixWidth = prefixWidths[nextPrefixCount] ?? 0;\n if (prefixWidth > availableWidth) {\n break;\n }\n const remainingWidth = availableWidth - prefixWidth;\n const maxSuffixCount = Math.max(0, getMaxSuffixCount(nextPrefixCount));\n const nextSuffixCount = Math.min(maxSuffixCount, findMaxFittingCount(suffixWidths, remainingWidth));\n const visibleUnits = nextPrefixCount + nextSuffixCount;\n const balanceScore = -Math.abs(nextPrefixCount - nextSuffixCount);\n if (\n visibleUnits > bestVisibleUnits ||\n (visibleUnits === bestVisibleUnits && balanceScore > bestBalanceScore) ||\n (visibleUnits === bestVisibleUnits && balanceScore === bestBalanceScore && nextPrefixCount > prefixCount)\n ) {\n prefixCount = nextPrefixCount;\n suffixCount = nextSuffixCount;\n bestVisibleUnits = visibleUnits;\n bestBalanceScore = balanceScore;\n }\n }\n break;\n }\n case \"end\":\n prefixCount = Math.min(unitCount, findMaxFittingCount(prefixWidths, availableWidth));\n break;\n }\n\n return { prefixCount, suffixCount };\n}\n","import { layoutNextLine, prepareWithSegments, type PreparedTextWithSegments } from \"@chenglou/pretext\";\nimport type { Context, TextOverflowWrapMode, TextWhiteSpaceMode, TextWordBreakMode } from \"../types\";\nimport { INTRINSIC_MAX_WIDTH, readLruValue, splitGraphemes, writeLruValue } from \"./core\";\n\nexport const PREPARED_TEXT_CACHE_CAPACITY = 512;\n\nconst LINE_START_CURSOR = { segmentIndex: 0, graphemeIndex: 0 } as const;\n\nconst preparedTextCache = new Map<string, PreparedTextWithSegments>();\nconst preparedUnitCache = new WeakMap<PreparedTextWithSegments, PreparedTextUnit[]>();\n\nexport type PreparedTextUnit = {\n text: string;\n width: number;\n};\n\nfunction getPreparedTextCacheKey(\n text: string,\n font: string,\n whiteSpace: TextWhiteSpaceMode,\n wordBreak: TextWordBreakMode,\n): string {\n return `${font}\\u0000${whiteSpace}\\u0000${wordBreak}\\u0000${text}`;\n}\n\nexport function readPreparedText(\n text: string,\n font: string,\n whiteSpace: TextWhiteSpaceMode,\n wordBreak: TextWordBreakMode,\n): PreparedTextWithSegments {\n const key = getPreparedTextCacheKey(text, font, whiteSpace, wordBreak);\n const cached = readLruValue(preparedTextCache, key);\n if (cached != null) {\n return cached;\n }\n return writeLruValue(\n preparedTextCache,\n key,\n prepareWithSegments(text, font, { whiteSpace, wordBreak }),\n PREPARED_TEXT_CACHE_CAPACITY,\n );\n}\n\nexport function readPreparedFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode,\n wordBreak: TextWordBreakMode,\n): { text: string; prepared: PreparedTextWithSegments } | undefined {\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const line = layoutNextLine(prepared, LINE_START_CURSOR, INTRINSIC_MAX_WIDTH);\n if (line == null) {\n return undefined;\n }\n return {\n text: line.text,\n prepared: readPreparedText(line.text, ctx.graphics.font, whiteSpace, wordBreak),\n };\n}\n\nexport function measurePreparedMinContentWidth(\n prepared: PreparedTextWithSegments,\n overflowWrap: TextOverflowWrapMode = \"break-word\",\n): number {\n let maxWidth = 0;\n let maxAnyWidth = 0;\n for (let index = 0; index < prepared.widths.length; index += 1) {\n const segmentWidth = prepared.widths[index] ?? 0;\n maxAnyWidth = Math.max(maxAnyWidth, segmentWidth);\n const segment = prepared.segments[index];\n if (segment != null && segment.trim().length > 0) {\n const breakableWidths = prepared.breakableFitAdvances[index];\n const minContentWidth = overflowWrap === \"anywhere\" && breakableWidths != null && breakableWidths.length > 0\n ? breakableWidths.reduce((widest, width) => Math.max(widest, width), 0)\n : segmentWidth;\n maxWidth = Math.max(maxWidth, minContentWidth);\n }\n }\n return maxWidth > 0 ? maxWidth : maxAnyWidth;\n}\n\nexport function getPreparedUnits(prepared: PreparedTextWithSegments): PreparedTextUnit[] {\n const cached = preparedUnitCache.get(prepared);\n if (cached != null) {\n return cached;\n }\n\n const units: PreparedTextUnit[] = [];\n for (let index = 0; index < prepared.segments.length; index += 1) {\n const segment = prepared.segments[index] ?? \"\";\n const segmentWidth = prepared.widths[index] ?? 0;\n const breakableWidths = prepared.breakableFitAdvances[index];\n if (breakableWidths != null && segment.length > 0) {\n const graphemes = splitGraphemes(segment);\n if (graphemes.length === breakableWidths.length) {\n for (let graphemeIndex = 0; graphemeIndex < graphemes.length; graphemeIndex += 1) {\n units.push({\n text: graphemes[graphemeIndex] ?? \"\",\n width: breakableWidths[graphemeIndex] ?? 0,\n });\n }\n continue;\n }\n }\n\n if (segment.length > 0 || segmentWidth > 0) {\n units.push({ text: segment, width: segmentWidth });\n }\n }\n\n preparedUnitCache.set(prepared, units);\n return units;\n}\n\nexport function joinUnitText(units: readonly PreparedTextUnit[], start: number, end: number): string {\n if (start >= end) {\n return \"\";\n }\n return units.slice(start, end).map((unit) => unit.text).join(\"\");\n}\n","import {\n layoutNextLine,\n layoutWithLines,\n measureLineStats,\n measureNaturalWidth,\n type PreparedTextWithSegments,\n} from \"@chenglou/pretext\";\nimport type {\n Context,\n TextEllipsisPosition,\n TextOverflowMode,\n TextOverflowWrapMode,\n TextWhiteSpaceMode,\n TextWordBreakMode,\n} from \"../types\";\nimport {\n ELLIPSIS_GLYPH,\n INTRINSIC_MAX_WIDTH,\n MIN_CONTENT_WIDTH_EPSILON,\n buildPrefixWidths,\n buildSuffixWidths,\n measureEllipsisWidth,\n measureFontShift,\n normalizeMaxLines,\n selectEllipsisUnitCounts,\n} from \"./core\";\nimport {\n getPreparedUnits,\n joinUnitText,\n measurePreparedMinContentWidth,\n readPreparedFirstLine,\n readPreparedText,\n} from \"./plain-core\";\n\nconst LINE_START_CURSOR = { segmentIndex: 0, graphemeIndex: 0 } as const;\n\nexport interface TextLayout {\n width: number;\n text: string;\n shift: number;\n}\n\nexport interface TextMeasurement {\n width: number;\n lineCount: number;\n}\n\nexport interface OverflowTextLayout extends TextLayout {\n overflowed: boolean;\n}\n\nexport interface OverflowTextBlockLayout {\n width: number;\n lines: OverflowTextLayout[];\n overflowed: boolean;\n}\n\ntype TextBlockLayout = {\n width: number;\n lines: TextLayout[];\n};\n\nfunction clampMaxWidth(maxWidth: number): number {\n return Math.max(0, maxWidth);\n}\n\nfunction createEllipsisOnlyLayout<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n maxWidth: number,\n shift: number,\n): OverflowTextLayout {\n const ellipsisWidth = measureEllipsisWidth(ctx);\n if (ellipsisWidth > maxWidth) {\n return { width: 0, text: \"\", shift, overflowed: true };\n }\n return { width: ellipsisWidth, text: ELLIPSIS_GLYPH, shift, overflowed: true };\n}\n\nfunction toTextBlockLayout(lines: { width: number; text: string }[], shift: number): TextBlockLayout {\n const mappedLines = lines.map((line) => ({ width: line.width, text: line.text, shift }));\n const width = mappedLines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0);\n return { width, lines: mappedLines };\n}\n\nfunction layoutPreparedEllipsis<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n prepared: PreparedTextWithSegments,\n text: string,\n maxWidth: number,\n shift: number,\n position: TextEllipsisPosition,\n forceEllipsis = false,\n): OverflowTextLayout {\n const intrinsicWidth = measureNaturalWidth(prepared);\n if (!forceEllipsis && intrinsicWidth <= maxWidth) {\n return { width: intrinsicWidth, text, shift, overflowed: false };\n }\n\n const ellipsisWidth = measureEllipsisWidth(ctx);\n if (ellipsisWidth > maxWidth) {\n return { width: 0, text: \"\", shift, overflowed: true };\n }\n\n const units = getPreparedUnits(prepared);\n if (units.length === 0) {\n return createEllipsisOnlyLayout(ctx, maxWidth, shift);\n }\n\n const availableWidth = Math.max(0, maxWidth - ellipsisWidth);\n const prefixWidths = buildPrefixWidths(units.map((unit) => unit.width));\n const suffixWidths = buildSuffixWidths(units.map((unit) => unit.width));\n const { prefixCount, suffixCount } = selectEllipsisUnitCounts({\n position,\n prefixWidths,\n suffixWidths,\n unitCount: units.length,\n availableWidth,\n });\n\n const prefixWidth = prefixWidths[prefixCount] ?? 0;\n const suffixWidth = suffixWidths[suffixCount] ?? 0;\n const prefixText = joinUnitText(units, 0, prefixCount);\n const suffixText = joinUnitText(units, units.length - suffixCount, units.length);\n\n return {\n width: prefixWidth + ellipsisWidth + suffixWidth,\n text: `${prefixText}${ELLIPSIS_GLYPH}${suffixText}`,\n shift,\n overflowed: true,\n };\n}\n\nfunction layoutForcedEllipsizedLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n shift: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): OverflowTextLayout {\n if (text.length === 0) {\n return createEllipsisOnlyLayout(ctx, maxWidth, shift);\n }\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n return layoutPreparedEllipsis(ctx, prepared, text, maxWidth, shift, \"end\", true);\n}\n\nexport function layoutFirstLineIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextLayout {\n const firstLine = readPreparedFirstLine(ctx, text, whiteSpace, wordBreak);\n if (firstLine == null) {\n return { width: 0, text: \"\", shift: 0 };\n }\n const shift = measureFontShift(ctx);\n return {\n width: measureNaturalWidth(firstLine.prepared),\n text: firstLine.text,\n shift,\n };\n}\n\nexport function measureTextIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextMeasurement {\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const { maxLineWidth: width, lineCount } = measureLineStats(prepared, INTRINSIC_MAX_WIDTH);\n if (lineCount === 0) {\n return { width: 0, lineCount: 0 };\n }\n return { width, lineCount };\n}\n\nexport function layoutTextIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextBlockLayout {\n const intrinsic = layoutWithLines(\n readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak),\n INTRINSIC_MAX_WIDTH,\n 0,\n );\n if (intrinsic.lines.length === 0) {\n return { width: 0, lines: [] };\n }\n return toTextBlockLayout(intrinsic.lines, measureFontShift(ctx));\n}\n\nexport function layoutFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n const shift = measureFontShift(ctx);\n if (clampedMaxWidth === 0) {\n return { width: 0, text: \"\", shift };\n }\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const line = layoutNextLine(prepared, LINE_START_CURSOR, clampedMaxWidth);\n if (line == null) {\n return { width: 0, text: \"\", shift };\n }\n return { width: line.width, text: line.text, shift };\n}\n\nexport function layoutEllipsizedFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n ellipsisPosition: TextEllipsisPosition = \"end\",\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): OverflowTextLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n const firstLine = readPreparedFirstLine(ctx, text, whiteSpace, wordBreak);\n if (firstLine == null) {\n return { width: 0, text: \"\", shift: 0, overflowed: false };\n }\n const shift = measureFontShift(ctx);\n if (clampedMaxWidth === 0) {\n return { width: 0, text: \"\", shift, overflowed: true };\n }\n return layoutPreparedEllipsis(ctx, firstLine.prepared, firstLine.text, clampedMaxWidth, shift, ellipsisPosition);\n}\n\nexport function measureText<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextMeasurement {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n if (clampedMaxWidth === 0) {\n return { width: 0, lineCount: 0 };\n }\n\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const { maxLineWidth: width, lineCount } = measureLineStats(prepared, clampedMaxWidth);\n return { width, lineCount };\n}\n\nexport function measureTextMinContent<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n overflowWrap: TextOverflowWrapMode = \"break-word\",\n): TextMeasurement {\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n if (prepared.widths.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n\n const width = measurePreparedMinContentWidth(prepared, overflowWrap);\n const lineMaxWidth = Math.max(width, MIN_CONTENT_WIDTH_EPSILON);\n const { lineCount } = measureLineStats(prepared, lineMaxWidth);\n return { width, lineCount };\n}\n\nexport function layoutText<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextBlockLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n if (clampedMaxWidth === 0) {\n return { width: 0, lines: [] };\n }\n\n const layout = layoutWithLines(\n readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak),\n clampedMaxWidth,\n 0,\n );\n if (layout.lines.length === 0) {\n return { width: 0, lines: [] };\n }\n return toTextBlockLayout(layout.lines, measureFontShift(ctx));\n}\n\nexport function layoutTextWithOverflow<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n options: {\n whiteSpace?: TextWhiteSpaceMode;\n wordBreak?: TextWordBreakMode;\n overflow?: TextOverflowMode;\n maxLines?: number;\n } = {},\n): OverflowTextBlockLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n const whiteSpace = options.whiteSpace ?? \"normal\";\n const wordBreak = options.wordBreak ?? \"normal\";\n const overflow = options.overflow ?? \"clip\";\n const normalizedMaxLines = normalizeMaxLines(options.maxLines);\n\n const layout = layoutText(ctx, text, clampedMaxWidth, whiteSpace, wordBreak);\n if (normalizedMaxLines == null || layout.lines.length <= normalizedMaxLines) {\n return {\n width: layout.width,\n lines: layout.lines.map((line) => ({ ...line, overflowed: false })),\n overflowed: false,\n };\n }\n\n const visibleLines = layout.lines.slice(0, normalizedMaxLines);\n if (overflow !== \"ellipsis\") {\n return {\n width: visibleLines.reduce((lineWidth, line) => Math.max(lineWidth, line.width), 0),\n lines: visibleLines.map((line) => ({ ...line, overflowed: false })),\n overflowed: true,\n };\n }\n\n const shift = visibleLines[visibleLines.length - 1]?.shift ?? measureFontShift(ctx);\n const lastVisibleLine = visibleLines[visibleLines.length - 1];\n const ellipsizedLastLine = lastVisibleLine == null || lastVisibleLine.text.length === 0\n ? createEllipsisOnlyLayout(ctx, clampedMaxWidth, shift)\n : layoutForcedEllipsizedLine(ctx, lastVisibleLine.text, clampedMaxWidth, shift, whiteSpace, wordBreak);\n\n const lines = [\n ...visibleLines.slice(0, -1).map((line) => ({ ...line, overflowed: false })),\n { ...ellipsizedLastLine, shift },\n ];\n return {\n width: lines.reduce((lineWidth, line) => Math.max(lineWidth, line.width), 0),\n lines,\n overflowed: true,\n };\n}\n","import {\n layoutNextRichInlineLineRange,\n materializeRichInlineLineRange,\n measureRichInlineStats,\n prepareRichInline,\n walkRichInlineLineRanges,\n type PreparedRichInline,\n type RichInlineLineRange,\n} from \"@chenglou/pretext/rich-inline\";\nimport type {\n Context,\n DynValue,\n InlineSpan,\n TextEllipsisPosition,\n TextOverflowMode,\n TextOverflowWrapMode,\n} from \"../types\";\nimport {\n ELLIPSIS_GLYPH,\n INTRINSIC_MAX_WIDTH,\n MIN_CONTENT_WIDTH_EPSILON,\n buildPrefixWidths,\n measureEllipsisWidth,\n measureFontShift,\n normalizeMaxLines,\n readLruValue,\n selectEllipsisUnitCounts,\n writeLruValue,\n} from \"./core\";\nimport { getPreparedUnits, measurePreparedMinContentWidth, readPreparedText } from \"./plain-core\";\n\nconst RICH_PREPARED_CACHE_CAPACITY = 256;\n\nconst richPreparedCache = new Map<string, PreparedRichInline>();\n\nexport interface RichFragmentLayout {\n itemIndex: number;\n text: string;\n font: string;\n color: DynValue<any, string> | undefined;\n gapBefore: number;\n occupiedWidth: number;\n shift: number;\n}\n\nexport interface RichLineLayout {\n width: number;\n fragments: RichFragmentLayout[];\n overflowed: boolean;\n}\n\nexport interface RichBlockLayout {\n width: number;\n lines: RichLineLayout[];\n overflowed: boolean;\n}\n\nexport interface RichMeasurement {\n width: number;\n lineCount: number;\n}\n\ntype RichUnitLayout = {\n fragmentIndex: number;\n itemIndex: number;\n text: string;\n width: number;\n font: string;\n color: DynValue<any, string> | undefined;\n leadingGap: number;\n};\n\nfunction withFont<C extends CanvasRenderingContext2D, T>(ctx: Context<C>, font: string, cb: () => T): T {\n const previousFont = ctx.graphics.font;\n ctx.graphics.font = font;\n try {\n return cb();\n } finally {\n ctx.graphics.font = previousFont;\n }\n}\n\nfunction getRichPreparedCacheKey<C extends CanvasRenderingContext2D>(\n spans: InlineSpan<C>[],\n defaultFont: string,\n): string {\n return spans\n .map((span) => `${span.font ?? defaultFont}\\u0000${span.text}\\u0000${span.break ?? \"\"}\\u0000${span.extraWidth ?? 0}`)\n .join(\"\\u0001\");\n}\n\nfunction readRichPrepared<C extends CanvasRenderingContext2D>(\n spans: InlineSpan<C>[],\n defaultFont: string,\n): PreparedRichInline {\n const key = getRichPreparedCacheKey(spans, defaultFont);\n const cached = readLruValue(richPreparedCache, key);\n if (cached != null) {\n return cached;\n }\n const items = spans.map((span) => ({\n text: span.text,\n font: span.font ?? defaultFont,\n break: span.break,\n extraWidth: span.extraWidth,\n }));\n return writeLruValue(richPreparedCache, key, prepareRichInline(items), RICH_PREPARED_CACHE_CAPACITY);\n}\n\nfunction measureRichFragmentShift<C extends CanvasRenderingContext2D>(ctx: Context<C>, font: string): number {\n return withFont(ctx, font, () => measureFontShift(ctx));\n}\n\nfunction materializeRichLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n lineRange: RichInlineLineRange,\n overflowed: boolean,\n): RichLineLayout {\n const prepared = readRichPrepared(spans, defaultFont);\n const richLine = materializeRichInlineLineRange(prepared, lineRange);\n const fragments: RichFragmentLayout[] = richLine.fragments.map((fragment) => {\n const span = spans[fragment.itemIndex];\n const font = span?.font ?? defaultFont;\n const color = span?.color ?? defaultColor;\n return {\n itemIndex: fragment.itemIndex,\n text: fragment.text,\n font,\n color: color as DynValue<any, string>,\n gapBefore: fragment.gapBefore,\n occupiedWidth: fragment.occupiedWidth,\n shift: measureRichFragmentShift(ctx, font),\n };\n });\n return { width: richLine.width, fragments, overflowed };\n}\n\nfunction flattenRichLineUnits(line: RichLineLayout): RichUnitLayout[] {\n const units: RichUnitLayout[] = [];\n for (let fragmentIndex = 0; fragmentIndex < line.fragments.length; fragmentIndex += 1) {\n const fragment = line.fragments[fragmentIndex]!;\n const prepared = readPreparedText(fragment.text, fragment.font, \"normal\", \"normal\");\n const fragmentUnits = getPreparedUnits(prepared);\n if (fragmentUnits.length === 0) {\n continue;\n }\n\n const textWidth = fragmentUnits.reduce((total, unit) => total + unit.width, 0);\n const trailingExtraWidth = Math.max(0, fragment.occupiedWidth - textWidth);\n\n for (let unitIndex = 0; unitIndex < fragmentUnits.length; unitIndex += 1) {\n const unit = fragmentUnits[unitIndex]!;\n units.push({\n fragmentIndex,\n itemIndex: fragment.itemIndex,\n text: unit.text,\n width: unit.width + (unitIndex === fragmentUnits.length - 1 ? trailingExtraWidth : 0),\n font: fragment.font,\n color: fragment.color,\n leadingGap: unitIndex === 0 ? fragment.gapBefore : 0,\n });\n }\n }\n return units;\n}\n\nfunction buildRichPrefixWidths(units: readonly RichUnitLayout[]): number[] {\n return buildPrefixWidths(units.map((unit) => unit.leadingGap + unit.width));\n}\n\nfunction buildRichSuffixWidths(units: readonly RichUnitLayout[]): number[] {\n const widths = [0];\n let total = 0;\n for (let index = units.length - 1; index >= 0; index -= 1) {\n const unit = units[index]!;\n total += unit.width;\n if (widths.length > 1) {\n total += unit.leadingGap;\n }\n widths.push(total);\n }\n return widths;\n}\n\nfunction materializeRichFragmentsFromUnits(\n units: readonly RichUnitLayout[],\n start: number,\n end: number,\n suppressLeadingGap: boolean,\n): RichFragmentLayout[] {\n const fragments: RichFragmentLayout[] = [];\n for (let index = start; index < end; index += 1) {\n const unit = units[index]!;\n const previous = fragments[fragments.length - 1];\n const previousUnit = units[index - 1];\n if (previous != null && previousUnit != null && previousUnit.fragmentIndex === unit.fragmentIndex) {\n previous.text += unit.text;\n previous.occupiedWidth += unit.width;\n continue;\n }\n\n fragments.push({\n itemIndex: unit.itemIndex,\n text: unit.text,\n font: unit.font,\n color: unit.color,\n gapBefore: fragments.length === 0 && suppressLeadingGap ? 0 : unit.leadingGap,\n occupiedWidth: unit.width,\n shift: 0,\n });\n }\n return fragments;\n}\n\nfunction measureRichFragmentsShift<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n fragments: readonly RichFragmentLayout[],\n): RichFragmentLayout[] {\n return fragments.map((fragment) => ({\n ...fragment,\n shift: measureRichFragmentShift(ctx, fragment.font),\n }));\n}\n\nfunction createRichEllipsisFragment<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n font: string,\n color: DynValue<C, string> | undefined,\n): RichFragmentLayout {\n return withFont(ctx, font, () => ({\n itemIndex: -1,\n text: ELLIPSIS_GLYPH,\n font,\n color,\n gapBefore: 0,\n occupiedWidth: measureEllipsisWidth(ctx),\n shift: measureFontShift(ctx),\n }));\n}\n\nfunction createRichEllipsisOnlyLayout<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n maxWidth: number,\n font: string,\n color: DynValue<C, string> | undefined,\n): RichLineLayout {\n const ellipsis = createRichEllipsisFragment(ctx, font, color);\n if (ellipsis.occupiedWidth > maxWidth) {\n return { width: 0, fragments: [], overflowed: true };\n }\n return { width: ellipsis.occupiedWidth, fragments: [ellipsis], overflowed: true };\n}\n\nfunction layoutPreparedRichEllipsis<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n line: RichLineLayout,\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n position: TextEllipsisPosition,\n): RichLineLayout {\n if (!line.overflowed && line.width <= maxWidth) {\n return { ...line, overflowed: false };\n }\n\n const units = flattenRichLineUnits(line);\n const fallbackFragment = line.fragments[0];\n const fallbackFont = fallbackFragment?.font ?? defaultFont;\n const fallbackColor = (fallbackFragment?.color ?? defaultColor) as DynValue<C, string>;\n const ellipsisOnly = createRichEllipsisOnlyLayout(ctx, maxWidth, fallbackFont, fallbackColor);\n if (ellipsisOnly.fragments.length === 0 || units.length === 0) {\n return ellipsisOnly;\n }\n\n const ellipsisWidth = ellipsisOnly.width;\n const availableWidth = Math.max(0, maxWidth - ellipsisWidth);\n const prefixWidths = buildRichPrefixWidths(units);\n const suffixWidths = buildRichSuffixWidths(units);\n const { prefixCount, suffixCount } = selectEllipsisUnitCounts({\n position,\n prefixWidths,\n suffixWidths,\n unitCount: units.length,\n availableWidth,\n getMaxSuffixCount: position === \"middle\"\n ? (nextPrefixCount) => Math.max(0, units.length - nextPrefixCount - 1)\n : undefined,\n });\n\n const prefixFragments = measureRichFragmentsShift(ctx, materializeRichFragmentsFromUnits(units, 0, prefixCount, false));\n const suffixStartIndex = units.length - suffixCount;\n const suffixFragments = measureRichFragmentsShift(\n ctx,\n materializeRichFragmentsFromUnits(units, suffixStartIndex, units.length, true),\n );\n\n const ellipsisSource =\n position === \"start\"\n ? (suffixFragments[0] ?? line.fragments[0])\n : position === \"middle\"\n ? (prefixFragments[prefixFragments.length - 1] ?? suffixFragments[0] ?? line.fragments[line.fragments.length - 1])\n : (prefixFragments[prefixFragments.length - 1] ?? line.fragments[line.fragments.length - 1]);\n const ellipsis = createRichEllipsisFragment(\n ctx,\n ellipsisSource?.font ?? defaultFont,\n (ellipsisSource?.color ?? defaultColor) as DynValue<C, string>,\n );\n\n const fragments =\n position === \"start\"\n ? [ellipsis, ...suffixFragments]\n : position === \"middle\"\n ? [...prefixFragments, ellipsis, ...suffixFragments]\n : [...prefixFragments, ellipsis];\n const width =\n position === \"start\"\n ? ellipsis.occupiedWidth + (suffixWidths[suffixCount] ?? 0)\n : position === \"middle\"\n ? (prefixWidths[prefixCount] ?? 0) + ellipsis.occupiedWidth + (suffixWidths[suffixCount] ?? 0)\n : (prefixWidths[prefixCount] ?? 0) + ellipsis.occupiedWidth;\n\n return { width, fragments, overflowed: true };\n}\n\nexport function layoutRichFirstLineIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichLineLayout {\n if (spans.length === 0) {\n return { width: 0, fragments: [], overflowed: false };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineRange = layoutNextRichInlineLineRange(prepared, INTRINSIC_MAX_WIDTH);\n if (lineRange == null) {\n return { width: 0, fragments: [], overflowed: false };\n }\n return materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, false);\n}\n\nexport function layoutRichFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichLineLayout {\n const clampedMaxWidth = Math.max(0, maxWidth);\n if (spans.length === 0 || clampedMaxWidth === 0) {\n return { width: 0, fragments: [], overflowed: false };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineRange = layoutNextRichInlineLineRange(prepared, clampedMaxWidth);\n if (lineRange == null) {\n return { width: 0, fragments: [], overflowed: false };\n }\n return materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, false);\n}\n\nexport function layoutRichEllipsizedFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n ellipsisPosition: TextEllipsisPosition = \"end\",\n): RichLineLayout {\n const clampedMaxWidth = Math.max(0, maxWidth);\n const intrinsicLine = layoutRichFirstLineIntrinsic(ctx, spans, defaultFont, defaultColor);\n if (intrinsicLine.fragments.length === 0) {\n return { ...intrinsicLine, overflowed: false };\n }\n if (clampedMaxWidth === 0) {\n return { width: 0, fragments: [], overflowed: true };\n }\n return layoutPreparedRichEllipsis(ctx, intrinsicLine, clampedMaxWidth, defaultFont, defaultColor, ellipsisPosition);\n}\n\nexport function measureRichText<C extends CanvasRenderingContext2D>(\n _ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n): RichMeasurement {\n if (spans.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const { maxLineWidth: width, lineCount } = measureRichInlineStats(prepared, maxWidth);\n return { width, lineCount };\n}\n\nexport function measureRichTextIntrinsic<C extends CanvasRenderingContext2D>(\n _ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n): RichMeasurement {\n if (spans.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const { maxLineWidth: width, lineCount } = measureRichInlineStats(prepared, INTRINSIC_MAX_WIDTH);\n return { width, lineCount };\n}\n\nexport function measureRichTextMinContent<C extends CanvasRenderingContext2D>(\n _ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n overflowWrap: TextOverflowWrapMode = \"break-word\",\n): RichMeasurement {\n if (spans.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n let maxWidth = 0;\n for (const span of spans) {\n if (span.text.trim().length === 0) {\n continue;\n }\n const font = span.font ?? defaultFont;\n const prepared = readPreparedText(span.text, font, \"normal\", \"normal\");\n const spanMinWidth = measurePreparedMinContentWidth(prepared, overflowWrap) + (span.extraWidth ?? 0);\n if (spanMinWidth > maxWidth) {\n maxWidth = spanMinWidth;\n }\n }\n if (maxWidth === 0) {\n return { width: 0, lineCount: 0 };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineMaxWidth = Math.max(maxWidth, MIN_CONTENT_WIDTH_EPSILON);\n const { lineCount } = measureRichInlineStats(prepared, lineMaxWidth);\n return { width: maxWidth, lineCount };\n}\n\nexport function layoutRichText<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichBlockLayout {\n if (spans.length === 0) {\n return { width: 0, lines: [], overflowed: false };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineRanges: RichInlineLineRange[] = [];\n walkRichInlineLineRanges(prepared, maxWidth, (lineRange) => lineRanges.push(lineRange));\n if (lineRanges.length === 0) {\n return { width: 0, lines: [], overflowed: false };\n }\n const lines = lineRanges.map((lineRange) => materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, false));\n const width = lines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0);\n return { width, lines, overflowed: false };\n}\n\nexport function layoutRichTextIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichBlockLayout {\n return layoutRichText(ctx, spans, INTRINSIC_MAX_WIDTH, defaultFont, defaultColor);\n}\n\nexport function layoutRichTextWithOverflow<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n maxLines?: number,\n overflow: TextOverflowMode = \"clip\",\n): RichBlockLayout {\n if (spans.length === 0) {\n return { width: 0, lines: [], overflowed: false };\n }\n\n const normalizedMaxLines = normalizeMaxLines(maxLines);\n const layout = layoutRichText(ctx, spans, maxWidth, defaultFont, defaultColor);\n if (normalizedMaxLines == null || layout.lines.length <= normalizedMaxLines) {\n return layout;\n }\n\n const visibleLines = layout.lines.slice(0, normalizedMaxLines);\n if (overflow !== \"ellipsis\") {\n return {\n width: visibleLines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0),\n lines: visibleLines,\n overflowed: true,\n };\n }\n\n const lastVisibleLine = visibleLines[visibleLines.length - 1];\n const ellipsizedLastLine = lastVisibleLine == null\n ? { width: 0, fragments: [], overflowed: true }\n : layoutPreparedRichEllipsis(\n ctx,\n { ...lastVisibleLine, overflowed: true },\n maxWidth,\n defaultFont,\n defaultColor,\n \"end\",\n );\n\n const lines = [...visibleLines.slice(0, -1), ellipsizedLastLine];\n return {\n width: lines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0),\n lines,\n overflowed: true,\n };\n}\n","import {\n layoutEllipsizedFirstLine,\n layoutFirstLine,\n layoutFirstLineIntrinsic,\n layoutRichEllipsizedFirstLine,\n layoutRichFirstLine,\n layoutRichFirstLineIntrinsic,\n layoutRichText,\n layoutRichTextIntrinsic,\n layoutRichTextWithOverflow,\n layoutText,\n layoutTextIntrinsic,\n layoutTextWithOverflow,\n measureRichText,\n measureRichTextIntrinsic,\n measureRichTextMinContent,\n measureText,\n measureTextMinContent,\n measureTextIntrinsic,\n type RichBlockLayout,\n type RichLineLayout,\n type RichMeasurement,\n type TextLayout,\n type TextMeasurement,\n} from \"../text\";\nimport type { Box, Context, InlineSpan, MultilineTextOptions, Node, PhysicalTextAlign, TextOptions } from \"../types\";\n\ntype SingleLineLayout = TextLayout;\ntype MultiLineDrawLayout = {\n width: number;\n lines: TextLayout[];\n};\ntype MultiLineMeasureLayout = TextMeasurement;\n\ntype TextLayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getTextLayout<T>(node: Node<C>, key: string): T | undefined;\n setTextLayout<T>(node: Node<C>, key: string, layout: T): void;\n};\n\nfunction resolvePhysicalTextAlign(\n options: Pick<MultilineTextOptions<any>, \"align\" | \"physicalAlign\">,\n): PhysicalTextAlign {\n if (options.physicalAlign != null) {\n return options.physicalAlign;\n }\n if (options.align != null) {\n switch (options.align) {\n case \"start\":\n return \"left\";\n case \"center\":\n return \"center\";\n case \"end\":\n return \"right\";\n }\n }\n return \"left\";\n}\n\nfunction normalizeTextMaxWidth(maxWidth: number | undefined): number | undefined {\n if (maxWidth == null) {\n return undefined;\n }\n return Math.max(0, maxWidth);\n}\n\nfunction getTextLayoutContext<C extends CanvasRenderingContext2D>(ctx: Context<C>): Context<C> & TextLayoutCacheAccess<C> {\n return ctx as Context<C> & TextLayoutCacheAccess<C>;\n}\n\nfunction readCachedTextLayout<C extends CanvasRenderingContext2D, T>(\n node: Node<C>,\n ctx: Context<C>,\n key: string,\n compute: () => T,\n): T {\n const textCtx = getTextLayoutContext(ctx);\n const cached = textCtx.getTextLayout<T>(node, key);\n if (cached != null) {\n return cached;\n }\n const layout = compute();\n textCtx.setTextLayout(node, key, layout);\n return layout;\n}\n\nfunction getSingleLineLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"single:intrinsic\" : `single:${maxWidth}`;\n}\n\nfunction getMultiLineMeasureLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"multi:measure:intrinsic\" : `multi:measure:${maxWidth}`;\n}\n\nfunction getRichMultiLineMeasureLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"rich:measure:intrinsic\" : `rich:measure:${maxWidth}`;\n}\n\nfunction getMultiLineDrawLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"multi:draw:intrinsic\" : `multi:draw:${maxWidth}`;\n}\n\nfunction getRichMultiLineDrawLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"rich:draw:intrinsic\" : `rich:draw:${maxWidth}`;\n}\n\nfunction getMultiLineOverflowLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"multi:overflow:intrinsic\" : `multi:overflow:${maxWidth}`;\n}\n\nfunction getRichMultiLineOverflowLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"rich:overflow:intrinsic\" : `rich:overflow:${maxWidth}`;\n}\n\nfunction shouldUseMultilineOverflowLayout(options: Pick<MultilineTextOptions<any>, \"maxLines\">): boolean {\n return options.maxLines != null;\n}\n\nfunction shouldReadConstrainedOverflowLayout(\n maxWidth: number | undefined,\n options: Pick<MultilineTextOptions<any>, \"maxLines\">,\n): maxWidth is number {\n return maxWidth != null && shouldUseMultilineOverflowLayout(options);\n}\n\nfunction measureBlockLayout<T extends { width: number; lines: ArrayLike<unknown> }>(layout: T): MultiLineMeasureLayout {\n return { width: layout.width, lineCount: layout.lines.length };\n}\n\nfunction getSingleLineMinContentLayoutKey(): string {\n return \"single:min-content\";\n}\n\nfunction getMultiLineMinContentLayoutKey(): string {\n return \"multi:min-content\";\n}\n\nfunction getRichMultiLineMinContentLayoutKey(): string {\n return \"rich:min-content\";\n}\n\nfunction getSingleLineLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: TextOptions<C>,\n): SingleLineLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getSingleLineLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutFirstLineIntrinsic(ctx, text, options.whiteSpace, options.wordBreak)\n : options.overflow === \"ellipsis\"\n ? layoutEllipsizedFirstLine(\n ctx,\n text,\n maxWidth,\n options.ellipsisPosition ?? \"end\",\n options.whiteSpace,\n options.wordBreak,\n )\n : layoutFirstLine(ctx, text, maxWidth, options.whiteSpace, options.wordBreak)\n );\n}\n\nfunction getRichSingleLineLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: TextOptions<C>,\n): RichLineLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getSingleLineLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutRichFirstLineIntrinsic(ctx, spans, options.font, options.color)\n : options.overflow === \"ellipsis\"\n ? layoutRichEllipsizedFirstLine(ctx, spans, maxWidth, options.font, options.color, options.ellipsisPosition ?? \"end\")\n : layoutRichFirstLine(ctx, spans, maxWidth, options.font, options.color)\n );\n}\n\nfunction getMultiLineOverflowLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: MultilineTextOptions<C>,\n): MultiLineDrawLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getMultiLineOverflowLayoutKey(maxWidth), () =>\n layoutTextWithOverflow(ctx, text, maxWidth ?? 0, {\n whiteSpace: options.whiteSpace,\n wordBreak: options.wordBreak,\n overflow: options.overflow,\n maxLines: options.maxLines,\n })\n );\n}\n\nfunction getMultiLineMeasureLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: MultilineTextOptions<C>,\n): MultiLineMeasureLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return measureBlockLayout(getMultiLineOverflowLayout(node, ctx, text, options));\n }\n return readCachedTextLayout(node, ctx, getMultiLineMeasureLayoutKey(maxWidth), () =>\n maxWidth == null\n ? measureTextIntrinsic(ctx, text, options.whiteSpace, options.wordBreak)\n : measureText(ctx, text, maxWidth, options.whiteSpace, options.wordBreak)\n );\n}\n\nfunction getMultiLineDrawLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: MultilineTextOptions<C>,\n): MultiLineDrawLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return getMultiLineOverflowLayout(node, ctx, text, options);\n }\n return readCachedTextLayout(node, ctx, getMultiLineDrawLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutTextIntrinsic(ctx, text, options.whiteSpace, options.wordBreak)\n : layoutText(ctx, text, maxWidth, options.whiteSpace, options.wordBreak)\n );\n}\n\nfunction getSingleLineMinContentLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: TextOptions<C>,\n): SingleLineLayout {\n return readCachedTextLayout(node, ctx, getSingleLineMinContentLayoutKey(), () => {\n const measurement = measureTextMinContent(ctx, text, options.whiteSpace, options.wordBreak, options.overflowWrap);\n const { shift } = layoutFirstLineIntrinsic(ctx, text, options.whiteSpace, options.wordBreak);\n return {\n width: measurement.width,\n text,\n shift,\n };\n });\n}\n\nfunction getRichSingleLineMinContentWidth<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: TextOptions<C>,\n): number {\n return readCachedTextLayout(node, ctx, getSingleLineMinContentLayoutKey(), () =>\n measureRichTextMinContent(ctx, spans, options.font, options.overflowWrap).width\n );\n}\n\nfunction drawRichLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n line: RichLineLayout,\n fallbackColor: TextOptions<C>[\"color\"],\n x: number,\n y: number,\n lineHeight: number,\n): void {\n let cursorX = x;\n for (let fragmentIndex = 0; fragmentIndex < line.fragments.length; fragmentIndex += 1) {\n const fragment = line.fragments[fragmentIndex]!;\n cursorX += fragment.gapBefore;\n ctx.with((g) => {\n g.font = fragment.font;\n g.fillStyle = ctx.resolveDynValue((fragment.color ?? fallbackColor) as typeof fallbackColor);\n g.textAlign = \"left\";\n g.fillText(fragment.text, cursorX, y + (lineHeight + fragment.shift) / 2);\n });\n cursorX += fragment.occupiedWidth;\n }\n}\n\nfunction getMultiLineMinContentLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n whiteSpace: MultilineTextOptions<C>[\"whiteSpace\"],\n wordBreak: MultilineTextOptions<C>[\"wordBreak\"],\n overflowWrap: MultilineTextOptions<C>[\"overflowWrap\"],\n): MultiLineMeasureLayout {\n return readCachedTextLayout(node, ctx, getMultiLineMinContentLayoutKey(), () =>\n measureTextMinContent(ctx, text, whiteSpace, wordBreak, overflowWrap)\n );\n}\n\nfunction getRichMultiLineMeasureLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichMeasurement {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return measureBlockLayout(getRichMultiLineOverflowLayout(node, ctx, spans, options));\n }\n return readCachedTextLayout(node, ctx, getRichMultiLineMeasureLayoutKey(maxWidth), () =>\n maxWidth == null\n ? measureRichTextIntrinsic(ctx, spans, options.font)\n : measureRichText(ctx, spans, maxWidth, options.font)\n );\n}\n\nfunction getRichMultiLineOverflowLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichBlockLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getRichMultiLineOverflowLayoutKey(maxWidth), () =>\n layoutRichTextWithOverflow(ctx, spans, maxWidth ?? 0, options.font, options.color, options.maxLines, options.overflow)\n );\n}\n\nfunction getRichMultiLineDrawLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichBlockLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return getRichMultiLineOverflowLayout(node, ctx, spans, options);\n }\n return readCachedTextLayout(node, ctx, getRichMultiLineDrawLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutRichTextIntrinsic(ctx, spans, options.font, options.color)\n : layoutRichText(ctx, spans, maxWidth, options.font, options.color)\n );\n}\n\nfunction getRichMultiLineMinContentLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichMeasurement {\n return readCachedTextLayout(node, ctx, getRichMultiLineMinContentLayoutKey(), () =>\n measureRichTextMinContent(ctx, spans, options.font, options.overflowWrap)\n );\n}\n\n/**\n * Draws wrapped text using the configured line height and alignment.\n * Accepts either a plain string or an array of `InlineSpan` items for mixed inline styles.\n */\nexport class MultilineText<C extends CanvasRenderingContext2D> implements Node<C> {\n /**\n * @param text Source text to measure and draw. Pass an `InlineSpan[]` for mixed inline styles.\n * @param options Text layout and drawing options.\n */\n constructor(\n readonly text: string | InlineSpan<C>[],\n readonly options: MultilineTextOptions<C>,\n ) {}\n\n measure(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const spans = this.text;\n const { width, lineCount } = getRichMultiLineMeasureLayout(this, ctx, spans, this.options);\n return { width, height: lineCount * this.options.lineHeight };\n }\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width, lineCount } = getMultiLineMeasureLayout(this, ctx, this.text as string, this.options);\n return { width, height: lineCount * this.options.lineHeight };\n });\n }\n\n measureMinContent(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const spans = this.text;\n const { width, lineCount } = getRichMultiLineMinContentLayout(this, ctx, spans, this.options);\n return { width, height: lineCount * this.options.lineHeight };\n }\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width, lineCount } = getMultiLineMinContentLayout(\n this,\n ctx,\n this.text as string,\n this.options.whiteSpace,\n this.options.wordBreak,\n this.options.overflowWrap,\n );\n return { width, height: lineCount * this.options.lineHeight };\n });\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n if (typeof this.text !== \"string\") {\n const spans = this.text;\n const { width, lines } = getRichMultiLineDrawLayout(this, ctx, spans, this.options);\n const align = resolvePhysicalTextAlign(this.options);\n const startX = align === \"right\" ? x + width : align === \"center\" ? x + width / 2 : x;\n for (const line of lines) {\n let cursorX = startX;\n for (let fi = 0; fi < line.fragments.length; fi++) {\n const frag = line.fragments[fi]!;\n cursorX += frag.gapBefore;\n ctx.with((g) => {\n g.font = frag.font;\n g.fillStyle = ctx.resolveDynValue((frag.color ?? this.options.color) as typeof this.options.color);\n if (align === \"right\") {\n g.textAlign = \"right\";\n } else if (align === \"center\") {\n g.textAlign = \"center\";\n } else {\n g.textAlign = \"left\";\n }\n g.fillText(frag.text, cursorX, y + (this.options.lineHeight + frag.shift) / 2);\n });\n cursorX += frag.occupiedWidth;\n }\n y += this.options.lineHeight;\n }\n return false;\n }\n return ctx.with((g) => {\n g.font = this.options.font;\n g.fillStyle = ctx.resolveDynValue(this.options.color);\n const { width, lines } = getMultiLineDrawLayout(this, ctx, this.text as string, this.options);\n switch (resolvePhysicalTextAlign(this.options)) {\n case \"left\":\n for (const { text, shift } of lines) {\n g.fillText(text, x, y + (this.options.lineHeight + shift) / 2);\n y += this.options.lineHeight;\n }\n break;\n case \"right\": {\n x += width;\n g.textAlign = \"right\";\n for (const { text, shift } of lines) {\n g.fillText(text, x, y + (this.options.lineHeight + shift) / 2);\n y += this.options.lineHeight;\n }\n break;\n }\n case \"center\": {\n x += width / 2;\n g.textAlign = \"center\";\n for (const { text, shift } of lines) {\n g.fillText(text, x, y + (this.options.lineHeight + shift) / 2);\n y += this.options.lineHeight;\n }\n break;\n }\n }\n return false;\n });\n }\n\n hittest(_ctx: Context<C>, _test: { x: number; y: number; type: \"click\" | \"auxclick\" | \"hover\" }): boolean {\n return false;\n }\n}\n\n/**\n * Draws a single line of text, clipped logically by measurement width.\n */\nexport class Text<C extends CanvasRenderingContext2D> implements Node<C> {\n /**\n * @param text Source text to measure and draw. Pass an `InlineSpan[]` for mixed inline styles.\n * @param options Text layout and drawing options.\n */\n constructor(\n readonly text: string | InlineSpan<C>[],\n readonly options: TextOptions<C>,\n ) {}\n\n measure(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const { width } = getRichSingleLineLayout(this, ctx, this.text, this.options);\n return { width, height: this.options.lineHeight };\n }\n const text = this.text;\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width } = getSingleLineLayout(this, ctx, text, this.options);\n return { width, height: this.options.lineHeight };\n });\n }\n\n measureMinContent(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const width = getRichSingleLineMinContentWidth(this, ctx, this.text, this.options);\n return { width, height: this.options.lineHeight };\n }\n const text = this.text;\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width } = getSingleLineMinContentLayout(this, ctx, text, this.options);\n return { width, height: this.options.lineHeight };\n });\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n if (typeof this.text !== \"string\") {\n const line = getRichSingleLineLayout(this, ctx, this.text, this.options);\n drawRichLine(ctx, line, this.options.color, x, y, this.options.lineHeight);\n return false;\n }\n const text = this.text;\n return ctx.with((g) => {\n g.font = this.options.font;\n g.fillStyle = ctx.resolveDynValue(this.options.color);\n const layout = getSingleLineLayout(this, ctx, text, this.options);\n g.fillText(layout.text, x, y + (this.options.lineHeight + layout.shift) / 2);\n return false;\n });\n }\n\n hittest(_ctx: Context<C>, _test: { x: number; y: number; type: \"click\" | \"auxclick\" | \"hover\" }): boolean {\n return false;\n }\n}\n","import { forEachNodeAncestor, getNodeRevision } from \"../internal/node-registry\";\nimport type {\n Box,\n Context,\n DynValue,\n FlexLayoutResult,\n HitTest,\n LayoutConstraints,\n Node,\n RendererOptions,\n} from \"../types\";\nimport { shallow } from \"../utils\";\n\nconst MAX_CONSTRAINT_VARIANTS = 8;\n\ntype LayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints): FlexLayoutResult<C> | undefined;\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints): void;\n};\n\ntype TextLayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getTextLayout<T>(node: Node<C>, key: string): T | undefined;\n setTextLayout<T>(node: Node<C>, key: string, layout: T): void;\n};\n\ntype BoxCacheEntry = {\n revision: number;\n box: Box;\n};\n\ntype LayoutCacheEntry<C extends CanvasRenderingContext2D> = {\n revision: number;\n layout: FlexLayoutResult<C>;\n};\n\ntype TextLayoutCacheEntry = {\n revision: number;\n layout: unknown;\n};\n\ntype RendererContext<C extends CanvasRenderingContext2D> = Context<C> & LayoutCacheAccess<C> & TextLayoutCacheAccess<C>;\n\nfunction constraintKey(constraints: LayoutConstraints | undefined): string {\n if (constraints == null) return \"\";\n return `${constraints.minWidth ?? \"\"},${constraints.maxWidth ?? \"\"},${constraints.minHeight ?? \"\"},${constraints.maxHeight ?? \"\"}`;\n}\n\n/**\n * Base renderer that provides measurement, layout caching, and drawing helpers.\n */\nexport class BaseRenderer<C extends CanvasRenderingContext2D, O extends {} = {}> {\n /** Canvas rendering context used by this renderer. */\n graphics: C;\n #ctx: RendererContext<C>;\n #lastWidth: number;\n #cache = new WeakMap<Node<C>, Map<string, BoxCacheEntry>>();\n #layoutCache = new WeakMap<Node<C>, Map<string, LayoutCacheEntry<C>>>();\n #textLayoutCache = new WeakMap<Node<C>, Map<string, TextLayoutCacheEntry>>();\n\n protected get context(): Context<C> {\n return shallow(this.#ctx);\n }\n\n /**\n * @param graphics Canvas rendering context used for all layout and drawing.\n * @param options Renderer-specific options.\n */\n constructor(\n graphics: C,\n readonly options: RendererOptions & O,\n ) {\n this.graphics = graphics;\n this.graphics.textRendering = \"optimizeLegibility\";\n const self = this;\n this.#ctx = {\n graphics: this.graphics,\n measureNode(node: Node<C>, constraints?: LayoutConstraints) {\n return self.measureNode(node, constraints);\n },\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints) {\n return self.getLayoutResult(node, constraints);\n },\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints) {\n self.setLayoutResult(node, result, constraints);\n },\n getTextLayout<T>(node: Node<C>, key: string) {\n return self.getTextLayout<T>(node, key);\n },\n setTextLayout<T>(node: Node<C>, key: string, layout: T) {\n self.setTextLayout(node, key, layout);\n },\n invalidateNode: this.invalidateNode.bind(this),\n resolveDynValue<T>(value: DynValue<C, T>): T {\n if (typeof value === \"function\") {\n return value(this.graphics);\n }\n return value as T;\n },\n with<T>(cb: (g: C) => T): T {\n this.graphics.save();\n try {\n return cb(this.graphics);\n } finally {\n this.graphics.restore();\n }\n },\n };\n this.#lastWidth = this.graphics.canvas.clientWidth;\n }\n\n #clearAllCaches(): void {\n this.#cache = new WeakMap<Node<C>, Map<string, BoxCacheEntry>>();\n this.#layoutCache = new WeakMap<Node<C>, Map<string, LayoutCacheEntry<C>>>();\n this.#textLayoutCache = new WeakMap<Node<C>, Map<string, TextLayoutCacheEntry>>();\n }\n\n #syncCachesToViewportWidth(): void {\n const width = this.graphics.canvas.clientWidth;\n if (this.#lastWidth === width) {\n return;\n }\n this.#clearAllCaches();\n this.#lastWidth = width;\n }\n\n protected getRootConstraints(): LayoutConstraints {\n return {\n maxWidth: this.graphics.canvas.clientWidth,\n };\n }\n\n protected getRootContext(): Context<C> {\n const ctx = this.context;\n ctx.constraints = this.getRootConstraints();\n return ctx;\n }\n\n protected measureRootNode(node: Node<C>): Box {\n return this.measureNode(node, this.getRootConstraints());\n }\n\n protected drawRootNode(node: Node<C>, x = 0, y = 0): boolean {\n this.measureRootNode(node);\n return node.draw(this.getRootContext(), x, y);\n }\n\n protected hittestRootNode(node: Node<C>, test: HitTest): boolean {\n this.measureRootNode(node);\n return node.hittest(this.getRootContext(), test);\n }\n\n /**\n * Drops cached measurements for a node and every ancestor that depends on it.\n */\n invalidateNode(node: Node<C>): void {\n this.#syncCachesToViewportWidth();\n this.#cache.delete(node);\n this.#layoutCache.delete(node);\n this.#textLayoutCache.delete(node);\n forEachNodeAncestor(node, (ancestor) => {\n this.#cache.delete(ancestor);\n this.#layoutCache.delete(ancestor);\n this.#textLayoutCache.delete(ancestor);\n });\n }\n\n /**\n * Returns the cached layout result for a node under the given constraints, if available.\n */\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints): FlexLayoutResult<C> | undefined {\n this.#syncCachesToViewportWidth();\n const nodeCache = this.#layoutCache.get(node);\n if (nodeCache == null) {\n return undefined;\n }\n const key = constraintKey(constraints);\n const cached = nodeCache.get(key);\n if (cached == null) {\n return undefined;\n }\n if (cached.revision !== getNodeRevision(node)) {\n nodeCache.delete(key);\n return undefined;\n }\n return cached.layout;\n }\n\n /**\n * Stores a layout result for later draw and hit-test passes.\n */\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints): void {\n this.#syncCachesToViewportWidth();\n let nodeCache = this.#layoutCache.get(node);\n if (nodeCache == null) {\n nodeCache = new Map();\n this.#layoutCache.set(node, nodeCache);\n } else if (nodeCache.size >= MAX_CONSTRAINT_VARIANTS) {\n const firstKey = nodeCache.keys().next().value!;\n nodeCache.delete(firstKey);\n }\n nodeCache.set(constraintKey(constraints), {\n revision: getNodeRevision(node),\n layout: result,\n });\n }\n\n protected getTextLayout<T>(node: Node<C>, key: string): T | undefined {\n this.#syncCachesToViewportWidth();\n const nodeCache = this.#textLayoutCache.get(node);\n if (nodeCache == null) {\n return undefined;\n }\n const cached = nodeCache.get(key);\n if (cached == null) {\n return undefined;\n }\n if (cached.revision !== getNodeRevision(node)) {\n nodeCache.delete(key);\n return undefined;\n }\n return cached.layout as T;\n }\n\n protected setTextLayout<T>(node: Node<C>, key: string, layout: T): void {\n this.#syncCachesToViewportWidth();\n let nodeCache = this.#textLayoutCache.get(node);\n if (nodeCache == null) {\n nodeCache = new Map();\n this.#textLayoutCache.set(node, nodeCache);\n } else if (nodeCache.size >= MAX_CONSTRAINT_VARIANTS) {\n const firstKey = nodeCache.keys().next().value!;\n nodeCache.delete(firstKey);\n }\n nodeCache.set(key, {\n revision: getNodeRevision(node),\n layout,\n });\n }\n\n /**\n * Measures a node under optional constraints, using cached results when possible.\n */\n measureNode(node: Node<C>, constraints?: LayoutConstraints): Box {\n this.#syncCachesToViewportWidth();\n {\n const nodeCache = this.#cache.get(node);\n if (nodeCache != null) {\n const key = constraintKey(constraints);\n const cached = nodeCache.get(key);\n if (cached != null) {\n if (cached.revision === getNodeRevision(node)) {\n return cached.box;\n }\n nodeCache.delete(key);\n }\n }\n }\n const ctx = this.context;\n if (constraints != null) {\n ctx.constraints = constraints;\n }\n const result = node.measure(ctx);\n const key = constraintKey(constraints);\n let nodeCache = this.#cache.get(node);\n if (nodeCache == null) {\n nodeCache = new Map();\n this.#cache.set(node, nodeCache);\n } else if (nodeCache.size >= MAX_CONSTRAINT_VARIANTS) {\n const firstKey = nodeCache.keys().next().value!;\n nodeCache.delete(firstKey);\n }\n nodeCache.set(key, {\n revision: getNodeRevision(node),\n box: result,\n });\n return result;\n }\n}\n\n/**\n * Immediate-mode renderer for a single root node.\n */\nexport class DebugRenderer<C extends CanvasRenderingContext2D> extends BaseRenderer<C> {\n /**\n * Clears the viewport and draws the provided root node.\n */\n draw(node: Node<C>): boolean {\n const { clientWidth: viewportWidth, clientHeight: viewportHeight } = this.graphics.canvas;\n this.graphics.clearRect(0, 0, viewportWidth, viewportHeight);\n return this.drawRootNode(node);\n }\n\n /**\n * Hit-tests the provided root node using viewport-relative coordinates.\n */\n hittest(node: Node<C>, test: HitTest): boolean {\n return this.hittestRootNode(node, test);\n }\n}\n","/**\n * Mutable list state shared with virtualized renderers.\n */\nexport interface ReplaceListItemAnimationOptions {\n /** Animation duration in milliseconds. */\n duration?: number;\n}\n\ntype ListReplaceChange<T extends {}> = {\n type: \"replace\";\n index: number;\n prevItem: T;\n nextItem: T;\n animation: ReplaceListItemAnimationOptions | undefined;\n};\n\ntype ListUnshiftChange = {\n type: \"unshift\";\n count: number;\n};\n\ntype ListPushChange = {\n type: \"push\";\n count: number;\n};\n\ntype ListResetChange = {\n type: \"reset\";\n};\n\ntype ListSetChange = {\n type: \"set\";\n};\n\nexport type ListStateChange<T extends {}> =\n | ListReplaceChange<T>\n | ListUnshiftChange\n | ListPushChange\n | ListResetChange\n | ListSetChange;\n\nexport type ListStateChangeListener<T extends {}> = (change: ListStateChange<T>) => void;\n\nconst listStateListeners = new WeakMap<ListState<{}>, Set<ListStateChangeListener<{}>>>();\n\nfunction emitListStateChange<T extends {}>(list: ListState<T>, change: ListStateChange<T>): void {\n const listeners = listStateListeners.get(list as unknown as ListState<{}>);\n if (listeners == null || listeners.size === 0) {\n return;\n }\n for (const listener of [...listeners]) {\n (listener as ListStateChangeListener<T>)(change);\n }\n}\n\nexport function subscribeListState<T extends {}>(list: ListState<T>, listener: ListStateChangeListener<T>): () => void {\n const key = list as unknown as ListState<{}>;\n let listeners = listStateListeners.get(key);\n if (listeners == null) {\n listeners = new Set();\n listStateListeners.set(key, listeners);\n }\n listeners.add(listener as ListStateChangeListener<{}>);\n return () => {\n const current = listStateListeners.get(key);\n if (current == null) {\n return;\n }\n current.delete(listener as ListStateChangeListener<{}>);\n if (current.size === 0) {\n listStateListeners.delete(key);\n }\n };\n}\n\nexport class ListState<T extends {}> {\n #items: T[];\n\n /** Pixel offset from the anchored item edge. */\n offset = 0;\n /** Anchor item index, or `undefined` to use the renderer default. */\n position: number | undefined;\n\n /** Items currently managed by the renderer. */\n get items(): T[] {\n return this.#items;\n }\n\n /** Replaces the full item collection while preserving scroll state. */\n set items(value: T[]) {\n this.#items = [...value];\n emitListStateChange(this, { type: \"set\" });\n }\n\n /**\n * @param items Initial list items.\n */\n constructor(items: T[] = []) {\n this.#items = [...items];\n }\n\n /** Prepends one or more items. */\n unshift(...items: T[]): void {\n this.unshiftAll(items);\n }\n\n /** Prepends an array of items. */\n unshiftAll(items: T[]): void {\n if (items.length === 0) {\n return;\n }\n if (this.position != null) {\n this.position += items.length;\n }\n this.#items = items.concat(this.#items);\n emitListStateChange(this, {\n type: \"unshift\",\n count: items.length,\n });\n }\n\n /** Appends one or more items. */\n push(...items: T[]): void {\n this.pushAll(items);\n }\n\n /** Appends an array of items. */\n pushAll(items: T[]): void {\n if (items.length === 0) {\n return;\n }\n this.#items.push(...items);\n emitListStateChange(this, {\n type: \"push\",\n count: items.length,\n });\n }\n\n /**\n * Replaces an existing item by index.\n */\n replace(index: number, item: T, animation?: ReplaceListItemAnimationOptions): void {\n const normalizedIndex = Number.isFinite(index) ? Math.trunc(index) : Number.NaN;\n if (!Number.isInteger(normalizedIndex) || normalizedIndex < 0 || normalizedIndex >= this.#items.length) {\n throw new RangeError(`replace() index ${index} is out of range for list length ${this.#items.length}.`);\n }\n\n const prevItem = this.#items[normalizedIndex]!;\n this.#items[normalizedIndex] = item;\n emitListStateChange(this, {\n type: \"replace\",\n index: normalizedIndex,\n prevItem,\n nextItem: item,\n animation:\n animation != null && Number.isFinite(animation.duration)\n ? { duration: animation.duration }\n : animation == null\n ? undefined\n : {},\n });\n }\n\n /**\n * Sets the current anchor item and pixel offset.\n */\n setAnchor(position: number, offset = 0): void {\n this.position = Number.isFinite(position) ? Math.trunc(position) : undefined;\n this.offset = Number.isFinite(offset) ? offset : 0;\n }\n\n /**\n * Replaces all items and clears scroll state.\n */\n reset(items: T[] = []): void {\n this.#items = [...items];\n this.offset = 0;\n this.position = undefined;\n emitListStateChange(this, { type: \"reset\" });\n }\n\n /** Clears the current scroll anchor while keeping the items. */\n resetScroll(): void {\n this.offset = 0;\n this.position = undefined;\n }\n\n /** Applies a relative pixel scroll delta. */\n applyScroll(delta: number): void {\n this.offset += delta;\n }\n}\n","import type { Node } from \"../types\";\n\nfunction isWeakMapKey(value: unknown): value is object {\n return (typeof value === \"object\" && value !== null) || typeof value === \"function\";\n}\n\n/**\n * Memoizes `renderItem` by object identity.\n */\nexport function memoRenderItem<C extends CanvasRenderingContext2D, T extends object>(\n renderItem: (item: T) => Node<C>,\n): ((item: T) => Node<C>) & { reset: (key: T) => boolean } {\n const cache = new WeakMap<object, Node<C>>();\n\n function fn(item: T): Node<C> {\n if (!isWeakMapKey(item)) {\n throw new TypeError(\"memoRenderItem() only supports object items. Use memoRenderItemBy() for primitive keys.\");\n }\n const key = item as unknown as object;\n const cached = cache.get(key);\n if (cached != null) {\n return cached;\n }\n const result = renderItem(item);\n cache.set(key, result);\n return result;\n }\n\n return Object.assign(fn, {\n reset: (key: T) => cache.delete(key as unknown as object),\n });\n}\n\n/**\n * Memoizes `renderItem` by a caller-provided cache key.\n */\nexport function memoRenderItemBy<C extends CanvasRenderingContext2D, T, K>(\n keyOf: (item: T) => K,\n renderItem: (item: T) => Node<C>,\n): ((item: T) => Node<C>) & { reset: (item: T) => boolean; resetKey: (key: K) => boolean } {\n const cache = new Map<K, Node<C>>();\n\n function fn(item: T): Node<C> {\n const key = keyOf(item);\n const cached = cache.get(key);\n if (cached != null) {\n return cached;\n }\n const result = renderItem(item);\n cache.set(key, result);\n return result;\n }\n\n return Object.assign(fn, {\n reset: (item: T) => cache.delete(keyOf(item)),\n resetKey: (key: K) => cache.delete(key),\n });\n}\n","import type { HitTest, Node, RenderFeedback } from \"../../types\";\nimport { BaseRenderer } from \"../base\";\nimport {\n ListState,\n subscribeListState,\n type ListStateChange,\n} from \"../list-state\";\nimport type { NormalizedListState, VisibleListState, VisibleWindow } from \"./solver\";\n\n/**\n * Options for programmatic scrolling to a target item.\n */\nexport interface JumpToOptions {\n /** Whether to animate the jump. Defaults to `true`. */\n animated?: boolean;\n /** Which edge of the item should align with the viewport. */\n block?: \"start\" | \"center\" | \"end\";\n /** Animation duration in milliseconds. */\n duration?: number;\n /** Called after the jump completes or finishes animating. */\n onComplete?: () => void;\n}\n\ntype ControlledState = {\n position?: number;\n offset: number;\n};\n\ntype JumpAnimation = {\n startAnchor: number;\n targetAnchor: number;\n startTime: number;\n duration: number;\n needsMoreFrames: boolean;\n onComplete: (() => void) | undefined;\n};\n\ntype ReplacementLayer<C extends CanvasRenderingContext2D> = {\n key: number;\n node: Node<C>;\n fromAlpha: number;\n toAlpha: number;\n startTime: number;\n duration: number;\n};\n\ntype ReplacementAnimation<C extends CanvasRenderingContext2D> = {\n currentLayerKey: number;\n layers: ReplacementLayer<C>[];\n fromHeight: number;\n toHeight: number;\n startTime: number;\n duration: number;\n};\n\ntype VirtualizedResolvedItem<C extends CanvasRenderingContext2D> = {\n draw: (y: number) => boolean;\n hittest: (test: HitTest, y: number) => boolean;\n};\n\nconst ALPHA_EPSILON = 1e-3;\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\nfunction sameState(state: ControlledState, position: number | undefined, offset: number): boolean {\n return Object.is(state.position, position) && Object.is(state.offset, offset);\n}\n\nfunction smoothstep(value: number): number {\n return value * value * (3 - 2 * value);\n}\n\nfunction getProgress(startTime: number, duration: number, now: number): number {\n if (!(duration > 0)) {\n return 1;\n }\n return clamp((now - startTime) / duration, 0, 1);\n}\n\nfunction interpolate(from: number, to: number, startTime: number, duration: number, now: number): number {\n const progress = getProgress(startTime, duration, now);\n const eased = progress >= 1 ? 1 : smoothstep(progress);\n return from + (to - from) * eased;\n}\n\nfunction getNow(): number {\n return globalThis.performance?.now() ?? Date.now();\n}\n\n/**\n * Shared base class for virtualized list renderers.\n */\nexport abstract class VirtualizedRenderer<C extends CanvasRenderingContext2D, T extends {}> extends BaseRenderer<\n C,\n {\n renderItem: (item: T) => Node<C>;\n list: ListState<T>;\n }\n> {\n static readonly MIN_JUMP_DURATION = 160;\n static readonly MAX_JUMP_DURATION = 420;\n static readonly JUMP_DURATION_PER_ITEM = 28;\n\n #controlledState: ControlledState | undefined;\n #jumpAnimation: JumpAnimation | undefined;\n #replacementAnimations = new Map<number, ReplacementAnimation<C>>();\n #nextReplacementLayerKey = 0;\n #unsubscribeListState: () => void;\n\n constructor(\n graphics: C,\n options: {\n renderItem: (item: T) => Node<C>;\n list: ListState<T>;\n },\n ) {\n super(graphics, options);\n this.#unsubscribeListState = subscribeListState(options.list, (change) => {\n this.#handleListStateChange(change);\n });\n }\n\n /** Current anchor item index. */\n get position(): number | undefined {\n return this.options.list.position;\n }\n\n /** Updates the current anchor item index. */\n set position(value: number | undefined) {\n this.options.list.position = value;\n }\n\n /** Pixel offset from the anchored item edge. */\n get offset(): number {\n return this.options.list.offset;\n }\n\n /** Updates the pixel offset from the anchored item edge. */\n set offset(value: number) {\n this.options.list.offset = value;\n }\n\n /** Items currently available to the renderer. */\n get items(): T[] {\n return this.options.list.items;\n }\n\n /** Replaces the current item collection. */\n set items(value: T[]) {\n this.options.list.items = value;\n }\n\n /** Renders the current visible window. */\n abstract render(feedback?: RenderFeedback): boolean;\n /** Hit-tests the current visible window. */\n abstract hittest(test: { x: number; y: number; type: \"click\" | \"auxclick\" | \"hover\" }): boolean;\n\n protected _readListState(): VisibleListState {\n return {\n position: this.position,\n offset: this.offset,\n };\n }\n\n protected _commitListState(state: NormalizedListState): void {\n this.position = state.position;\n this.offset = state.offset;\n }\n\n /**\n * Scrolls the viewport to the requested item index.\n */\n jumpTo(index: number, options: JumpToOptions = {}): void {\n if (this.items.length === 0) {\n this.#cancelJumpAnimation();\n return;\n }\n\n const targetIndex = this._clampItemIndex(index);\n const currentState = this._normalizeListState(this._readListState());\n const targetBlock = options.block ?? this._getDefaultJumpBlock();\n const targetAnchor = this._getTargetAnchor(targetIndex, targetBlock);\n\n const animated = options.animated ?? true;\n if (!animated) {\n this.#cancelJumpAnimation();\n this._applyAnchor(targetAnchor);\n options.onComplete?.();\n return;\n }\n\n const startAnchor = this._readAnchor(currentState);\n if (!Number.isFinite(startAnchor)) {\n this.#cancelJumpAnimation();\n this._applyAnchor(targetAnchor);\n options.onComplete?.();\n return;\n }\n\n const duration = clamp(\n options.duration ??\n VirtualizedRenderer.MIN_JUMP_DURATION +\n Math.abs(targetAnchor - startAnchor) * VirtualizedRenderer.JUMP_DURATION_PER_ITEM,\n 0,\n VirtualizedRenderer.MAX_JUMP_DURATION,\n );\n\n if (duration <= 0 || Math.abs(targetAnchor - startAnchor) <= Number.EPSILON) {\n this.#cancelJumpAnimation();\n this._applyAnchor(targetAnchor);\n options.onComplete?.();\n return;\n }\n\n this.#jumpAnimation = {\n startAnchor,\n targetAnchor,\n startTime: getNow(),\n duration,\n needsMoreFrames: true,\n onComplete: options.onComplete,\n };\n this.#controlledState = this._readListState();\n }\n\n protected _resetRenderFeedback(feedback?: RenderFeedback): void {\n if (feedback == null) {\n return;\n }\n feedback.minIdx = Number.NaN;\n feedback.maxIdx = Number.NaN;\n feedback.min = Number.NaN;\n feedback.max = Number.NaN;\n }\n\n protected _accumulateRenderFeedback(feedback: RenderFeedback, idx: number, top: number, height: number): void {\n if (!Number.isFinite(top) || !Number.isFinite(height) || height <= 0) {\n return;\n }\n\n const viewportHeight = this.graphics.canvas.clientHeight;\n const visibleTop = clamp(-top, 0, height);\n const visibleBottom = clamp(viewportHeight - top, 0, height);\n if (visibleBottom <= visibleTop) {\n return;\n }\n\n const itemMin = idx + visibleTop / height;\n const itemMax = idx + visibleBottom / height;\n feedback.minIdx = Number.isNaN(feedback.minIdx) ? idx : Math.min(idx, feedback.minIdx);\n feedback.maxIdx = Number.isNaN(feedback.maxIdx) ? idx : Math.max(idx, feedback.maxIdx);\n feedback.min = Number.isNaN(feedback.min) ? itemMin : Math.min(itemMin, feedback.min);\n feedback.max = Number.isNaN(feedback.max) ? itemMax : Math.max(itemMax, feedback.max);\n }\n\n protected _renderDrawList(\n list: VisibleWindow<VirtualizedResolvedItem<C>>[\"drawList\"],\n shift: number,\n feedback?: RenderFeedback,\n ): boolean {\n let result = false;\n const viewportHeight = this.graphics.canvas.clientHeight;\n\n for (const { idx, value: item, offset, height } of list) {\n const y = offset + shift;\n if (feedback != null) {\n this._accumulateRenderFeedback(feedback, idx, y, height);\n }\n if (y + height < 0 || y > viewportHeight) {\n continue;\n }\n if (item.draw(y)) {\n result = true;\n }\n }\n\n return result;\n }\n\n protected _renderVisibleWindow(window: VisibleWindow<VirtualizedResolvedItem<C>>, feedback?: RenderFeedback): boolean {\n this._resetRenderFeedback(feedback);\n return this._renderDrawList(window.drawList, window.shift, feedback);\n }\n\n protected _hittestVisibleWindow(window: VisibleWindow<VirtualizedResolvedItem<C>>, test: HitTest): boolean {\n for (const { value: item, offset, height } of window.drawList) {\n const y = offset + window.shift;\n if (test.y < y || test.y >= y + height) {\n continue;\n }\n return item.hittest(test, y);\n }\n return false;\n }\n\n protected _prepareRender(): boolean {\n const now = getNow();\n const keepReplacing = this.#prepareReplacementAnimations(now);\n const animation = this.#jumpAnimation;\n if (animation == null) {\n return keepReplacing;\n }\n if (this.items.length === 0) {\n this.#cancelJumpAnimation();\n return keepReplacing;\n }\n if (this.#controlledState != null && !sameState(this.#controlledState, this.position, this.offset)) {\n this.#cancelJumpAnimation();\n return keepReplacing;\n }\n\n const anchor = interpolate(animation.startAnchor, animation.targetAnchor, animation.startTime, animation.duration, now);\n const progress = getProgress(animation.startTime, animation.duration, now);\n this._applyAnchor(anchor);\n animation.needsMoreFrames = progress < 1;\n return keepReplacing || animation.needsMoreFrames;\n }\n\n protected _finishRender(requestRedraw: boolean): boolean {\n const animation = this.#jumpAnimation;\n if (animation == null) {\n return requestRedraw;\n }\n\n if (animation.needsMoreFrames) {\n this.#controlledState = this._readListState();\n return true;\n }\n\n const onComplete = animation.onComplete;\n this.#cancelJumpAnimation();\n onComplete?.();\n return requestRedraw || this.#jumpAnimation != null;\n }\n\n protected _clampItemIndex(index: number): number {\n return clamp(Number.isFinite(index) ? Math.trunc(index) : 0, 0, this.items.length - 1);\n }\n\n protected _getItemHeight(index: number): number {\n const now = getNow();\n const replacement = this.#readReplacementAnimation(index, now);\n if (replacement != null) {\n return this.#sampleReplacementHeight(replacement, now);\n }\n const item = this.items[index];\n const node = this.options.renderItem(item);\n return this.measureRootNode(node).height;\n }\n\n protected _resolveItem(item: T, index: number, now: number): { value: VirtualizedResolvedItem<C>; height: number } {\n const replacement = this.#readReplacementAnimation(index, now);\n if (replacement == null) {\n const node = this.options.renderItem(item);\n return {\n value: {\n draw: (y) => this.drawRootNode(node, 0, y),\n hittest: (test, y) =>\n node.hittest(this.getRootContext(), {\n ...test,\n y: test.y - y,\n }),\n },\n height: this.measureRootNode(node).height,\n };\n }\n\n const slotHeight = this.#sampleReplacementHeight(replacement, now);\n const layers = replacement.layers\n .map((layer) => ({\n alpha: this.#sampleLayerAlpha(layer, now),\n node: layer.node,\n nodeHeight: this.measureRootNode(layer.node).height,\n }))\n .filter((layer) => layer.alpha > ALPHA_EPSILON);\n\n return {\n value: {\n draw: (y) => this.#drawReplacementLayers(layers, slotHeight, y),\n hittest: () => false,\n },\n height: slotHeight,\n };\n }\n\n protected _getAnchorAtOffset(index: number, offset: number): number {\n if (this.items.length === 0) {\n return 0;\n }\n\n let currentIndex = this._clampItemIndex(index);\n let remaining = Number.isFinite(offset) ? offset : 0;\n while (true) {\n if (remaining < 0) {\n if (currentIndex === 0) {\n return 0;\n }\n currentIndex -= 1;\n const height = this._getItemHeight(currentIndex);\n if (height > 0) {\n remaining += height;\n }\n continue;\n }\n\n const height = this._getItemHeight(currentIndex);\n if (height > 0) {\n if (remaining <= height) {\n return currentIndex + remaining / height;\n }\n remaining -= height;\n } else if (remaining === 0) {\n return currentIndex;\n }\n\n if (currentIndex === this.items.length - 1) {\n return this.items.length;\n }\n currentIndex += 1;\n }\n }\n\n protected abstract _normalizeListState(state: VisibleListState): NormalizedListState;\n protected abstract _readAnchor(state: NormalizedListState): number;\n protected abstract _applyAnchor(anchor: number): void;\n protected abstract _getDefaultJumpBlock(): NonNullable<JumpToOptions[\"block\"]>;\n protected abstract _getTargetAnchor(index: number, block: NonNullable<JumpToOptions[\"block\"]>): number;\n protected abstract _getAnimatedLayerOffset(slotHeight: number, nodeHeight: number): number;\n\n #cancelJumpAnimation(): void {\n this.#jumpAnimation = undefined;\n this.#controlledState = undefined;\n }\n\n #createReplacementLayer(node: Node<C>, fromAlpha: number, toAlpha: number, startTime: number, duration: number): ReplacementLayer<C> {\n return {\n key: ++this.#nextReplacementLayerKey,\n node,\n fromAlpha,\n toAlpha,\n startTime,\n duration,\n };\n }\n\n #sampleLayerAlpha(layer: ReplacementLayer<C>, now: number): number {\n return interpolate(layer.fromAlpha, layer.toAlpha, layer.startTime, layer.duration, now);\n }\n\n #sampleReplacementHeight(animation: ReplacementAnimation<C>, now: number): number {\n return interpolate(animation.fromHeight, animation.toHeight, animation.startTime, animation.duration, now);\n }\n\n #isLayerComplete(layer: ReplacementLayer<C>, now: number): boolean {\n return getProgress(layer.startTime, layer.duration, now) >= 1 && Math.abs(layer.toAlpha - this.#sampleLayerAlpha(layer, now)) <= ALPHA_EPSILON;\n }\n\n #readReplacementAnimation(index: number, now: number): ReplacementAnimation<C> | undefined {\n const animation = this.#replacementAnimations.get(index);\n if (animation == null) {\n return undefined;\n }\n\n const currentLayer = animation.layers.find((layer) => layer.key === animation.currentLayerKey);\n if (currentLayer == null) {\n this.#replacementAnimations.delete(index);\n return undefined;\n }\n\n animation.layers = animation.layers.filter((layer) => layer.key === animation.currentLayerKey || !this.#isLayerComplete(layer, now));\n if (\n getProgress(animation.startTime, animation.duration, now) >= 1 &&\n this.#isLayerComplete(currentLayer, now) &&\n animation.layers.length === 1\n ) {\n this.#replacementAnimations.delete(index);\n return undefined;\n }\n\n return animation;\n }\n\n #prepareReplacementAnimations(now: number): boolean {\n let keepAnimating = false;\n for (const index of [...this.#replacementAnimations.keys()]) {\n if (this.#readReplacementAnimation(index, now) != null) {\n keepAnimating = true;\n }\n }\n return keepAnimating;\n }\n\n #drawReplacementLayers(\n layers: { alpha: number; node: Node<C>; nodeHeight: number }[],\n slotHeight: number,\n y: number,\n ): boolean {\n if (slotHeight <= 0) {\n return false;\n }\n\n let result = false;\n const width = this.graphics.canvas.clientWidth;\n for (const layer of layers) {\n const alpha = clamp(layer.alpha, 0, 1);\n if (alpha <= ALPHA_EPSILON) {\n continue;\n }\n\n this.graphics.save();\n try {\n this.graphics.beginPath?.();\n this.graphics.rect?.(0, y, width, slotHeight);\n this.graphics.clip?.();\n if (typeof this.graphics.globalAlpha === \"number\") {\n this.graphics.globalAlpha *= alpha;\n }\n const layerY = y + this._getAnimatedLayerOffset(slotHeight, layer.nodeHeight);\n if (this.drawRootNode(layer.node, 0, layerY)) {\n result = true;\n }\n } finally {\n this.graphics.restore();\n }\n }\n return result;\n }\n\n #handleListStateChange(change: ListStateChange<T>): void {\n switch (change.type) {\n case \"replace\":\n this.#handleReplace(change.index, change.prevItem, change.nextItem, change.animation?.duration);\n break;\n case \"unshift\": {\n if (change.count <= 0 || this.#replacementAnimations.size === 0) {\n return;\n }\n const shifted = new Map<number, ReplacementAnimation<C>>();\n for (const [index, animation] of this.#replacementAnimations) {\n shifted.set(index + change.count, animation);\n }\n this.#replacementAnimations = shifted;\n break;\n }\n case \"push\":\n break;\n case \"reset\":\n case \"set\":\n this.#replacementAnimations.clear();\n break;\n }\n }\n\n #handleReplace(index: number, prevItem: T, nextItem: T, duration: number | undefined): void {\n const normalizedDuration = Number.isFinite(duration) ? Math.max(0, duration!) : 0;\n if (normalizedDuration <= 0) {\n this.#replacementAnimations.delete(index);\n return;\n }\n\n const now = getNow();\n const nextNode = this.options.renderItem(nextItem);\n const nextHeight = this.measureRootNode(nextNode).height;\n const animation = this.#readReplacementAnimation(index, now);\n if (animation == null) {\n const prevNode = this.options.renderItem(prevItem);\n const outgoing = this.#createReplacementLayer(prevNode, 1, 0, now, normalizedDuration);\n const incoming = this.#createReplacementLayer(nextNode, 0, 1, now, normalizedDuration);\n this.#replacementAnimations.set(index, {\n currentLayerKey: incoming.key,\n layers: [outgoing, incoming],\n fromHeight: this.measureRootNode(prevNode).height,\n toHeight: nextHeight,\n startTime: now,\n duration: normalizedDuration,\n });\n return;\n }\n\n const currentLayer = animation.layers.find((layer) => layer.key === animation.currentLayerKey);\n const currentNode = currentLayer?.node ?? this.options.renderItem(prevItem);\n const currentAlpha = currentLayer == null ? 1 : this.#sampleLayerAlpha(currentLayer, now);\n const layers = animation.layers.filter(\n (layer) => layer.key !== animation.currentLayerKey && !this.#isLayerComplete(layer, now),\n );\n if (currentAlpha > ALPHA_EPSILON) {\n layers.push(this.#createReplacementLayer(currentNode, currentAlpha, 0, now, normalizedDuration));\n }\n const incoming = this.#createReplacementLayer(nextNode, 0, 1, now, normalizedDuration);\n layers.push(incoming);\n this.#replacementAnimations.set(index, {\n currentLayerKey: incoming.key,\n layers,\n fromHeight: this.#sampleReplacementHeight(animation, now),\n toHeight: nextHeight,\n startTime: now,\n duration: normalizedDuration,\n });\n }\n}\n","export interface VisibleListState {\n position?: number;\n offset: number;\n}\n\nexport interface NormalizedListState {\n position: number;\n offset: number;\n}\n\nexport interface VisibleWindowEntry<T> {\n idx: number;\n value: T;\n offset: number;\n height: number;\n}\n\nexport interface VisibleWindow<T> {\n drawList: VisibleWindowEntry<T>[];\n shift: number;\n}\n\nexport interface VisibleWindowResult<T> {\n normalizedState: NormalizedListState;\n window: VisibleWindow<T>;\n}\n\ntype ResolvedItem<T> = {\n value: T;\n height: number;\n};\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\nfunction normalizeOffset(offset: number): number {\n return Number.isFinite(offset) ? offset : 0;\n}\n\nexport function normalizeTimelineState(itemCount: number, state: VisibleListState): NormalizedListState {\n if (itemCount <= 0) {\n return { position: 0, offset: 0 };\n }\n\n const position = state.position;\n if (typeof position !== \"number\" || !Number.isFinite(position)) {\n return {\n position: 0,\n offset: normalizeOffset(state.offset),\n };\n }\n\n return {\n position: clamp(Math.trunc(position), 0, itemCount - 1),\n offset: normalizeOffset(state.offset),\n };\n}\n\nexport function normalizeChatState(itemCount: number, state: VisibleListState): NormalizedListState {\n if (itemCount <= 0) {\n return { position: 0, offset: 0 };\n }\n\n const position = state.position;\n if (typeof position !== \"number\" || !Number.isFinite(position)) {\n return {\n position: itemCount - 1,\n offset: normalizeOffset(state.offset),\n };\n }\n\n return {\n position: clamp(Math.trunc(position), 0, itemCount - 1),\n offset: normalizeOffset(state.offset),\n };\n}\n\nexport function resolveTimelineVisibleWindow<T, V>(\n items: readonly T[],\n state: VisibleListState,\n viewportHeight: number,\n resolveItem: (item: T, idx: number) => ResolvedItem<V>,\n): VisibleWindowResult<V> {\n const normalizedState = normalizeTimelineState(items.length, state);\n if (items.length === 0) {\n return {\n normalizedState,\n window: { drawList: [], shift: 0 },\n };\n }\n\n let { position, offset } = normalizedState;\n let drawLength = 0;\n\n if (offset > 0) {\n if (position === 0) {\n offset = 0;\n } else {\n for (let i = position - 1; i >= 0; i -= 1) {\n const { height } = resolveItem(items[i]!, i);\n position = i;\n offset -= height;\n if (offset <= 0) {\n break;\n }\n }\n if (position === 0 && offset > 0) {\n offset = 0;\n }\n }\n }\n\n let y = offset;\n const drawList: VisibleWindowEntry<V>[] = [];\n for (let i = position; i < items.length; i += 1) {\n const { value, height } = resolveItem(items[i]!, i);\n if (y + height > 0) {\n drawList.push({ idx: i, value, offset: y, height });\n drawLength += height;\n } else {\n offset += height;\n position = i + 1;\n }\n y += height;\n if (y >= viewportHeight) {\n break;\n }\n }\n\n let shift = 0;\n if (y < viewportHeight) {\n if (position === 0 && drawLength < viewportHeight) {\n shift = -offset;\n offset = 0;\n } else {\n shift = viewportHeight - y;\n y = (offset += shift);\n let lastIdx = -1;\n for (let i = position - 1; i >= 0; i -= 1) {\n const { value, height } = resolveItem(items[i]!, i);\n drawLength += height;\n y -= height;\n drawList.push({ idx: i, value, offset: y - shift, height });\n lastIdx = i;\n if (y < 0) {\n break;\n }\n }\n if (lastIdx === 0 && drawLength < viewportHeight) {\n shift = drawList.at(-1)?.offset == null ? 0 : -drawList.at(-1)!.offset;\n position = 0;\n offset = 0;\n }\n }\n }\n\n return {\n normalizedState: { position, offset },\n window: { drawList, shift },\n };\n}\n\nexport function resolveChatVisibleWindow<T, V>(\n items: readonly T[],\n state: VisibleListState,\n viewportHeight: number,\n resolveItem: (item: T, idx: number) => ResolvedItem<V>,\n): VisibleWindowResult<V> {\n const normalizedState = normalizeChatState(items.length, state);\n if (items.length === 0) {\n return {\n normalizedState,\n window: { drawList: [], shift: 0 },\n };\n }\n\n let { position, offset } = normalizedState;\n let drawLength = 0;\n\n if (offset < 0) {\n if (position === items.length - 1) {\n offset = 0;\n } else {\n for (let i = position + 1; i < items.length; i += 1) {\n const { height } = resolveItem(items[i]!, i);\n position = i;\n offset += height;\n if (offset > 0) {\n break;\n }\n }\n }\n }\n\n let y = viewportHeight + offset;\n const drawList: VisibleWindowEntry<V>[] = [];\n for (let i = position; i >= 0; i -= 1) {\n const { value, height } = resolveItem(items[i]!, i);\n y -= height;\n if (y <= viewportHeight) {\n drawList.push({ idx: i, value, offset: y, height });\n drawLength += height;\n } else {\n offset -= height;\n position = i - 1;\n }\n if (y < 0) {\n break;\n }\n }\n\n let shift = 0;\n if (y > 0) {\n shift = -y;\n if (drawLength < viewportHeight) {\n y = drawLength;\n for (let i = position + 1; i < items.length; i += 1) {\n const { value, height } = resolveItem(items[i]!, i);\n drawList.push({ idx: i, value, offset: y - shift, height });\n y = (drawLength += height);\n position = i;\n if (y >= viewportHeight) {\n break;\n }\n }\n offset = drawLength < viewportHeight ? 0 : drawLength - viewportHeight;\n } else {\n offset = drawLength - viewportHeight;\n }\n }\n\n return {\n normalizedState: { position, offset },\n window: { drawList, shift },\n };\n}\n","import type { HitTest, RenderFeedback } from \"../../types\";\nimport { VirtualizedRenderer } from \"./base\";\nimport type { JumpToOptions } from \"./base\";\nimport {\n normalizeChatState,\n resolveChatVisibleWindow,\n type NormalizedListState,\n type VisibleListState,\n} from \"./solver\";\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\n/**\n * Virtualized renderer anchored to the bottom, suitable for chat-style UIs.\n */\nexport class ChatRenderer<C extends CanvasRenderingContext2D, T extends {}> extends VirtualizedRenderer<C, T> {\n #resolveVisibleWindow() {\n const now = globalThis.performance?.now() ?? Date.now();\n return resolveChatVisibleWindow(\n this.items,\n this._readListState(),\n this.graphics.canvas.clientHeight,\n (item, idx) => {\n return this._resolveItem(item, idx, now);\n },\n );\n }\n\n protected _getDefaultJumpBlock(): NonNullable<JumpToOptions[\"block\"]> {\n return \"end\";\n }\n\n protected _normalizeListState(state: VisibleListState): NormalizedListState {\n return normalizeChatState(this.items.length, state);\n }\n\n protected _readAnchor(state: NormalizedListState): number {\n if (this.items.length === 0) {\n return 0;\n }\n const height = this._getItemHeight(state.position);\n return height > 0 ? state.position + 1 - state.offset / height : state.position + 1;\n }\n\n protected _applyAnchor(anchor: number): void {\n if (this.items.length === 0) {\n return;\n }\n const clampedAnchor = clamp(anchor, 0, this.items.length);\n const position = clamp(Math.ceil(clampedAnchor) - 1, 0, this.items.length - 1);\n const height = this._getItemHeight(position);\n const offset = height > 0 ? (position + 1 - clampedAnchor) * height : 0;\n this._commitListState({\n position,\n offset: Object.is(offset, -0) ? 0 : offset,\n });\n }\n\n protected _getTargetAnchor(index: number, block: NonNullable<JumpToOptions[\"block\"]>): number {\n const height = this._getItemHeight(index);\n const viewportHeight = this.graphics.canvas.clientHeight;\n\n switch (block) {\n case \"start\":\n return this._getAnchorAtOffset(index, viewportHeight);\n case \"center\":\n return this._getAnchorAtOffset(index, height / 2 + viewportHeight / 2);\n case \"end\":\n return this._getAnchorAtOffset(index, height);\n }\n }\n\n protected _getAnimatedLayerOffset(slotHeight: number, nodeHeight: number): number {\n return slotHeight - nodeHeight;\n }\n\n render(feedback?: RenderFeedback): boolean {\n const keepAnimating = this._prepareRender();\n const { clientWidth: viewportWidth, clientHeight: viewportHeight } = this.graphics.canvas;\n this.graphics.clearRect(0, 0, viewportWidth, viewportHeight);\n const solution = this.#resolveVisibleWindow();\n const requestRedraw = this._renderVisibleWindow(solution.window, feedback);\n this._commitListState(solution.normalizedState);\n return this._finishRender(keepAnimating || requestRedraw);\n }\n\n hittest(test: HitTest): boolean {\n return this._hittestVisibleWindow(this.#resolveVisibleWindow().window, test);\n }\n}\n","import type { HitTest, RenderFeedback } from \"../../types\";\nimport { VirtualizedRenderer } from \"./base\";\nimport type { JumpToOptions } from \"./base\";\nimport {\n normalizeTimelineState,\n resolveTimelineVisibleWindow,\n type NormalizedListState,\n type VisibleListState,\n} from \"./solver\";\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\n/**\n * Virtualized renderer anchored to the top, suitable for timeline-style UIs.\n */\nexport class TimelineRenderer<C extends CanvasRenderingContext2D, T extends {}> extends VirtualizedRenderer<C, T> {\n #resolveVisibleWindow() {\n const now = globalThis.performance?.now() ?? Date.now();\n return resolveTimelineVisibleWindow(\n this.items,\n this._readListState(),\n this.graphics.canvas.clientHeight,\n (item, idx) => {\n return this._resolveItem(item, idx, now);\n },\n );\n }\n\n protected _getDefaultJumpBlock(): NonNullable<JumpToOptions[\"block\"]> {\n return \"start\";\n }\n\n protected _normalizeListState(state: VisibleListState): NormalizedListState {\n return normalizeTimelineState(this.items.length, state);\n }\n\n protected _readAnchor(state: NormalizedListState): number {\n if (this.items.length === 0) {\n return 0;\n }\n const height = this._getItemHeight(state.position);\n return height > 0 ? state.position - state.offset / height : state.position;\n }\n\n protected _applyAnchor(anchor: number): void {\n if (this.items.length === 0) {\n return;\n }\n const clampedAnchor = clamp(anchor, 0, this.items.length);\n const position = clamp(Math.floor(clampedAnchor), 0, this.items.length - 1);\n const height = this._getItemHeight(position);\n const offset = height > 0 ? -(clampedAnchor - position) * height : 0;\n this._commitListState({\n position,\n offset: Object.is(offset, -0) ? 0 : offset,\n });\n }\n\n protected _getTargetAnchor(index: number, block: NonNullable<JumpToOptions[\"block\"]>): number {\n const height = this._getItemHeight(index);\n const viewportHeight = this.graphics.canvas.clientHeight;\n\n switch (block) {\n case \"start\":\n return this._getAnchorAtOffset(index, 0);\n case \"center\":\n return this._getAnchorAtOffset(index, height / 2 - viewportHeight / 2);\n case \"end\":\n return this._getAnchorAtOffset(index, height - viewportHeight);\n }\n }\n\n protected _getAnimatedLayerOffset(_slotHeight: number, _nodeHeight: number): number {\n return 0;\n }\n\n render(feedback?: RenderFeedback): boolean {\n const keepAnimating = this._prepareRender();\n const { clientWidth: viewportWidth, clientHeight: viewportHeight } = this.graphics.canvas;\n this.graphics.clearRect(0, 0, viewportWidth, viewportHeight);\n const solution = this.#resolveVisibleWindow();\n const requestRedraw = this._renderVisibleWindow(solution.window, feedback);\n this._commitListState(solution.normalizedState);\n return this._finishRender(keepAnimating || requestRedraw);\n }\n\n hittest(test: HitTest): boolean {\n return this._hittestVisibleWindow(this.#resolveVisibleWindow().window, test);\n }\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.mjs","names":["#children","#inner","clampToConstraints","#top","#bottom","#left","#right","LINE_START_CURSOR","#ctx","#lastWidth","#cache","#layoutCache","#textLayoutCache","#clearAllCaches","#syncCachesToViewportWidth","#items","clamp","#unsubscribeListState","#handleListStateChange","#cancelJumpAnimation","#jumpAnimation","#controlledState","#prepareReplacementAnimations","#readReplacementAnimation","#sampleReplacementHeight","#sampleLayerAlpha","#drawReplacementLayers","#nextReplacementLayerKey","#replacementAnimations","#isLayerComplete","#handleReplace","#createReplacementLayer","clamp","clamp","#resolveVisibleWindow","#resolveVisibleWindow"],"sources":["../src/internal/node-registry.ts","../src/utils.ts","../src/nodes/base.ts","../src/layout.ts","../src/nodes/shared.ts","../src/nodes/box.ts","../src/nodes/flex.ts","../src/nodes/place.ts","../src/text/core.ts","../src/text/plain-core.ts","../src/text/plain.ts","../src/text/rich.ts","../src/nodes/text.ts","../src/renderer/base.ts","../src/renderer/list-state.ts","../src/renderer/memo.ts","../src/renderer/virtualized/base.ts","../src/renderer/virtualized/solver.ts","../src/renderer/virtualized/chat.ts","../src/renderer/virtualized/timeline.ts"],"sourcesContent":["import type { Node } from \"../types\";\n\nconst registry = new WeakMap<Node<any>, Node<any>>();\nconst revisions = new WeakMap<Node<any>, number>();\n\nfunction getOwnershipError(): Error {\n return new Error(\"A node can only be attached to one parent. Shared nodes are not supported.\");\n}\n\nfunction getDetachOwnershipError(): Error {\n return new Error(\"Cannot detach or replace a node from a parent that does not own it.\");\n}\n\nfunction bumpRevision(node: Node<any>): void {\n revisions.set(node, (revisions.get(node) ?? 0) + 1);\n}\n\nexport function getNodeRevision<C extends CanvasRenderingContext2D>(node: Node<C>): number {\n return revisions.get(node) ?? 0;\n}\n\nexport function attachNodeToParent<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n parent: Node<C>,\n): void {\n if (registry.has(node)) {\n throw getOwnershipError();\n }\n registry.set(node, parent);\n bumpRevision(parent);\n}\n\nexport function attachNodesToParent<C extends CanvasRenderingContext2D>(\n nodes: Iterable<Node<C>>,\n parent: Node<C>,\n): void {\n for (const node of nodes) {\n attachNodeToParent(node, parent);\n }\n}\n\nexport function detachNodeFromParent<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n parent?: Node<C>,\n): void {\n const currentParent = registry.get(node);\n if (currentParent == null) {\n return;\n }\n if (parent != null && currentParent !== parent) {\n throw getDetachOwnershipError();\n }\n registry.delete(node);\n bumpRevision(currentParent);\n}\n\nexport function replaceNodeParent<C extends CanvasRenderingContext2D>(\n previousNode: Node<C>,\n nextNode: Node<C>,\n parent: Node<C>,\n): void {\n if (previousNode === nextNode) {\n return;\n }\n const currentParent = registry.get(previousNode);\n if (currentParent !== parent) {\n throw getDetachOwnershipError();\n }\n if (registry.has(nextNode)) {\n throw getOwnershipError();\n }\n registry.delete(previousNode);\n registry.set(nextNode, parent);\n bumpRevision(parent);\n}\n\nexport function replaceNodesParent<C extends CanvasRenderingContext2D>(\n previousNodes: Iterable<Node<C>>,\n nextNodes: Iterable<Node<C>>,\n parent: Node<C>,\n): void {\n const previousSnapshot = Array.from(previousNodes);\n const nextSnapshot = Array.from(nextNodes);\n if (\n previousSnapshot.length === nextSnapshot.length &&\n previousSnapshot.every((node, index) => node === nextSnapshot[index])\n ) {\n return;\n }\n\n const previousSet = new Set(previousSnapshot);\n const nextSet = new Set<Node<C>>();\n for (const node of nextSnapshot) {\n if (nextSet.has(node)) {\n throw getOwnershipError();\n }\n nextSet.add(node);\n const currentParent = registry.get(node);\n if (currentParent != null && currentParent !== parent) {\n throw getOwnershipError();\n }\n }\n\n for (const node of previousSnapshot) {\n if (!nextSet.has(node)) {\n const currentParent = registry.get(node);\n if (currentParent !== parent) {\n throw getDetachOwnershipError();\n }\n registry.delete(node);\n }\n }\n\n for (const node of nextSnapshot) {\n if (!previousSet.has(node)) {\n registry.set(node, parent);\n }\n }\n\n bumpRevision(parent);\n}\n\nexport function forEachNodeAncestor<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n visitor: (ancestor: Node<C>) => void,\n): void {\n let current: Node<C> | undefined = node;\n while ((current = registry.get(current))) {\n visitor(current);\n }\n}\n\nexport function registerNodeParent<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n parent: Node<C>,\n): void {\n attachNodeToParent(node, parent);\n}\n\nexport function unregisterNodeParent<C extends CanvasRenderingContext2D>(node: Node<C>, parent?: Node<C>): void {\n detachNodeFromParent(node, parent);\n}\n\nexport function getNodeParent<C extends CanvasRenderingContext2D>(node: Node<C>): Node<C> | undefined {\n return registry.get(node);\n}\n","export function shallow<T extends object>(object: T): T {\n return Object.create(object) as T;\n}\n","import { attachNodeToParent, replaceNodeParent, replaceNodesParent } from \"../internal/node-registry\";\nimport type { Box, Context, HitTest, LayoutConstraints, Node } from \"../types\";\nimport { shallow } from \"../utils\";\n\nexport function withNodeConstraints<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n constraints: LayoutConstraints | undefined,\n): Context<C> {\n if (constraints === ctx.constraints) {\n return ctx;\n }\n const next = shallow(ctx);\n next.constraints = constraints;\n return next;\n}\n\nexport function measureNodeMinContent<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n node: Node<C>,\n constraints: LayoutConstraints | undefined = ctx.constraints,\n): Box {\n const nextCtx = withNodeConstraints(ctx, constraints);\n if (node.measureMinContent != null) {\n return node.measureMinContent(nextCtx);\n }\n return node.measure(nextCtx);\n}\n\n/**\n * A node that owns an ordered list of child nodes.\n */\nexport abstract class Group<C extends CanvasRenderingContext2D> implements Node<C> {\n #children: Node<C>[];\n\n /**\n * @param children Initial child nodes, in layout order.\n */\n constructor(children: Node<C>[]) {\n this.#children = [...children];\n replaceNodesParent([], this.#children, this);\n }\n\n /** Child nodes managed by this group. */\n get children(): readonly Node<C>[] {\n return this.#children;\n }\n\n /**\n * Replaces the full child list while updating parent links.\n */\n replaceChildren(nextChildren: Node<C>[]): void {\n const nextSnapshot = [...nextChildren];\n replaceNodesParent(this.#children, nextSnapshot, this);\n this.#children = nextSnapshot;\n }\n\n abstract measure(ctx: Context<C>): Box;\n abstract draw(ctx: Context<C>, x: number, y: number): boolean;\n abstract hittest(ctx: Context<C>, test: HitTest): boolean;\n}\n\n/**\n * A node that forwards layout and drawing to a single inner node.\n */\nexport class Wrapper<C extends CanvasRenderingContext2D> implements Node<C> {\n #inner: Node<C>;\n\n /**\n * @param inner Wrapped child node.\n */\n constructor(inner: Node<C>) {\n this.#inner = inner;\n attachNodeToParent(this.#inner, this);\n }\n\n /** The wrapped child node. */\n get inner(): Node<C> {\n return this.#inner;\n }\n\n /** Replaces the wrapped child node. */\n set inner(newNode: Node<C>) {\n if (newNode === this.#inner) {\n return;\n }\n replaceNodeParent(this.#inner, newNode, this);\n this.#inner = newNode;\n }\n\n measure(ctx: Context<C>): Box {\n return this.inner.measure(ctx);\n }\n\n measureMinContent(ctx: Context<C>): Box {\n return measureNodeMinContent(ctx, this.inner);\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n return this.inner.draw(ctx, x, y);\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n return this.inner.hittest(ctx, test);\n }\n}\n","import type { Box, ChildLayoutResult, FlexLayoutResult, LayoutConstraints, LayoutRect } from \"./types\";\n\n/**\n * 创建 LayoutRect 的辅助函数\n */\nexport function createRect(x: number, y: number, width: number, height: number): LayoutRect {\n return { x, y, width, height };\n}\n\n/**\n * 合并多个 rect 得到包含所有 rect 的最小外接矩形\n */\nexport function mergeRects(rects: LayoutRect[]): LayoutRect {\n if (rects.length === 0) {\n return createRect(0, 0, 0, 0);\n }\n\n let minX = Infinity;\n let minY = Infinity;\n let maxX = -Infinity;\n let maxY = -Infinity;\n\n for (const rect of rects) {\n minX = Math.min(minX, rect.x);\n minY = Math.min(minY, rect.y);\n maxX = Math.max(maxX, rect.x + rect.width);\n maxY = Math.max(maxY, rect.y + rect.height);\n }\n\n return createRect(minX, minY, maxX - minX, maxY - minY);\n}\n\n/**\n * 从子节点布局结果计算容器的 contentBox\n */\nexport function computeContentBox<C extends CanvasRenderingContext2D>(\n children: ChildLayoutResult<C>[],\n): LayoutRect {\n return mergeRects(children.map((child) => child.contentBox));\n}\n\n/**\n * 根据约束和实际内容计算最终的 containerBox\n */\nexport function computeContainerBox(\n contentBox: LayoutRect,\n constraints?: LayoutConstraints,\n): LayoutRect {\n let width = contentBox.width;\n let height = contentBox.height;\n\n if (constraints?.minWidth != null) {\n width = Math.max(width, constraints.minWidth);\n }\n if (constraints?.maxWidth != null) {\n width = Math.min(width, constraints.maxWidth);\n }\n if (constraints?.minHeight != null) {\n height = Math.max(height, constraints.minHeight);\n }\n if (constraints?.maxHeight != null) {\n height = Math.min(height, constraints.maxHeight);\n }\n\n return createRect(contentBox.x, contentBox.y, width, height);\n}\n\n/**\n * 将 Box 转换为 LayoutRect(位置为 0,0)\n */\nexport function boxToRect(box: Box): LayoutRect {\n return createRect(0, 0, box.width, box.height);\n}\n\n/**\n * 检查点是否在 rect 内\n */\nexport function pointInRect(x: number, y: number, rect: LayoutRect): boolean {\n return x >= rect.x && x < rect.x + rect.width && y >= rect.y && y < rect.y + rect.height;\n}\n\n/**\n * 平移 rect 的位置\n */\nexport function offsetRect(rect: LayoutRect, dx: number, dy: number): LayoutRect {\n return createRect(rect.x + dx, rect.y + dy, rect.width, rect.height);\n}\n\n/**\n * 读取单子节点布局结果中的唯一 child。\n */\nexport function getSingleChildLayout<C extends CanvasRenderingContext2D>(\n layout: FlexLayoutResult<C>,\n): ChildLayoutResult<C> | undefined {\n return layout.children[0];\n}\n\n/**\n * 在布局结果中按指定盒模型查找命中的 child,并返回局部坐标。\n */\nexport function findChildAtPoint<C extends CanvasRenderingContext2D>(\n children: ChildLayoutResult<C>[],\n x: number,\n y: number,\n box: \"rect\" | \"contentBox\" = \"contentBox\",\n):\n | {\n child: ChildLayoutResult<C>;\n localX: number;\n localY: number;\n }\n | undefined {\n for (let i = children.length - 1; i >= 0; i -= 1) {\n const child = children[i]!;\n const target = box === \"rect\" ? child.rect : child.contentBox;\n if (!pointInRect(x, y, target)) {\n continue;\n }\n return {\n child,\n localX: x - target.x,\n localY: y - target.y,\n };\n }\n return undefined;\n}\n","import { findChildAtPoint } from \"../layout\";\nimport type { Context, FlexLayoutResult, HitTest, LayoutConstraints, Node } from \"../types\";\nimport { shallow } from \"../utils\";\n\ntype LayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints): FlexLayoutResult<C> | undefined;\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints): void;\n};\n\ntype LayoutContext<C extends CanvasRenderingContext2D> = Context<C> & LayoutCacheAccess<C>;\n\nexport function withConstraints<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n constraints: LayoutConstraints | undefined,\n): Context<C> {\n const next = shallow(ctx);\n next.constraints = constraints;\n return next;\n}\n\nfunction getLayoutContext<C extends CanvasRenderingContext2D>(ctx: Context<C>): LayoutContext<C> {\n return ctx as LayoutContext<C>;\n}\n\nexport function readLayoutResult<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n): FlexLayoutResult<C> | undefined {\n return getLayoutContext(ctx).getLayoutResult(node, ctx.constraints);\n}\n\nexport function writeLayoutResult<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n result: FlexLayoutResult<C>,\n): void {\n getLayoutContext(ctx).setLayoutResult(node, result, ctx.constraints);\n}\n\nfunction ensureLayoutResult<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n): FlexLayoutResult<C> | undefined {\n return readLayoutResult(node, ctx);\n}\n\nexport function drawLayoutChildren<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n x: number,\n y: number,\n): boolean {\n const layoutResult = ensureLayoutResult(node, ctx);\n if (!layoutResult) {\n return false;\n }\n\n let result = false;\n for (const childResult of layoutResult.children) {\n const childDrawn = childResult.node.draw(\n withConstraints(ctx, childResult.constraints),\n x + childResult.contentBox.x,\n y + childResult.contentBox.y,\n );\n result = childDrawn || result;\n }\n return result;\n}\n\nexport function hittestLayoutChildren<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n test: HitTest,\n box: \"rect\" | \"contentBox\" = \"contentBox\",\n): boolean {\n const layoutResult = ensureLayoutResult(node, ctx);\n if (!layoutResult) {\n return false;\n }\n\n const hit = findChildAtPoint(layoutResult.children, test.x, test.y, box);\n if (!hit) {\n return false;\n }\n\n return hit.child.node.hittest(\n withConstraints(ctx, hit.child.constraints),\n {\n ...test,\n x: hit.localX,\n y: hit.localY,\n },\n );\n}\n","import { createRect, findChildAtPoint, getSingleChildLayout } from \"../layout\";\nimport type { Box, Context, HitTest, Node } from \"../types\";\nimport { measureNodeMinContent } from \"./base\";\nimport { Wrapper } from \"./base\";\nimport { readLayoutResult, withConstraints, writeLayoutResult } from \"./shared\";\n\nfunction clampToConstraints(value: number, min?: number, max?: number): number {\n let result = value;\n if (min != null) {\n result = Math.max(result, min);\n }\n if (max != null) {\n result = Math.min(result, max);\n }\n return result;\n}\n\nfunction shrinkConstraint(value: number | undefined, padding: number): number | undefined {\n if (value == null) {\n return undefined;\n }\n return Math.max(0, value - padding);\n}\n\n/**\n * Adds padding around a single child node.\n */\nexport class PaddingBox<C extends CanvasRenderingContext2D> extends Wrapper<C> {\n /**\n * @param inner Wrapped child node.\n * @param padding Padding in CSS pixels on each side.\n */\n constructor(\n inner: Node<C>,\n readonly padding: {\n top?: number;\n bottom?: number;\n left?: number;\n right?: number;\n } = {},\n ) {\n super(inner);\n }\n\n get #top(): number {\n return this.padding.top ?? 0;\n }\n\n get #bottom(): number {\n return this.padding.bottom ?? 0;\n }\n\n get #left(): number {\n return this.padding.left ?? 0;\n }\n\n get #right(): number {\n return this.padding.right ?? 0;\n }\n\n measure(ctx: Context<C>): Box {\n const paddingLeft = this.#left;\n const paddingRight = this.#right;\n const paddingTop = this.#top;\n const paddingBottom = this.#bottom;\n const horizontalPadding = paddingLeft + paddingRight;\n const verticalPadding = paddingTop + paddingBottom;\n const childConstraints = ctx.constraints\n ? {\n ...ctx.constraints,\n minWidth: shrinkConstraint(ctx.constraints.minWidth, horizontalPadding),\n maxWidth: shrinkConstraint(ctx.constraints.maxWidth, horizontalPadding),\n minHeight: shrinkConstraint(ctx.constraints.minHeight, verticalPadding),\n maxHeight: shrinkConstraint(ctx.constraints.maxHeight, verticalPadding),\n }\n : undefined;\n const { width, height } = ctx.measureNode(this.inner, childConstraints);\n const containerBox = createRect(\n 0,\n 0,\n clampToConstraints(width + horizontalPadding, ctx.constraints?.minWidth, ctx.constraints?.maxWidth),\n clampToConstraints(height + verticalPadding, ctx.constraints?.minHeight, ctx.constraints?.maxHeight),\n );\n const childRect = createRect(paddingLeft, paddingTop, width, height);\n writeLayoutResult(this, ctx, {\n containerBox,\n contentBox: childRect,\n children: [\n {\n node: this.inner,\n rect: childRect,\n contentBox: childRect,\n constraints: childConstraints,\n },\n ],\n constraints: ctx.constraints,\n });\n return {\n width: containerBox.width,\n height: containerBox.height,\n };\n }\n\n measureMinContent(ctx: Context<C>): Box {\n const paddingLeft = this.#left;\n const paddingRight = this.#right;\n const paddingTop = this.#top;\n const paddingBottom = this.#bottom;\n const horizontalPadding = paddingLeft + paddingRight;\n const verticalPadding = paddingTop + paddingBottom;\n const childConstraints = ctx.constraints\n ? {\n ...ctx.constraints,\n minWidth: shrinkConstraint(ctx.constraints.minWidth, horizontalPadding),\n maxWidth: shrinkConstraint(ctx.constraints.maxWidth, horizontalPadding),\n minHeight: shrinkConstraint(ctx.constraints.minHeight, verticalPadding),\n maxHeight: shrinkConstraint(ctx.constraints.maxHeight, verticalPadding),\n }\n : undefined;\n const { width, height } = measureNodeMinContent(ctx, this.inner, childConstraints);\n return {\n width: width + horizontalPadding,\n height: height + verticalPadding,\n };\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return this.inner.draw(ctx, x + this.#left, y + this.#top);\n }\n\n const childResult = getSingleChildLayout(layoutResult);\n if (!childResult) {\n return false;\n }\n\n return childResult.node.draw(\n withConstraints(ctx, childResult.constraints),\n x + childResult.rect.x,\n y + childResult.rect.y,\n );\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return false;\n }\n\n const hit = findChildAtPoint(layoutResult.children, test.x, test.y, \"rect\");\n if (!hit) {\n return false;\n }\n\n return hit.child.node.hittest(\n withConstraints(ctx, hit.child.constraints),\n {\n ...test,\n x: hit.localX,\n y: hit.localY,\n },\n );\n }\n}\n\n/**\n * A leaf node with a fixed size and no drawing behavior.\n */\nexport class Fixed<C extends CanvasRenderingContext2D> implements Node<C> {\n /**\n * @param width Fixed width in CSS pixels.\n * @param height Fixed height in CSS pixels.\n */\n constructor(\n readonly width: number,\n readonly height: number,\n ) {}\n\n measure(_ctx: Context<C>): Box {\n return { width: this.width, height: this.height };\n }\n\n measureMinContent(_ctx: Context<C>): Box {\n return { width: this.width, height: this.height };\n }\n\n draw(_ctx: Context<C>, _x: number, _y: number): boolean {\n return false;\n }\n\n hittest(_ctx: Context<C>, _test: HitTest): boolean {\n return false;\n }\n}\n","import { computeContentBox, createRect } from \"../layout\";\nimport type {\n Axis,\n Box,\n ChildLayoutResult,\n Context,\n CrossAxisAlignment,\n FlexContainerOptions,\n FlexItemOptions,\n HitTest,\n LayoutConstraints,\n Node,\n} from \"../types\";\nimport { Group, measureNodeMinContent, Wrapper } from \"./base\";\nimport { drawLayoutChildren, hittestLayoutChildren, writeLayoutResult } from \"./shared\";\n\nfunction getMainSize(axis: Axis, box: Box): number {\n return axis === \"row\" ? box.width : box.height;\n}\n\nfunction getCrossSize(axis: Axis, box: Box): number {\n return axis === \"row\" ? box.height : box.width;\n}\n\nfunction getMinMain(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.minWidth : constraints?.minHeight;\n}\n\nfunction getMaxMain(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.maxWidth : constraints?.maxHeight;\n}\n\nfunction getMinCross(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.minHeight : constraints?.minWidth;\n}\n\nfunction getMaxCross(axis: Axis, constraints?: LayoutConstraints): number | undefined {\n return axis === \"row\" ? constraints?.maxHeight : constraints?.maxWidth;\n}\n\nfunction createAxisConstraints(\n axis: Axis,\n constraints: LayoutConstraints | undefined,\n main: { min?: number; max?: number },\n cross: { min?: number; max?: number } = {},\n): LayoutConstraints | undefined {\n if (\n constraints == null &&\n main.min == null &&\n main.max == null &&\n cross.min == null &&\n cross.max == null\n ) {\n return undefined;\n }\n\n const next: LayoutConstraints = {\n ...constraints,\n };\n if (axis === \"row\") {\n next.minWidth = main.min;\n next.maxWidth = main.max;\n next.minHeight = cross.min;\n next.maxHeight = cross.max;\n } else {\n next.minHeight = main.min;\n next.maxHeight = main.max;\n next.minWidth = cross.min;\n next.maxWidth = cross.max;\n }\n return next;\n}\n\nfunction clampToConstraints(value: number, min?: number, max?: number): number {\n let result = value;\n if (min != null) {\n result = Math.max(result, min);\n }\n if (max != null) {\n result = Math.min(result, max);\n }\n return result;\n}\n\nfunction constraintsEqual(left: LayoutConstraints | undefined, right: LayoutConstraints | undefined): boolean {\n if (left === right) {\n return true;\n }\n if (left == null || right == null) {\n return left == null && right == null;\n }\n return left.minWidth === right.minWidth\n && left.maxWidth === right.maxWidth\n && left.minHeight === right.minHeight\n && left.maxHeight === right.maxHeight;\n}\n\nfunction getCrossAlignment(alignSelf: CrossAxisAlignment | \"auto\" | undefined, alignItems: CrossAxisAlignment): CrossAxisAlignment {\n if (alignSelf == null || alignSelf === \"auto\") {\n return alignItems;\n }\n return alignSelf;\n}\n\nfunction getJustifySpacing(\n justifyContent: NonNullable<FlexContainerOptions[\"justifyContent\"]>,\n freeSpace: number,\n itemCount: number,\n gap: number,\n): { leading: number; between: number } {\n switch (justifyContent) {\n case \"center\":\n return { leading: freeSpace / 2, between: gap };\n case \"end\":\n return { leading: freeSpace, between: gap };\n case \"space-between\":\n return {\n leading: 0,\n between: itemCount > 1 ? gap + freeSpace / (itemCount - 1) : gap,\n };\n case \"space-around\":\n return {\n leading: itemCount > 0 ? freeSpace / itemCount / 2 : 0,\n between: itemCount > 0 ? gap + freeSpace / itemCount : gap,\n };\n case \"space-evenly\":\n return {\n leading: itemCount > 0 ? freeSpace / (itemCount + 1) : 0,\n between: itemCount > 0 ? gap + freeSpace / (itemCount + 1) : gap,\n };\n case \"start\":\n default:\n return { leading: 0, between: gap };\n }\n}\n\nfunction getCrossOffset(align: CrossAxisAlignment, frameCross: number, contentCross: number): number {\n switch (align) {\n case \"center\":\n return (frameCross - contentCross) / 2;\n case \"end\":\n return frameCross - contentCross;\n case \"stretch\":\n case \"start\":\n default:\n return 0;\n }\n}\n\nfunction createRectFromAxis(axis: Axis, main: number, cross: number, mainSize: number, crossSize: number) {\n return axis === \"row\"\n ? createRect(main, cross, mainSize, crossSize)\n : createRect(cross, main, crossSize, mainSize);\n}\n\nconst SHRINK_EPSILON = 1e-6;\n\ntype FlexMeasurement<C extends CanvasRenderingContext2D> = {\n child: Node<C>;\n item: FlexItemOptions;\n basisMeasured: Box;\n measured: Box;\n basisConstraints?: LayoutConstraints;\n initialConstraints?: LayoutConstraints;\n finalConstraints?: LayoutConstraints;\n allocatedMain?: number;\n grow: number;\n shrink: number;\n effectiveAlign: CrossAxisAlignment;\n stretch: boolean;\n basis: number;\n minContentMain: number;\n finalMain: number;\n frozen: boolean;\n frameMain: number;\n frameCross: number;\n};\n\ntype MeasuredLayout<C extends CanvasRenderingContext2D> = {\n box: Box;\n layout: {\n containerBox: ChildLayoutResult<C>[\"rect\"];\n contentBox: ChildLayoutResult<C>[\"contentBox\"];\n children: ChildLayoutResult<C>[];\n constraints?: LayoutConstraints;\n };\n};\n\nfunction readFlexItemOptions<C extends CanvasRenderingContext2D>(child: Node<C>): FlexItemOptions {\n if (child instanceof FlexItem) {\n return child.item;\n }\n return {};\n}\n\nexport function computeFlexLayout<C extends CanvasRenderingContext2D>(\n children: readonly Node<C>[],\n options: FlexContainerOptions,\n constraints: LayoutConstraints | undefined,\n measureChild: (node: Node<C>, constraints?: LayoutConstraints) => Box,\n measureChildMinContent: (node: Node<C>, constraints?: LayoutConstraints) => Box,\n): MeasuredLayout<C> {\n const axis = options.direction ?? \"row\";\n const gap = options.gap ?? 0;\n const justifyContent = options.justifyContent ?? \"start\";\n const alignItems = options.alignItems ?? \"start\";\n const reverse = options.reverse ?? false;\n const mainAxisSize = options.mainAxisSize ?? \"fill\";\n const orderedChildren = reverse ? [...children].reverse() : children;\n const maxMain = getMaxMain(axis, constraints);\n const minMain = getMinMain(axis, constraints);\n const maxCross = getMaxCross(axis, constraints);\n const minCross = getMinCross(axis, constraints);\n const gapTotal = orderedChildren.length > 1 ? gap * (orderedChildren.length - 1) : 0;\n const finiteMain = maxMain != null;\n const finiteCross = maxCross != null;\n const availableMain = finiteMain ? Math.max(0, maxMain - gapTotal) : undefined;\n let totalGrow = 0;\n let totalBasis = 0;\n let nonGrowBasis = 0;\n const measurements = new Map<Node<C>, FlexMeasurement<C>>();\n const basisConstraints = createAxisConstraints(\n axis,\n constraints,\n {\n min: undefined,\n max: undefined,\n },\n {\n min: undefined,\n max: maxCross,\n },\n );\n\n for (const child of orderedChildren) {\n const item = readFlexItemOptions(child);\n const grow = item.grow ?? 0;\n const shrink = item.shrink ?? 0;\n totalGrow += grow;\n const effectiveAlign = getCrossAlignment(item.alignSelf, alignItems);\n const stretch = effectiveAlign === \"stretch\";\n const basisMeasured = measureChild(child, basisConstraints);\n const basis = getMainSize(axis, basisMeasured);\n\n totalBasis += basis;\n if (grow <= 0) {\n nonGrowBasis += basis;\n }\n\n measurements.set(child, {\n child,\n item,\n basisMeasured,\n measured: basisMeasured,\n basisConstraints,\n initialConstraints: basisConstraints,\n finalConstraints: basisConstraints,\n allocatedMain: undefined,\n grow,\n shrink,\n effectiveAlign,\n stretch,\n basis,\n minContentMain: basis,\n finalMain: basis,\n frozen: false,\n frameMain: basis,\n frameCross: getCrossSize(axis, basisMeasured),\n });\n }\n\n const entersShrinkPath = finiteMain && availableMain != null && totalBasis - availableMain > SHRINK_EPSILON;\n\n if (entersShrinkPath) {\n const totalDeficit = totalBasis - availableMain!;\n let remainingDeficit = totalDeficit;\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n const minContentMeasured = measureChildMinContent(child, measurement.basisConstraints);\n measurement.minContentMain = Math.min(measurement.basis, getMainSize(axis, minContentMeasured));\n measurement.finalMain = measurement.basis;\n measurement.frozen = measurement.shrink <= 0 || measurement.basis - measurement.minContentMain <= SHRINK_EPSILON;\n }\n\n while (remainingDeficit > SHRINK_EPSILON) {\n const active = orderedChildren\n .map((child) => measurements.get(child)!)\n .filter((measurement) => !measurement.frozen && measurement.shrink > 0);\n const totalScaled = active.reduce((sum, measurement) => sum + measurement.shrink * measurement.basis, 0);\n\n if (active.length === 0 || totalScaled <= SHRINK_EPSILON) {\n break;\n }\n\n let frozeAny = false;\n for (const measurement of active) {\n const tentative = measurement.basis - remainingDeficit * ((measurement.shrink * measurement.basis) / totalScaled);\n if (tentative <= measurement.minContentMain + SHRINK_EPSILON) {\n measurement.finalMain = measurement.minContentMain;\n measurement.frozen = true;\n frozeAny = true;\n } else {\n measurement.finalMain = tentative;\n }\n }\n\n if (!frozeAny) {\n remainingDeficit = 0;\n break;\n }\n\n let absorbedDeficit = 0;\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (measurement.frozen) {\n absorbedDeficit += Math.max(0, measurement.basis - measurement.finalMain);\n }\n }\n remainingDeficit = Math.max(0, totalDeficit - absorbedDeficit);\n }\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n measurement.measured = measurement.basisMeasured;\n measurement.initialConstraints = measurement.basisConstraints;\n measurement.finalConstraints = createAxisConstraints(\n axis,\n constraints,\n {\n min: undefined,\n max: measurement.finalMain,\n },\n {\n min: undefined,\n max: maxCross,\n },\n );\n measurement.allocatedMain = undefined;\n measurement.frameMain = measurement.finalMain;\n measurement.frameCross = getCrossSize(axis, measurement.measured);\n }\n } else {\n const remainingMain = finiteMain && availableMain != null ? Math.max(0, availableMain - nonGrowBasis) : undefined;\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (!(measurement.grow > 0 && finiteMain && remainingMain != null && totalGrow > 0)) {\n measurement.measured = measurement.basisMeasured;\n measurement.initialConstraints = measurement.basisConstraints;\n measurement.finalConstraints = finiteMain\n ? createAxisConstraints(\n axis,\n constraints,\n {\n min: undefined,\n max: measurement.finalMain,\n },\n {\n min: undefined,\n max: maxCross,\n },\n )\n : measurement.basisConstraints;\n measurement.allocatedMain = undefined;\n measurement.finalMain = measurement.basis;\n measurement.frameMain = measurement.basis;\n measurement.frameCross = getCrossSize(axis, measurement.measured);\n continue;\n }\n\n const allocatedMain = (remainingMain * measurement.grow) / totalGrow;\n const childConstraints = createAxisConstraints(\n axis,\n constraints,\n {\n max: allocatedMain,\n },\n {\n min: undefined,\n max: maxCross,\n },\n );\n const measured = measureChild(child, childConstraints);\n measurement.measured = measured;\n measurement.initialConstraints = childConstraints;\n measurement.finalConstraints = childConstraints;\n measurement.allocatedMain = allocatedMain;\n measurement.finalMain = allocatedMain;\n measurement.frameMain = allocatedMain;\n measurement.frameCross = getCrossSize(axis, measured);\n }\n }\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (!constraintsEqual(measurement.initialConstraints, measurement.finalConstraints)) {\n measurement.measured = measureChild(child, measurement.finalConstraints);\n }\n measurement.frameMain = measurement.finalMain;\n measurement.frameCross = getCrossSize(axis, measurement.measured);\n }\n\n let contentMain = gapTotal;\n let contentCross = 0;\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n contentMain += measurement.frameMain;\n contentCross = Math.max(contentCross, measurement.frameCross);\n }\n\n const containerMain = finiteMain && mainAxisSize === \"fill\"\n ? Math.max(maxMain!, contentMain)\n : clampToConstraints(contentMain, minMain, maxMain);\n const containerCross = clampToConstraints(contentCross, minCross, maxCross);\n if (finiteCross) {\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n if (!measurement.stretch) {\n continue;\n }\n\n const finalConstraints = createAxisConstraints(\n axis,\n measurement.finalConstraints,\n {\n min: getMinMain(axis, measurement.finalConstraints),\n max: getMaxMain(axis, measurement.finalConstraints),\n },\n {\n min: containerCross,\n max: containerCross,\n },\n );\n const remeasured = measureChild(child, finalConstraints);\n measurement.measured = remeasured;\n measurement.finalConstraints = finalConstraints;\n measurement.frameCross = containerCross;\n measurement.frameMain = measurement.allocatedMain ?? getMainSize(axis, remeasured);\n }\n\n contentMain = gapTotal;\n contentCross = 0;\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n contentMain += measurement.frameMain;\n contentCross = Math.max(contentCross, getCrossSize(axis, measurement.measured));\n }\n }\n\n const finalContainerMain = finiteMain && mainAxisSize === \"fill\"\n ? Math.max(maxMain!, contentMain)\n : clampToConstraints(contentMain, minMain, maxMain);\n const freeSpace = Math.max(0, finalContainerMain - contentMain);\n const spacing = getJustifySpacing(justifyContent, freeSpace, orderedChildren.length, gap);\n const childResults: ChildLayoutResult<C>[] = [];\n let cursor = spacing.leading;\n\n for (const child of orderedChildren) {\n const measurement = measurements.get(child)!;\n const frameCross = measurement.stretch && finiteCross ? containerCross : measurement.frameCross;\n const contentMainSize = getMainSize(axis, measurement.measured);\n const contentCrossSize = getCrossSize(axis, measurement.measured);\n const rectCross = measurement.stretch ? 0 : getCrossOffset(measurement.effectiveAlign, containerCross, frameCross);\n const contentCrossOffset = rectCross + getCrossOffset(measurement.effectiveAlign, frameCross, contentCrossSize);\n const rect = createRectFromAxis(axis, cursor, rectCross, measurement.frameMain, frameCross);\n const contentBox = createRectFromAxis(axis, cursor, contentCrossOffset, contentMainSize, contentCrossSize);\n\n childResults.push({\n node: child,\n rect,\n contentBox,\n constraints: measurement.finalConstraints,\n });\n cursor += measurement.frameMain + spacing.between;\n }\n\n const containerBox = axis === \"row\"\n ? createRect(0, 0, finalContainerMain, containerCross)\n : createRect(0, 0, containerCross, finalContainerMain);\n const finalContentBox = childResults.length > 0\n ? computeContentBox(childResults)\n : createRect(0, 0, 0, 0);\n\n return {\n box: {\n width: containerBox.width,\n height: containerBox.height,\n },\n layout: {\n containerBox,\n contentBox: finalContentBox,\n children: childResults,\n constraints,\n },\n };\n}\n\n/**\n * Wraps a child node with per-item flex options.\n */\nexport class FlexItem<C extends CanvasRenderingContext2D> extends Wrapper<C> {\n /**\n * @param inner Wrapped child node.\n * @param item Flex behavior overrides for the child.\n */\n constructor(\n inner: Node<C>,\n readonly item: FlexItemOptions = {},\n ) {\n super(inner);\n }\n}\n\n/**\n * Lays out children in a single flex row or column.\n */\nexport class Flex<C extends CanvasRenderingContext2D> extends Group<C> {\n /**\n * @param children Child nodes in visual order.\n * @param options Flex container configuration.\n */\n constructor(\n children: Node<C>[],\n readonly options: FlexContainerOptions = {},\n ) {\n super(children);\n }\n\n measure(ctx: Context<C>): Box {\n const result = computeFlexLayout(\n this.children,\n this.options,\n ctx.constraints,\n (node, constraints) => ctx.measureNode(node, constraints),\n (node, constraints) => measureNodeMinContent(ctx, node, constraints),\n );\n writeLayoutResult(this, ctx, result.layout);\n return result.box;\n }\n\n measureMinContent(ctx: Context<C>): Box {\n const axis = this.options.direction ?? \"row\";\n const gap = this.options.gap ?? 0;\n const orderedChildren = this.options.reverse ? [...this.children].reverse() : this.children;\n const gapTotal = orderedChildren.length > 1 ? gap * (orderedChildren.length - 1) : 0;\n const childConstraints = createAxisConstraints(\n axis,\n ctx.constraints,\n {\n min: undefined,\n max: undefined,\n },\n {\n min: undefined,\n max: getMaxCross(axis, ctx.constraints),\n },\n );\n\n let width = axis === \"row\" ? gapTotal : 0;\n let height = axis === \"column\" ? gapTotal : 0;\n\n for (const child of orderedChildren) {\n const measured = measureNodeMinContent(ctx, child, childConstraints);\n if (axis === \"row\") {\n width += measured.width;\n height = Math.max(height, measured.height);\n } else {\n width = Math.max(width, measured.width);\n height += measured.height;\n }\n }\n\n return { width, height };\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n return drawLayoutChildren(this, ctx, x, y);\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n return hittestLayoutChildren(this, ctx, test, \"contentBox\");\n }\n}\n","import { createRect, findChildAtPoint, getSingleChildLayout } from \"../layout\";\nimport type { Box, Context, HitTest, Node, TextAlign } from \"../types\";\nimport { measureNodeMinContent, Wrapper } from \"./base\";\nimport { readLayoutResult, withConstraints, writeLayoutResult } from \"./shared\";\n\nfunction resolveHorizontalOffset(align: TextAlign, availableWidth: number, childWidth: number): number {\n switch (align) {\n case \"center\":\n return (availableWidth - childWidth) / 2;\n case \"end\":\n return availableWidth - childWidth;\n case \"start\":\n return 0;\n }\n}\n\n/**\n * Aligns a single child horizontally within the available width.\n */\nexport class Place<C extends CanvasRenderingContext2D> extends Wrapper<C> {\n /**\n * @param inner Wrapped child node.\n * @param options Alignment behavior for the child.\n */\n constructor(\n inner: Node<C>,\n readonly options: {\n align?: TextAlign;\n expand?: boolean;\n } = {},\n ) {\n super(inner);\n }\n\n measure(ctx: Context<C>): Box {\n const availableWidth = ctx.constraints?.maxWidth;\n const expand = this.options.expand ?? true;\n const childConstraints = ctx.constraints\n ? {\n ...ctx.constraints,\n }\n : undefined;\n const childBox = ctx.measureNode(this.inner, childConstraints);\n let width = expand && availableWidth != null ? availableWidth : childBox.width;\n if (ctx.constraints?.minWidth != null) {\n width = Math.max(width, ctx.constraints.minWidth);\n }\n if (ctx.constraints?.maxWidth != null) {\n width = Math.min(width, ctx.constraints.maxWidth);\n }\n\n const align = this.options.align ?? \"start\";\n const childRect = createRect(resolveHorizontalOffset(align, width, childBox.width), 0, childBox.width, childBox.height);\n\n writeLayoutResult(this, ctx, {\n containerBox: createRect(0, 0, width, childBox.height),\n contentBox: childRect,\n children: [\n {\n node: this.inner,\n rect: childRect,\n contentBox: createRect(0, 0, childBox.width, childBox.height),\n constraints: childConstraints,\n },\n ],\n constraints: ctx.constraints,\n });\n\n return {\n width,\n height: childBox.height,\n };\n }\n\n measureMinContent(ctx: Context<C>): Box {\n return measureNodeMinContent(ctx, this.inner);\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return this.inner.draw(ctx, x, y);\n }\n\n const childResult = getSingleChildLayout(layoutResult);\n if (!childResult) {\n return false;\n }\n const childCtx = withConstraints(ctx, childResult.constraints);\n return childResult.node.draw(childCtx, x + childResult.rect.x, y + childResult.rect.y);\n }\n\n hittest(ctx: Context<C>, test: HitTest): boolean {\n const layoutResult = readLayoutResult(this, ctx);\n if (!layoutResult) {\n return false;\n }\n\n const hit = findChildAtPoint(layoutResult.children, test.x, test.y, \"rect\");\n if (!hit) {\n return false;\n }\n\n return hit.child.node.hittest(\n withConstraints(ctx, hit.child.constraints),\n {\n ...test,\n x: hit.localX,\n y: hit.localY,\n },\n );\n }\n}\n","import type { Context, TextEllipsisPosition } from \"../types\";\n\nexport const FONT_SHIFT_PROBE = \"M\";\nexport const ELLIPSIS_GLYPH = \"…\";\nexport const INTRINSIC_MAX_WIDTH = Number.POSITIVE_INFINITY;\nexport const MIN_CONTENT_WIDTH_EPSILON = 0.001;\nexport const FONT_SHIFT_CACHE_CAPACITY = 64;\nexport const ELLIPSIS_WIDTH_CACHE_CAPACITY = 64;\n\nconst fontShiftCache = new Map<string, number>();\nconst ellipsisWidthCache = new Map<string, number>();\n\nlet sharedGraphemeSegmenter: Intl.Segmenter | null | undefined;\n\nexport function readLruValue<T>(cache: Map<string, T>, key: string): T | undefined {\n const cached = cache.get(key);\n if (cached == null) {\n return undefined;\n }\n cache.delete(key);\n cache.set(key, cached);\n return cached;\n}\n\nexport function writeLruValue<T>(cache: Map<string, T>, key: string, value: T, capacity: number): T {\n if (cache.has(key)) {\n cache.delete(key);\n } else if (cache.size >= capacity) {\n const firstKey = cache.keys().next().value;\n if (firstKey != null) {\n cache.delete(firstKey);\n }\n }\n cache.set(key, value);\n return value;\n}\n\nexport function measureFontShift<C extends CanvasRenderingContext2D>(ctx: Context<C>): number {\n const font = ctx.graphics.font;\n const cached = readLruValue(fontShiftCache, font);\n if (cached != null) {\n return cached;\n }\n const {\n fontBoundingBoxAscent: ascent = 0,\n fontBoundingBoxDescent: descent = 0,\n } = ctx.graphics.measureText(FONT_SHIFT_PROBE);\n return writeLruValue(fontShiftCache, font, ascent - descent, FONT_SHIFT_CACHE_CAPACITY);\n}\n\nexport function measureEllipsisWidth<C extends CanvasRenderingContext2D>(ctx: Context<C>): number {\n const font = ctx.graphics.font;\n const cached = readLruValue(ellipsisWidthCache, font);\n if (cached != null) {\n return cached;\n }\n return writeLruValue(\n ellipsisWidthCache,\n font,\n ctx.graphics.measureText(ELLIPSIS_GLYPH).width,\n ELLIPSIS_WIDTH_CACHE_CAPACITY,\n );\n}\n\nfunction getGraphemeSegmenter(): Intl.Segmenter | null {\n if (sharedGraphemeSegmenter !== undefined) {\n return sharedGraphemeSegmenter;\n }\n sharedGraphemeSegmenter = typeof Intl.Segmenter === \"function\"\n ? new Intl.Segmenter(undefined, { granularity: \"grapheme\" })\n : null;\n return sharedGraphemeSegmenter;\n}\n\nexport function splitGraphemes(text: string): string[] {\n const segmenter = getGraphemeSegmenter();\n if (segmenter == null) {\n return Array.from(text);\n }\n const graphemes: string[] = [];\n for (const part of segmenter.segment(text)) {\n graphemes.push(part.segment);\n }\n return graphemes;\n}\n\nexport function buildPrefixWidths(widths: readonly number[]): number[] {\n const cumulativeWidths = [0];\n let total = 0;\n for (const width of widths) {\n total += width;\n cumulativeWidths.push(total);\n }\n return cumulativeWidths;\n}\n\nexport function buildSuffixWidths(widths: readonly number[]): number[] {\n const cumulativeWidths = [0];\n let total = 0;\n for (let index = widths.length - 1; index >= 0; index -= 1) {\n total += widths[index] ?? 0;\n cumulativeWidths.push(total);\n }\n return cumulativeWidths;\n}\n\nexport function findMaxFittingCount(cumulativeWidths: readonly number[], maxWidth: number): number {\n if (maxWidth <= 0) {\n return 0;\n }\n let low = 0;\n let high = cumulativeWidths.length - 1;\n while (low < high) {\n const mid = Math.floor((low + high + 1) / 2);\n if ((cumulativeWidths[mid] ?? 0) <= maxWidth) {\n low = mid;\n } else {\n high = mid - 1;\n }\n }\n return low;\n}\n\nexport function normalizeMaxLines(maxLines: number | undefined): number | undefined {\n if (maxLines == null || !Number.isFinite(maxLines)) {\n return undefined;\n }\n return Math.max(1, Math.trunc(maxLines));\n}\n\nexport function selectEllipsisUnitCounts({\n position,\n prefixWidths,\n suffixWidths,\n unitCount,\n availableWidth,\n getMaxSuffixCount = (prefixCount) => unitCount - prefixCount,\n}: {\n position: TextEllipsisPosition;\n prefixWidths: readonly number[];\n suffixWidths: readonly number[];\n unitCount: number;\n availableWidth: number;\n getMaxSuffixCount?: (prefixCount: number) => number;\n}): { prefixCount: number; suffixCount: number } {\n let prefixCount = 0;\n let suffixCount = 0;\n\n switch (position) {\n case \"start\":\n suffixCount = Math.min(unitCount, findMaxFittingCount(suffixWidths, availableWidth));\n break;\n case \"middle\": {\n let bestVisibleUnits = -1;\n let bestBalanceScore = Number.NEGATIVE_INFINITY;\n for (let nextPrefixCount = 0; nextPrefixCount <= unitCount; nextPrefixCount += 1) {\n const prefixWidth = prefixWidths[nextPrefixCount] ?? 0;\n if (prefixWidth > availableWidth) {\n break;\n }\n const remainingWidth = availableWidth - prefixWidth;\n const maxSuffixCount = Math.max(0, getMaxSuffixCount(nextPrefixCount));\n const nextSuffixCount = Math.min(maxSuffixCount, findMaxFittingCount(suffixWidths, remainingWidth));\n const visibleUnits = nextPrefixCount + nextSuffixCount;\n const balanceScore = -Math.abs(nextPrefixCount - nextSuffixCount);\n if (\n visibleUnits > bestVisibleUnits ||\n (visibleUnits === bestVisibleUnits && balanceScore > bestBalanceScore) ||\n (visibleUnits === bestVisibleUnits && balanceScore === bestBalanceScore && nextPrefixCount > prefixCount)\n ) {\n prefixCount = nextPrefixCount;\n suffixCount = nextSuffixCount;\n bestVisibleUnits = visibleUnits;\n bestBalanceScore = balanceScore;\n }\n }\n break;\n }\n case \"end\":\n prefixCount = Math.min(unitCount, findMaxFittingCount(prefixWidths, availableWidth));\n break;\n }\n\n return { prefixCount, suffixCount };\n}\n","import { layoutNextLine, prepareWithSegments, type PreparedTextWithSegments } from \"@chenglou/pretext\";\nimport type { Context, TextOverflowWrapMode, TextWhiteSpaceMode, TextWordBreakMode } from \"../types\";\nimport { INTRINSIC_MAX_WIDTH, readLruValue, splitGraphemes, writeLruValue } from \"./core\";\n\nexport const PREPARED_TEXT_CACHE_CAPACITY = 512;\n\nconst LINE_START_CURSOR = { segmentIndex: 0, graphemeIndex: 0 } as const;\n\nconst preparedTextCache = new Map<string, PreparedTextWithSegments>();\nconst preparedUnitCache = new WeakMap<PreparedTextWithSegments, PreparedTextUnit[]>();\n\nexport type PreparedTextUnit = {\n text: string;\n width: number;\n};\n\nfunction getPreparedTextCacheKey(\n text: string,\n font: string,\n whiteSpace: TextWhiteSpaceMode,\n wordBreak: TextWordBreakMode,\n): string {\n return `${font}\\u0000${whiteSpace}\\u0000${wordBreak}\\u0000${text}`;\n}\n\nexport function readPreparedText(\n text: string,\n font: string,\n whiteSpace: TextWhiteSpaceMode,\n wordBreak: TextWordBreakMode,\n): PreparedTextWithSegments {\n const key = getPreparedTextCacheKey(text, font, whiteSpace, wordBreak);\n const cached = readLruValue(preparedTextCache, key);\n if (cached != null) {\n return cached;\n }\n return writeLruValue(\n preparedTextCache,\n key,\n prepareWithSegments(text, font, { whiteSpace, wordBreak }),\n PREPARED_TEXT_CACHE_CAPACITY,\n );\n}\n\nexport function readPreparedFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode,\n wordBreak: TextWordBreakMode,\n): { text: string; prepared: PreparedTextWithSegments } | undefined {\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const line = layoutNextLine(prepared, LINE_START_CURSOR, INTRINSIC_MAX_WIDTH);\n if (line == null) {\n return undefined;\n }\n return {\n text: line.text,\n prepared: readPreparedText(line.text, ctx.graphics.font, whiteSpace, wordBreak),\n };\n}\n\nexport function measurePreparedMinContentWidth(\n prepared: PreparedTextWithSegments,\n overflowWrap: TextOverflowWrapMode = \"break-word\",\n): number {\n let maxWidth = 0;\n let maxAnyWidth = 0;\n for (let index = 0; index < prepared.widths.length; index += 1) {\n const segmentWidth = prepared.widths[index] ?? 0;\n maxAnyWidth = Math.max(maxAnyWidth, segmentWidth);\n const segment = prepared.segments[index];\n if (segment != null && segment.trim().length > 0) {\n const breakableWidths = prepared.breakableFitAdvances[index];\n const minContentWidth = overflowWrap === \"anywhere\" && breakableWidths != null && breakableWidths.length > 0\n ? breakableWidths.reduce((widest, width) => Math.max(widest, width), 0)\n : segmentWidth;\n maxWidth = Math.max(maxWidth, minContentWidth);\n }\n }\n return maxWidth > 0 ? maxWidth : maxAnyWidth;\n}\n\nexport function getPreparedUnits(prepared: PreparedTextWithSegments): PreparedTextUnit[] {\n const cached = preparedUnitCache.get(prepared);\n if (cached != null) {\n return cached;\n }\n\n const units: PreparedTextUnit[] = [];\n for (let index = 0; index < prepared.segments.length; index += 1) {\n const segment = prepared.segments[index] ?? \"\";\n const segmentWidth = prepared.widths[index] ?? 0;\n const breakableWidths = prepared.breakableFitAdvances[index];\n if (breakableWidths != null && segment.length > 0) {\n const graphemes = splitGraphemes(segment);\n if (graphemes.length === breakableWidths.length) {\n for (let graphemeIndex = 0; graphemeIndex < graphemes.length; graphemeIndex += 1) {\n units.push({\n text: graphemes[graphemeIndex] ?? \"\",\n width: breakableWidths[graphemeIndex] ?? 0,\n });\n }\n continue;\n }\n }\n\n if (segment.length > 0 || segmentWidth > 0) {\n units.push({ text: segment, width: segmentWidth });\n }\n }\n\n preparedUnitCache.set(prepared, units);\n return units;\n}\n\nexport function joinUnitText(units: readonly PreparedTextUnit[], start: number, end: number): string {\n if (start >= end) {\n return \"\";\n }\n return units.slice(start, end).map((unit) => unit.text).join(\"\");\n}\n","import {\n layoutNextLine,\n layoutWithLines,\n measureLineStats,\n measureNaturalWidth,\n type PreparedTextWithSegments,\n} from \"@chenglou/pretext\";\nimport type {\n Context,\n TextEllipsisPosition,\n TextOverflowMode,\n TextOverflowWrapMode,\n TextWhiteSpaceMode,\n TextWordBreakMode,\n} from \"../types\";\nimport {\n ELLIPSIS_GLYPH,\n INTRINSIC_MAX_WIDTH,\n MIN_CONTENT_WIDTH_EPSILON,\n buildPrefixWidths,\n buildSuffixWidths,\n measureEllipsisWidth,\n measureFontShift,\n normalizeMaxLines,\n selectEllipsisUnitCounts,\n} from \"./core\";\nimport {\n getPreparedUnits,\n joinUnitText,\n measurePreparedMinContentWidth,\n readPreparedFirstLine,\n readPreparedText,\n} from \"./plain-core\";\n\nconst LINE_START_CURSOR = { segmentIndex: 0, graphemeIndex: 0 } as const;\n\nexport interface TextLayout {\n width: number;\n text: string;\n shift: number;\n}\n\nexport interface TextMeasurement {\n width: number;\n lineCount: number;\n}\n\nexport interface OverflowTextLayout extends TextLayout {\n overflowed: boolean;\n}\n\nexport interface OverflowTextBlockLayout {\n width: number;\n lines: OverflowTextLayout[];\n overflowed: boolean;\n}\n\ntype TextBlockLayout = {\n width: number;\n lines: TextLayout[];\n};\n\nfunction clampMaxWidth(maxWidth: number): number {\n return Math.max(0, maxWidth);\n}\n\nfunction createEllipsisOnlyLayout<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n maxWidth: number,\n shift: number,\n): OverflowTextLayout {\n const ellipsisWidth = measureEllipsisWidth(ctx);\n if (ellipsisWidth > maxWidth) {\n return { width: 0, text: \"\", shift, overflowed: true };\n }\n return { width: ellipsisWidth, text: ELLIPSIS_GLYPH, shift, overflowed: true };\n}\n\nfunction toTextBlockLayout(lines: { width: number; text: string }[], shift: number): TextBlockLayout {\n const mappedLines = lines.map((line) => ({ width: line.width, text: line.text, shift }));\n const width = mappedLines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0);\n return { width, lines: mappedLines };\n}\n\nfunction layoutPreparedEllipsis<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n prepared: PreparedTextWithSegments,\n text: string,\n maxWidth: number,\n shift: number,\n position: TextEllipsisPosition,\n forceEllipsis = false,\n): OverflowTextLayout {\n const intrinsicWidth = measureNaturalWidth(prepared);\n if (!forceEllipsis && intrinsicWidth <= maxWidth) {\n return { width: intrinsicWidth, text, shift, overflowed: false };\n }\n\n const ellipsisWidth = measureEllipsisWidth(ctx);\n if (ellipsisWidth > maxWidth) {\n return { width: 0, text: \"\", shift, overflowed: true };\n }\n\n const units = getPreparedUnits(prepared);\n if (units.length === 0) {\n return createEllipsisOnlyLayout(ctx, maxWidth, shift);\n }\n\n const availableWidth = Math.max(0, maxWidth - ellipsisWidth);\n const prefixWidths = buildPrefixWidths(units.map((unit) => unit.width));\n const suffixWidths = buildSuffixWidths(units.map((unit) => unit.width));\n const { prefixCount, suffixCount } = selectEllipsisUnitCounts({\n position,\n prefixWidths,\n suffixWidths,\n unitCount: units.length,\n availableWidth,\n });\n\n const prefixWidth = prefixWidths[prefixCount] ?? 0;\n const suffixWidth = suffixWidths[suffixCount] ?? 0;\n const prefixText = joinUnitText(units, 0, prefixCount);\n const suffixText = joinUnitText(units, units.length - suffixCount, units.length);\n\n return {\n width: prefixWidth + ellipsisWidth + suffixWidth,\n text: `${prefixText}${ELLIPSIS_GLYPH}${suffixText}`,\n shift,\n overflowed: true,\n };\n}\n\nfunction layoutForcedEllipsizedLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n shift: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): OverflowTextLayout {\n if (text.length === 0) {\n return createEllipsisOnlyLayout(ctx, maxWidth, shift);\n }\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n return layoutPreparedEllipsis(ctx, prepared, text, maxWidth, shift, \"end\", true);\n}\n\nexport function layoutFirstLineIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextLayout {\n const firstLine = readPreparedFirstLine(ctx, text, whiteSpace, wordBreak);\n if (firstLine == null) {\n return { width: 0, text: \"\", shift: 0 };\n }\n const shift = measureFontShift(ctx);\n return {\n width: measureNaturalWidth(firstLine.prepared),\n text: firstLine.text,\n shift,\n };\n}\n\nexport function measureTextIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextMeasurement {\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const { maxLineWidth: width, lineCount } = measureLineStats(prepared, INTRINSIC_MAX_WIDTH);\n if (lineCount === 0) {\n return { width: 0, lineCount: 0 };\n }\n return { width, lineCount };\n}\n\nexport function layoutTextIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextBlockLayout {\n const intrinsic = layoutWithLines(\n readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak),\n INTRINSIC_MAX_WIDTH,\n 0,\n );\n if (intrinsic.lines.length === 0) {\n return { width: 0, lines: [] };\n }\n return toTextBlockLayout(intrinsic.lines, measureFontShift(ctx));\n}\n\nexport function layoutFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n const shift = measureFontShift(ctx);\n if (clampedMaxWidth === 0) {\n return { width: 0, text: \"\", shift };\n }\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const line = layoutNextLine(prepared, LINE_START_CURSOR, clampedMaxWidth);\n if (line == null) {\n return { width: 0, text: \"\", shift };\n }\n return { width: line.width, text: line.text, shift };\n}\n\nexport function layoutEllipsizedFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n ellipsisPosition: TextEllipsisPosition = \"end\",\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): OverflowTextLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n const firstLine = readPreparedFirstLine(ctx, text, whiteSpace, wordBreak);\n if (firstLine == null) {\n return { width: 0, text: \"\", shift: 0, overflowed: false };\n }\n const shift = measureFontShift(ctx);\n if (clampedMaxWidth === 0) {\n return { width: 0, text: \"\", shift, overflowed: true };\n }\n return layoutPreparedEllipsis(ctx, firstLine.prepared, firstLine.text, clampedMaxWidth, shift, ellipsisPosition);\n}\n\nexport function measureText<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextMeasurement {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n if (clampedMaxWidth === 0) {\n return { width: 0, lineCount: 0 };\n }\n\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n const { maxLineWidth: width, lineCount } = measureLineStats(prepared, clampedMaxWidth);\n return { width, lineCount };\n}\n\nexport function measureTextMinContent<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n overflowWrap: TextOverflowWrapMode = \"break-word\",\n): TextMeasurement {\n const prepared = readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak);\n if (prepared.widths.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n\n const width = measurePreparedMinContentWidth(prepared, overflowWrap);\n const lineMaxWidth = Math.max(width, MIN_CONTENT_WIDTH_EPSILON);\n const { lineCount } = measureLineStats(prepared, lineMaxWidth);\n return { width, lineCount };\n}\n\nexport function layoutText<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n whiteSpace: TextWhiteSpaceMode = \"normal\",\n wordBreak: TextWordBreakMode = \"normal\",\n): TextBlockLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n if (clampedMaxWidth === 0) {\n return { width: 0, lines: [] };\n }\n\n const layout = layoutWithLines(\n readPreparedText(text, ctx.graphics.font, whiteSpace, wordBreak),\n clampedMaxWidth,\n 0,\n );\n if (layout.lines.length === 0) {\n return { width: 0, lines: [] };\n }\n return toTextBlockLayout(layout.lines, measureFontShift(ctx));\n}\n\nexport function layoutTextWithOverflow<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n text: string,\n maxWidth: number,\n options: {\n whiteSpace?: TextWhiteSpaceMode;\n wordBreak?: TextWordBreakMode;\n overflow?: TextOverflowMode;\n maxLines?: number;\n } = {},\n): OverflowTextBlockLayout {\n const clampedMaxWidth = clampMaxWidth(maxWidth);\n const whiteSpace = options.whiteSpace ?? \"normal\";\n const wordBreak = options.wordBreak ?? \"normal\";\n const overflow = options.overflow ?? \"clip\";\n const normalizedMaxLines = normalizeMaxLines(options.maxLines);\n\n const layout = layoutText(ctx, text, clampedMaxWidth, whiteSpace, wordBreak);\n if (normalizedMaxLines == null || layout.lines.length <= normalizedMaxLines) {\n return {\n width: layout.width,\n lines: layout.lines.map((line) => ({ ...line, overflowed: false })),\n overflowed: false,\n };\n }\n\n const visibleLines = layout.lines.slice(0, normalizedMaxLines);\n if (overflow !== \"ellipsis\") {\n return {\n width: visibleLines.reduce((lineWidth, line) => Math.max(lineWidth, line.width), 0),\n lines: visibleLines.map((line) => ({ ...line, overflowed: false })),\n overflowed: true,\n };\n }\n\n const shift = visibleLines[visibleLines.length - 1]?.shift ?? measureFontShift(ctx);\n const lastVisibleLine = visibleLines[visibleLines.length - 1];\n const ellipsizedLastLine = lastVisibleLine == null || lastVisibleLine.text.length === 0\n ? createEllipsisOnlyLayout(ctx, clampedMaxWidth, shift)\n : layoutForcedEllipsizedLine(ctx, lastVisibleLine.text, clampedMaxWidth, shift, whiteSpace, wordBreak);\n\n const lines = [\n ...visibleLines.slice(0, -1).map((line) => ({ ...line, overflowed: false })),\n { ...ellipsizedLastLine, shift },\n ];\n return {\n width: lines.reduce((lineWidth, line) => Math.max(lineWidth, line.width), 0),\n lines,\n overflowed: true,\n };\n}\n","import {\n layoutNextRichInlineLineRange,\n materializeRichInlineLineRange,\n measureRichInlineStats,\n prepareRichInline,\n type PreparedRichInline,\n type RichInlineCursor,\n type RichInlineFragmentRange,\n type RichInlineLineRange,\n} from \"@chenglou/pretext/rich-inline\";\nimport type {\n Context,\n DynValue,\n InlineSpan,\n TextEllipsisPosition,\n TextOverflowMode,\n TextOverflowWrapMode,\n} from \"../types\";\nimport {\n ELLIPSIS_GLYPH,\n INTRINSIC_MAX_WIDTH,\n MIN_CONTENT_WIDTH_EPSILON,\n buildPrefixWidths,\n measureEllipsisWidth,\n measureFontShift,\n normalizeMaxLines,\n readLruValue,\n selectEllipsisUnitCounts,\n writeLruValue,\n} from \"./core\";\nimport { getPreparedUnits, measurePreparedMinContentWidth, readPreparedText } from \"./plain-core\";\n\nconst RICH_PREPARED_CACHE_CAPACITY = 256;\n\nconst LEADING_COLLAPSIBLE_BOUNDARY_RE = /^[ \\t\\n\\f\\r]+/;\nconst TRAILING_COLLAPSIBLE_BOUNDARY_RE = /[ \\t\\n\\f\\r]+$/;\n\ntype RichPreparedState = {\n prepared: PreparedRichInline;\n preparedItemIndexBySourceItemIndex: (number | undefined)[];\n};\n\nconst richPreparedCache = new Map<string, RichPreparedState>();\n\nexport interface RichFragmentLayout {\n itemIndex: number;\n text: string;\n font: string;\n color: DynValue<any, string> | undefined;\n gapBefore: number;\n occupiedWidth: number;\n shift: number;\n}\n\nexport interface RichLineLayout {\n width: number;\n fragments: RichFragmentLayout[];\n overflowed: boolean;\n}\n\nexport interface RichBlockLayout {\n width: number;\n lines: RichLineLayout[];\n overflowed: boolean;\n}\n\nexport interface RichMeasurement {\n width: number;\n lineCount: number;\n}\n\ntype RichUnitLayout = {\n fragmentIndex: number;\n itemIndex: number;\n text: string;\n width: number;\n font: string;\n color: DynValue<any, string> | undefined;\n leadingGap: number;\n};\n\nfunction withFont<C extends CanvasRenderingContext2D, T>(ctx: Context<C>, font: string, cb: () => T): T {\n const previousFont = ctx.graphics.font;\n ctx.graphics.font = font;\n try {\n return cb();\n } finally {\n ctx.graphics.font = previousFont;\n }\n}\n\nfunction getRichPreparedCacheKey<C extends CanvasRenderingContext2D>(\n spans: InlineSpan<C>[],\n defaultFont: string,\n): string {\n return spans\n .map((span) => `${span.font ?? defaultFont}\\u0000${span.text}\\u0000${span.break ?? \"\"}\\u0000${span.extraWidth ?? 0}`)\n .join(\"\\u0001\");\n}\n\nfunction readRichPrepared<C extends CanvasRenderingContext2D>(\n spans: InlineSpan<C>[],\n defaultFont: string,\n): RichPreparedState {\n const key = getRichPreparedCacheKey(spans, defaultFont);\n const cached = readLruValue(richPreparedCache, key);\n if (cached != null) {\n return cached;\n }\n const items = spans.map((span) => ({\n text: span.text,\n font: span.font ?? defaultFont,\n break: span.break,\n extraWidth: span.extraWidth,\n }));\n const preparedItemIndexBySourceItemIndex = buildPreparedItemIndexBySourceItemIndex(spans);\n return writeLruValue(\n richPreparedCache,\n key,\n {\n prepared: prepareRichInline(items),\n preparedItemIndexBySourceItemIndex,\n },\n RICH_PREPARED_CACHE_CAPACITY,\n );\n}\n\nfunction trimRichInlineBoundaryWhitespace(text: string): string {\n return text\n .replace(LEADING_COLLAPSIBLE_BOUNDARY_RE, \"\")\n .replace(TRAILING_COLLAPSIBLE_BOUNDARY_RE, \"\");\n}\n\nfunction buildPreparedItemIndexBySourceItemIndex<C extends CanvasRenderingContext2D>(\n spans: InlineSpan<C>[],\n): (number | undefined)[] {\n const preparedItemIndexBySourceItemIndex: (number | undefined)[] = Array.from({ length: spans.length });\n let preparedItemIndex = 0;\n for (let index = 0; index < spans.length; index += 1) {\n if (trimRichInlineBoundaryWhitespace(spans[index]!.text).length === 0) {\n continue;\n }\n preparedItemIndexBySourceItemIndex[index] = preparedItemIndex;\n preparedItemIndex += 1;\n }\n return preparedItemIndexBySourceItemIndex;\n}\n\nfunction getRichFragmentStartCursor(\n prepared: RichPreparedState,\n fragment: RichInlineFragmentRange,\n): RichInlineCursor | null {\n const itemIndex = prepared.preparedItemIndexBySourceItemIndex[fragment.itemIndex];\n if (itemIndex == null) {\n return null;\n }\n return {\n itemIndex,\n segmentIndex: fragment.start.segmentIndex,\n graphemeIndex: fragment.start.graphemeIndex,\n };\n}\n\nfunction splitOverflowingRichLineRange(\n prepared: RichPreparedState,\n lineRange: RichInlineLineRange,\n maxWidth: number,\n): RichInlineLineRange {\n if (lineRange.width <= maxWidth || lineRange.fragments.length <= 1) {\n return lineRange;\n }\n\n const trailingFragment = lineRange.fragments[lineRange.fragments.length - 1]!;\n const splitCursor = getRichFragmentStartCursor(prepared, trailingFragment);\n if (splitCursor == null) {\n return lineRange;\n }\n\n const fragments = lineRange.fragments.slice(0, -1);\n const width = fragments.reduce((total, fragment) => total + fragment.gapBefore + fragment.occupiedWidth, 0);\n return {\n fragments,\n width,\n end: splitCursor,\n };\n}\n\nfunction layoutNextConstrainedRichInlineLineRange(\n prepared: RichPreparedState,\n maxWidth: number,\n start?: RichInlineCursor,\n): RichInlineLineRange | null {\n const lineRange = layoutNextRichInlineLineRange(prepared.prepared, maxWidth, start);\n if (lineRange == null) {\n return null;\n }\n return splitOverflowingRichLineRange(prepared, lineRange, maxWidth);\n}\n\nfunction walkConstrainedRichInlineLineRanges(\n prepared: RichPreparedState,\n maxWidth: number,\n onLine: (lineRange: RichInlineLineRange) => void,\n): number {\n let lineCount = 0;\n let cursor: RichInlineCursor | undefined;\n while (true) {\n const lineRange = layoutNextConstrainedRichInlineLineRange(prepared, maxWidth, cursor);\n if (lineRange == null) {\n return lineCount;\n }\n onLine(lineRange);\n lineCount += 1;\n cursor = lineRange.end;\n }\n}\n\nfunction measureConstrainedRichInlineStats(\n prepared: RichPreparedState,\n maxWidth: number,\n): { lineCount: number; maxLineWidth: number } {\n let lineCount = 0;\n let maxLineWidth = 0;\n walkConstrainedRichInlineLineRanges(prepared, maxWidth, (lineRange) => {\n lineCount += 1;\n if (lineRange.width > maxLineWidth) {\n maxLineWidth = lineRange.width;\n }\n });\n return {\n lineCount,\n maxLineWidth,\n };\n}\n\nfunction measureRichFragmentShift<C extends CanvasRenderingContext2D>(ctx: Context<C>, font: string): number {\n return withFont(ctx, font, () => measureFontShift(ctx));\n}\n\nfunction materializeRichLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n lineRange: RichInlineLineRange,\n overflowed: boolean,\n): RichLineLayout {\n const prepared = readRichPrepared(spans, defaultFont);\n const richLine = materializeRichInlineLineRange(prepared.prepared, lineRange);\n const fragments: RichFragmentLayout[] = richLine.fragments.map((fragment) => {\n const span = spans[fragment.itemIndex];\n const font = span?.font ?? defaultFont;\n const color = span?.color ?? defaultColor;\n return {\n itemIndex: fragment.itemIndex,\n text: fragment.text,\n font,\n color: color as DynValue<any, string>,\n gapBefore: fragment.gapBefore,\n occupiedWidth: fragment.occupiedWidth,\n shift: measureRichFragmentShift(ctx, font),\n };\n });\n return { width: richLine.width, fragments, overflowed };\n}\n\nfunction flattenRichLineUnits(line: RichLineLayout): RichUnitLayout[] {\n const units: RichUnitLayout[] = [];\n for (let fragmentIndex = 0; fragmentIndex < line.fragments.length; fragmentIndex += 1) {\n const fragment = line.fragments[fragmentIndex]!;\n const prepared = readPreparedText(fragment.text, fragment.font, \"normal\", \"normal\");\n const fragmentUnits = getPreparedUnits(prepared);\n if (fragmentUnits.length === 0) {\n continue;\n }\n\n const textWidth = fragmentUnits.reduce((total, unit) => total + unit.width, 0);\n const trailingExtraWidth = Math.max(0, fragment.occupiedWidth - textWidth);\n\n for (let unitIndex = 0; unitIndex < fragmentUnits.length; unitIndex += 1) {\n const unit = fragmentUnits[unitIndex]!;\n units.push({\n fragmentIndex,\n itemIndex: fragment.itemIndex,\n text: unit.text,\n width: unit.width + (unitIndex === fragmentUnits.length - 1 ? trailingExtraWidth : 0),\n font: fragment.font,\n color: fragment.color,\n leadingGap: unitIndex === 0 ? fragment.gapBefore : 0,\n });\n }\n }\n return units;\n}\n\nfunction buildRichPrefixWidths(units: readonly RichUnitLayout[]): number[] {\n return buildPrefixWidths(units.map((unit) => unit.leadingGap + unit.width));\n}\n\nfunction buildRichSuffixWidths(units: readonly RichUnitLayout[]): number[] {\n const widths = [0];\n let total = 0;\n for (let index = units.length - 1; index >= 0; index -= 1) {\n const unit = units[index]!;\n total += unit.width;\n if (widths.length > 1) {\n total += unit.leadingGap;\n }\n widths.push(total);\n }\n return widths;\n}\n\nfunction materializeRichFragmentsFromUnits(\n units: readonly RichUnitLayout[],\n start: number,\n end: number,\n suppressLeadingGap: boolean,\n): RichFragmentLayout[] {\n const fragments: RichFragmentLayout[] = [];\n for (let index = start; index < end; index += 1) {\n const unit = units[index]!;\n const previous = fragments[fragments.length - 1];\n const previousUnit = units[index - 1];\n if (previous != null && previousUnit != null && previousUnit.fragmentIndex === unit.fragmentIndex) {\n previous.text += unit.text;\n previous.occupiedWidth += unit.width;\n continue;\n }\n\n fragments.push({\n itemIndex: unit.itemIndex,\n text: unit.text,\n font: unit.font,\n color: unit.color,\n gapBefore: fragments.length === 0 && suppressLeadingGap ? 0 : unit.leadingGap,\n occupiedWidth: unit.width,\n shift: 0,\n });\n }\n return fragments;\n}\n\nfunction measureRichFragmentsShift<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n fragments: readonly RichFragmentLayout[],\n): RichFragmentLayout[] {\n return fragments.map((fragment) => ({\n ...fragment,\n shift: measureRichFragmentShift(ctx, fragment.font),\n }));\n}\n\nfunction createRichEllipsisFragment<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n font: string,\n color: DynValue<C, string> | undefined,\n): RichFragmentLayout {\n return withFont(ctx, font, () => ({\n itemIndex: -1,\n text: ELLIPSIS_GLYPH,\n font,\n color,\n gapBefore: 0,\n occupiedWidth: measureEllipsisWidth(ctx),\n shift: measureFontShift(ctx),\n }));\n}\n\nfunction createRichEllipsisOnlyLayout<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n maxWidth: number,\n font: string,\n color: DynValue<C, string> | undefined,\n): RichLineLayout {\n const ellipsis = createRichEllipsisFragment(ctx, font, color);\n if (ellipsis.occupiedWidth > maxWidth) {\n return { width: 0, fragments: [], overflowed: true };\n }\n return { width: ellipsis.occupiedWidth, fragments: [ellipsis], overflowed: true };\n}\n\nfunction layoutPreparedRichEllipsis<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n line: RichLineLayout,\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n position: TextEllipsisPosition,\n): RichLineLayout {\n if (!line.overflowed && line.width <= maxWidth) {\n return { ...line, overflowed: false };\n }\n\n const units = flattenRichLineUnits(line);\n const fallbackFragment = line.fragments[0];\n const fallbackFont = fallbackFragment?.font ?? defaultFont;\n const fallbackColor = (fallbackFragment?.color ?? defaultColor) as DynValue<C, string>;\n const ellipsisOnly = createRichEllipsisOnlyLayout(ctx, maxWidth, fallbackFont, fallbackColor);\n if (ellipsisOnly.fragments.length === 0 || units.length === 0) {\n return ellipsisOnly;\n }\n\n const ellipsisWidth = ellipsisOnly.width;\n const availableWidth = Math.max(0, maxWidth - ellipsisWidth);\n const prefixWidths = buildRichPrefixWidths(units);\n const suffixWidths = buildRichSuffixWidths(units);\n const { prefixCount, suffixCount } = selectEllipsisUnitCounts({\n position,\n prefixWidths,\n suffixWidths,\n unitCount: units.length,\n availableWidth,\n getMaxSuffixCount: position === \"middle\"\n ? (nextPrefixCount) => Math.max(0, units.length - nextPrefixCount - 1)\n : undefined,\n });\n\n const prefixFragments = measureRichFragmentsShift(ctx, materializeRichFragmentsFromUnits(units, 0, prefixCount, false));\n const suffixStartIndex = units.length - suffixCount;\n const suffixFragments = measureRichFragmentsShift(\n ctx,\n materializeRichFragmentsFromUnits(units, suffixStartIndex, units.length, true),\n );\n\n const ellipsisSource =\n position === \"start\"\n ? (suffixFragments[0] ?? line.fragments[0])\n : position === \"middle\"\n ? (prefixFragments[prefixFragments.length - 1] ?? suffixFragments[0] ?? line.fragments[line.fragments.length - 1])\n : (prefixFragments[prefixFragments.length - 1] ?? line.fragments[line.fragments.length - 1]);\n const ellipsis = createRichEllipsisFragment(\n ctx,\n ellipsisSource?.font ?? defaultFont,\n (ellipsisSource?.color ?? defaultColor) as DynValue<C, string>,\n );\n\n const fragments =\n position === \"start\"\n ? [ellipsis, ...suffixFragments]\n : position === \"middle\"\n ? [...prefixFragments, ellipsis, ...suffixFragments]\n : [...prefixFragments, ellipsis];\n const width =\n position === \"start\"\n ? ellipsis.occupiedWidth + (suffixWidths[suffixCount] ?? 0)\n : position === \"middle\"\n ? (prefixWidths[prefixCount] ?? 0) + ellipsis.occupiedWidth + (suffixWidths[suffixCount] ?? 0)\n : (prefixWidths[prefixCount] ?? 0) + ellipsis.occupiedWidth;\n\n return { width, fragments, overflowed: true };\n}\n\nexport function layoutRichFirstLineIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichLineLayout {\n if (spans.length === 0) {\n return { width: 0, fragments: [], overflowed: false };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineRange = layoutNextRichInlineLineRange(prepared.prepared, INTRINSIC_MAX_WIDTH);\n if (lineRange == null) {\n return { width: 0, fragments: [], overflowed: false };\n }\n return materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, false);\n}\n\nexport function layoutRichFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichLineLayout {\n const clampedMaxWidth = Math.max(0, maxWidth);\n if (spans.length === 0 || clampedMaxWidth === 0) {\n return { width: 0, fragments: [], overflowed: false };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineRange = layoutNextConstrainedRichInlineLineRange(prepared, clampedMaxWidth);\n if (lineRange == null) {\n return { width: 0, fragments: [], overflowed: false };\n }\n return materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, false);\n}\n\nexport function layoutRichEllipsizedFirstLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n ellipsisPosition: TextEllipsisPosition = \"end\",\n): RichLineLayout {\n const clampedMaxWidth = Math.max(0, maxWidth);\n const intrinsicLine = layoutRichFirstLineIntrinsic(ctx, spans, defaultFont, defaultColor);\n if (intrinsicLine.fragments.length === 0) {\n return { ...intrinsicLine, overflowed: false };\n }\n if (clampedMaxWidth === 0) {\n return { width: 0, fragments: [], overflowed: true };\n }\n return layoutPreparedRichEllipsis(ctx, intrinsicLine, clampedMaxWidth, defaultFont, defaultColor, ellipsisPosition);\n}\n\nexport function measureRichText<C extends CanvasRenderingContext2D>(\n _ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n): RichMeasurement {\n if (spans.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const { maxLineWidth: width, lineCount } = measureConstrainedRichInlineStats(prepared, maxWidth);\n return { width, lineCount };\n}\n\nexport function measureRichTextIntrinsic<C extends CanvasRenderingContext2D>(\n _ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n): RichMeasurement {\n if (spans.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const { maxLineWidth: width, lineCount } = measureRichInlineStats(prepared.prepared, INTRINSIC_MAX_WIDTH);\n return { width, lineCount };\n}\n\nexport function measureRichTextMinContent<C extends CanvasRenderingContext2D>(\n _ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n overflowWrap: TextOverflowWrapMode = \"break-word\",\n): RichMeasurement {\n if (spans.length === 0) {\n return { width: 0, lineCount: 0 };\n }\n let maxWidth = 0;\n for (const span of spans) {\n if (span.text.trim().length === 0) {\n continue;\n }\n const font = span.font ?? defaultFont;\n const prepared = readPreparedText(span.text, font, \"normal\", \"normal\");\n const spanMinWidth = measurePreparedMinContentWidth(prepared, overflowWrap) + (span.extraWidth ?? 0);\n if (spanMinWidth > maxWidth) {\n maxWidth = spanMinWidth;\n }\n }\n if (maxWidth === 0) {\n return { width: 0, lineCount: 0 };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineMaxWidth = Math.max(maxWidth, MIN_CONTENT_WIDTH_EPSILON);\n const { lineCount } = measureConstrainedRichInlineStats(prepared, lineMaxWidth);\n return { width: maxWidth, lineCount };\n}\n\nexport function layoutRichText<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichBlockLayout {\n if (spans.length === 0) {\n return { width: 0, lines: [], overflowed: false };\n }\n const prepared = readRichPrepared(spans, defaultFont);\n const lineRanges: RichInlineLineRange[] = [];\n walkConstrainedRichInlineLineRanges(prepared, maxWidth, (lineRange) => lineRanges.push(lineRange));\n if (lineRanges.length === 0) {\n return { width: 0, lines: [], overflowed: false };\n }\n const lines = lineRanges.map((lineRange) => materializeRichLine(ctx, spans, defaultFont, defaultColor, lineRange, false));\n const width = lines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0);\n return { width, lines, overflowed: false };\n}\n\nexport function layoutRichTextIntrinsic<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n): RichBlockLayout {\n return layoutRichText(ctx, spans, INTRINSIC_MAX_WIDTH, defaultFont, defaultColor);\n}\n\nexport function layoutRichTextWithOverflow<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n maxWidth: number,\n defaultFont: string,\n defaultColor: DynValue<C, string>,\n maxLines?: number,\n overflow: TextOverflowMode = \"clip\",\n): RichBlockLayout {\n if (spans.length === 0) {\n return { width: 0, lines: [], overflowed: false };\n }\n\n const normalizedMaxLines = normalizeMaxLines(maxLines);\n const layout = layoutRichText(ctx, spans, maxWidth, defaultFont, defaultColor);\n if (normalizedMaxLines == null || layout.lines.length <= normalizedMaxLines) {\n return layout;\n }\n\n const visibleLines = layout.lines.slice(0, normalizedMaxLines);\n if (overflow !== \"ellipsis\") {\n return {\n width: visibleLines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0),\n lines: visibleLines,\n overflowed: true,\n };\n }\n\n const lastVisibleLine = visibleLines[visibleLines.length - 1];\n const ellipsizedLastLine = lastVisibleLine == null\n ? { width: 0, fragments: [], overflowed: true }\n : layoutPreparedRichEllipsis(\n ctx,\n { ...lastVisibleLine, overflowed: true },\n maxWidth,\n defaultFont,\n defaultColor,\n \"end\",\n );\n\n const lines = [...visibleLines.slice(0, -1), ellipsizedLastLine];\n return {\n width: lines.reduce((maxLineWidth, line) => Math.max(maxLineWidth, line.width), 0),\n lines,\n overflowed: true,\n };\n}\n","import {\n layoutEllipsizedFirstLine,\n layoutFirstLine,\n layoutFirstLineIntrinsic,\n layoutRichEllipsizedFirstLine,\n layoutRichFirstLine,\n layoutRichFirstLineIntrinsic,\n layoutRichText,\n layoutRichTextIntrinsic,\n layoutRichTextWithOverflow,\n layoutText,\n layoutTextIntrinsic,\n layoutTextWithOverflow,\n measureRichText,\n measureRichTextIntrinsic,\n measureRichTextMinContent,\n measureText,\n measureTextMinContent,\n measureTextIntrinsic,\n type RichBlockLayout,\n type RichLineLayout,\n type RichMeasurement,\n type TextLayout,\n type TextMeasurement,\n} from \"../text\";\nimport type { Box, Context, InlineSpan, MultilineTextOptions, Node, PhysicalTextAlign, TextOptions } from \"../types\";\n\ntype SingleLineLayout = TextLayout;\ntype MultiLineDrawLayout = {\n width: number;\n lines: TextLayout[];\n};\ntype MultiLineMeasureLayout = TextMeasurement;\n\ntype TextLayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getTextLayout<T>(node: Node<C>, key: string): T | undefined;\n setTextLayout<T>(node: Node<C>, key: string, layout: T): void;\n};\n\nfunction resolvePhysicalTextAlign(\n options: Pick<MultilineTextOptions<any>, \"align\" | \"physicalAlign\">,\n): PhysicalTextAlign {\n if (options.physicalAlign != null) {\n return options.physicalAlign;\n }\n if (options.align != null) {\n switch (options.align) {\n case \"start\":\n return \"left\";\n case \"center\":\n return \"center\";\n case \"end\":\n return \"right\";\n }\n }\n return \"left\";\n}\n\nfunction normalizeTextMaxWidth(maxWidth: number | undefined): number | undefined {\n if (maxWidth == null) {\n return undefined;\n }\n return Math.max(0, maxWidth);\n}\n\nfunction getTextLayoutContext<C extends CanvasRenderingContext2D>(ctx: Context<C>): Context<C> & TextLayoutCacheAccess<C> {\n return ctx as Context<C> & TextLayoutCacheAccess<C>;\n}\n\nfunction readCachedTextLayout<C extends CanvasRenderingContext2D, T>(\n node: Node<C>,\n ctx: Context<C>,\n key: string,\n compute: () => T,\n): T {\n const textCtx = getTextLayoutContext(ctx);\n const cached = textCtx.getTextLayout<T>(node, key);\n if (cached != null) {\n return cached;\n }\n const layout = compute();\n textCtx.setTextLayout(node, key, layout);\n return layout;\n}\n\nfunction getSingleLineLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"single:intrinsic\" : `single:${maxWidth}`;\n}\n\nfunction getMultiLineMeasureLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"multi:measure:intrinsic\" : `multi:measure:${maxWidth}`;\n}\n\nfunction getRichMultiLineMeasureLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"rich:measure:intrinsic\" : `rich:measure:${maxWidth}`;\n}\n\nfunction getMultiLineDrawLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"multi:draw:intrinsic\" : `multi:draw:${maxWidth}`;\n}\n\nfunction getRichMultiLineDrawLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"rich:draw:intrinsic\" : `rich:draw:${maxWidth}`;\n}\n\nfunction getMultiLineOverflowLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"multi:overflow:intrinsic\" : `multi:overflow:${maxWidth}`;\n}\n\nfunction getRichMultiLineOverflowLayoutKey(maxWidth: number | undefined): string {\n return maxWidth == null ? \"rich:overflow:intrinsic\" : `rich:overflow:${maxWidth}`;\n}\n\nfunction shouldUseMultilineOverflowLayout(options: Pick<MultilineTextOptions<any>, \"maxLines\">): boolean {\n return options.maxLines != null;\n}\n\nfunction shouldReadConstrainedOverflowLayout(\n maxWidth: number | undefined,\n options: Pick<MultilineTextOptions<any>, \"maxLines\">,\n): maxWidth is number {\n return maxWidth != null && shouldUseMultilineOverflowLayout(options);\n}\n\nfunction measureBlockLayout<T extends { width: number; lines: ArrayLike<unknown> }>(layout: T): MultiLineMeasureLayout {\n return { width: layout.width, lineCount: layout.lines.length };\n}\n\nfunction getSingleLineMinContentLayoutKey(): string {\n return \"single:min-content\";\n}\n\nfunction getMultiLineMinContentLayoutKey(): string {\n return \"multi:min-content\";\n}\n\nfunction getRichMultiLineMinContentLayoutKey(): string {\n return \"rich:min-content\";\n}\n\nfunction getSingleLineLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: TextOptions<C>,\n): SingleLineLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getSingleLineLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutFirstLineIntrinsic(ctx, text, options.whiteSpace, options.wordBreak)\n : options.overflow === \"ellipsis\"\n ? layoutEllipsizedFirstLine(\n ctx,\n text,\n maxWidth,\n options.ellipsisPosition ?? \"end\",\n options.whiteSpace,\n options.wordBreak,\n )\n : layoutFirstLine(ctx, text, maxWidth, options.whiteSpace, options.wordBreak)\n );\n}\n\nfunction getRichSingleLineLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: TextOptions<C>,\n): RichLineLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getSingleLineLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutRichFirstLineIntrinsic(ctx, spans, options.font, options.color)\n : options.overflow === \"ellipsis\"\n ? layoutRichEllipsizedFirstLine(ctx, spans, maxWidth, options.font, options.color, options.ellipsisPosition ?? \"end\")\n : layoutRichFirstLine(ctx, spans, maxWidth, options.font, options.color)\n );\n}\n\nfunction getMultiLineOverflowLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: MultilineTextOptions<C>,\n): MultiLineDrawLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getMultiLineOverflowLayoutKey(maxWidth), () =>\n layoutTextWithOverflow(ctx, text, maxWidth ?? 0, {\n whiteSpace: options.whiteSpace,\n wordBreak: options.wordBreak,\n overflow: options.overflow,\n maxLines: options.maxLines,\n })\n );\n}\n\nfunction getMultiLineMeasureLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: MultilineTextOptions<C>,\n): MultiLineMeasureLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return measureBlockLayout(getMultiLineOverflowLayout(node, ctx, text, options));\n }\n return readCachedTextLayout(node, ctx, getMultiLineMeasureLayoutKey(maxWidth), () =>\n maxWidth == null\n ? measureTextIntrinsic(ctx, text, options.whiteSpace, options.wordBreak)\n : measureText(ctx, text, maxWidth, options.whiteSpace, options.wordBreak)\n );\n}\n\nfunction getMultiLineDrawLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: MultilineTextOptions<C>,\n): MultiLineDrawLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return getMultiLineOverflowLayout(node, ctx, text, options);\n }\n return readCachedTextLayout(node, ctx, getMultiLineDrawLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutTextIntrinsic(ctx, text, options.whiteSpace, options.wordBreak)\n : layoutText(ctx, text, maxWidth, options.whiteSpace, options.wordBreak)\n );\n}\n\nfunction getSingleLineMinContentLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n options: TextOptions<C>,\n): SingleLineLayout {\n return readCachedTextLayout(node, ctx, getSingleLineMinContentLayoutKey(), () => {\n const measurement = measureTextMinContent(ctx, text, options.whiteSpace, options.wordBreak, options.overflowWrap);\n const { shift } = layoutFirstLineIntrinsic(ctx, text, options.whiteSpace, options.wordBreak);\n return {\n width: measurement.width,\n text,\n shift,\n };\n });\n}\n\nfunction getRichSingleLineMinContentWidth<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: TextOptions<C>,\n): number {\n return readCachedTextLayout(node, ctx, getSingleLineMinContentLayoutKey(), () =>\n measureRichTextMinContent(ctx, spans, options.font, options.overflowWrap).width\n );\n}\n\nfunction drawRichLine<C extends CanvasRenderingContext2D>(\n ctx: Context<C>,\n line: RichLineLayout,\n fallbackColor: TextOptions<C>[\"color\"],\n x: number,\n y: number,\n lineHeight: number,\n): void {\n let cursorX = x;\n for (let fragmentIndex = 0; fragmentIndex < line.fragments.length; fragmentIndex += 1) {\n const fragment = line.fragments[fragmentIndex]!;\n cursorX += fragment.gapBefore;\n ctx.with((g) => {\n g.font = fragment.font;\n g.fillStyle = ctx.resolveDynValue((fragment.color ?? fallbackColor) as typeof fallbackColor);\n g.textAlign = \"left\";\n g.fillText(fragment.text, cursorX, y + (lineHeight + fragment.shift) / 2);\n });\n cursorX += fragment.occupiedWidth;\n }\n}\n\nfunction getMultiLineMinContentLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n text: string,\n whiteSpace: MultilineTextOptions<C>[\"whiteSpace\"],\n wordBreak: MultilineTextOptions<C>[\"wordBreak\"],\n overflowWrap: MultilineTextOptions<C>[\"overflowWrap\"],\n): MultiLineMeasureLayout {\n return readCachedTextLayout(node, ctx, getMultiLineMinContentLayoutKey(), () =>\n measureTextMinContent(ctx, text, whiteSpace, wordBreak, overflowWrap)\n );\n}\n\nfunction getRichMultiLineMeasureLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichMeasurement {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return measureBlockLayout(getRichMultiLineOverflowLayout(node, ctx, spans, options));\n }\n return readCachedTextLayout(node, ctx, getRichMultiLineMeasureLayoutKey(maxWidth), () =>\n maxWidth == null\n ? measureRichTextIntrinsic(ctx, spans, options.font)\n : measureRichText(ctx, spans, maxWidth, options.font)\n );\n}\n\nfunction getRichMultiLineOverflowLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichBlockLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n return readCachedTextLayout(node, ctx, getRichMultiLineOverflowLayoutKey(maxWidth), () =>\n layoutRichTextWithOverflow(ctx, spans, maxWidth ?? 0, options.font, options.color, options.maxLines, options.overflow)\n );\n}\n\nfunction getRichMultiLineDrawLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichBlockLayout {\n const maxWidth = normalizeTextMaxWidth(ctx.constraints?.maxWidth);\n if (shouldReadConstrainedOverflowLayout(maxWidth, options)) {\n return getRichMultiLineOverflowLayout(node, ctx, spans, options);\n }\n return readCachedTextLayout(node, ctx, getRichMultiLineDrawLayoutKey(maxWidth), () =>\n maxWidth == null\n ? layoutRichTextIntrinsic(ctx, spans, options.font, options.color)\n : layoutRichText(ctx, spans, maxWidth, options.font, options.color)\n );\n}\n\nfunction getRichMultiLineMinContentLayout<C extends CanvasRenderingContext2D>(\n node: Node<C>,\n ctx: Context<C>,\n spans: InlineSpan<C>[],\n options: MultilineTextOptions<C>,\n): RichMeasurement {\n return readCachedTextLayout(node, ctx, getRichMultiLineMinContentLayoutKey(), () =>\n measureRichTextMinContent(ctx, spans, options.font, options.overflowWrap)\n );\n}\n\n/**\n * Draws wrapped text using the configured line height and alignment.\n * Accepts either a plain string or an array of `InlineSpan` items for mixed inline styles.\n */\nexport class MultilineText<C extends CanvasRenderingContext2D> implements Node<C> {\n /**\n * @param text Source text to measure and draw. Pass an `InlineSpan[]` for mixed inline styles.\n * @param options Text layout and drawing options.\n */\n constructor(\n readonly text: string | InlineSpan<C>[],\n readonly options: MultilineTextOptions<C>,\n ) {}\n\n measure(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const spans = this.text;\n const { width, lineCount } = getRichMultiLineMeasureLayout(this, ctx, spans, this.options);\n return { width, height: lineCount * this.options.lineHeight };\n }\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width, lineCount } = getMultiLineMeasureLayout(this, ctx, this.text as string, this.options);\n return { width, height: lineCount * this.options.lineHeight };\n });\n }\n\n measureMinContent(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const spans = this.text;\n const { width, lineCount } = getRichMultiLineMinContentLayout(this, ctx, spans, this.options);\n return { width, height: lineCount * this.options.lineHeight };\n }\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width, lineCount } = getMultiLineMinContentLayout(\n this,\n ctx,\n this.text as string,\n this.options.whiteSpace,\n this.options.wordBreak,\n this.options.overflowWrap,\n );\n return { width, height: lineCount * this.options.lineHeight };\n });\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n if (typeof this.text !== \"string\") {\n const spans = this.text;\n const { width, lines } = getRichMultiLineDrawLayout(this, ctx, spans, this.options);\n const align = resolvePhysicalTextAlign(this.options);\n const startX = align === \"right\" ? x + width : align === \"center\" ? x + width / 2 : x;\n for (const line of lines) {\n let cursorX = startX;\n for (let fi = 0; fi < line.fragments.length; fi++) {\n const frag = line.fragments[fi]!;\n cursorX += frag.gapBefore;\n ctx.with((g) => {\n g.font = frag.font;\n g.fillStyle = ctx.resolveDynValue((frag.color ?? this.options.color) as typeof this.options.color);\n if (align === \"right\") {\n g.textAlign = \"right\";\n } else if (align === \"center\") {\n g.textAlign = \"center\";\n } else {\n g.textAlign = \"left\";\n }\n g.fillText(frag.text, cursorX, y + (this.options.lineHeight + frag.shift) / 2);\n });\n cursorX += frag.occupiedWidth;\n }\n y += this.options.lineHeight;\n }\n return false;\n }\n return ctx.with((g) => {\n g.font = this.options.font;\n g.fillStyle = ctx.resolveDynValue(this.options.color);\n const { width, lines } = getMultiLineDrawLayout(this, ctx, this.text as string, this.options);\n switch (resolvePhysicalTextAlign(this.options)) {\n case \"left\":\n for (const { text, shift } of lines) {\n g.fillText(text, x, y + (this.options.lineHeight + shift) / 2);\n y += this.options.lineHeight;\n }\n break;\n case \"right\": {\n x += width;\n g.textAlign = \"right\";\n for (const { text, shift } of lines) {\n g.fillText(text, x, y + (this.options.lineHeight + shift) / 2);\n y += this.options.lineHeight;\n }\n break;\n }\n case \"center\": {\n x += width / 2;\n g.textAlign = \"center\";\n for (const { text, shift } of lines) {\n g.fillText(text, x, y + (this.options.lineHeight + shift) / 2);\n y += this.options.lineHeight;\n }\n break;\n }\n }\n return false;\n });\n }\n\n hittest(_ctx: Context<C>, _test: { x: number; y: number; type: \"click\" | \"auxclick\" | \"hover\" }): boolean {\n return false;\n }\n}\n\n/**\n * Draws a single line of text, clipped logically by measurement width.\n */\nexport class Text<C extends CanvasRenderingContext2D> implements Node<C> {\n /**\n * @param text Source text to measure and draw. Pass an `InlineSpan[]` for mixed inline styles.\n * @param options Text layout and drawing options.\n */\n constructor(\n readonly text: string | InlineSpan<C>[],\n readonly options: TextOptions<C>,\n ) {}\n\n measure(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const { width } = getRichSingleLineLayout(this, ctx, this.text, this.options);\n return { width, height: this.options.lineHeight };\n }\n const text = this.text;\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width } = getSingleLineLayout(this, ctx, text, this.options);\n return { width, height: this.options.lineHeight };\n });\n }\n\n measureMinContent(ctx: Context<C>): Box {\n if (typeof this.text !== \"string\") {\n const width = getRichSingleLineMinContentWidth(this, ctx, this.text, this.options);\n return { width, height: this.options.lineHeight };\n }\n const text = this.text;\n return ctx.with((g) => {\n g.font = this.options.font;\n const { width } = getSingleLineMinContentLayout(this, ctx, text, this.options);\n return { width, height: this.options.lineHeight };\n });\n }\n\n draw(ctx: Context<C>, x: number, y: number): boolean {\n if (typeof this.text !== \"string\") {\n const line = getRichSingleLineLayout(this, ctx, this.text, this.options);\n drawRichLine(ctx, line, this.options.color, x, y, this.options.lineHeight);\n return false;\n }\n const text = this.text;\n return ctx.with((g) => {\n g.font = this.options.font;\n g.fillStyle = ctx.resolveDynValue(this.options.color);\n const layout = getSingleLineLayout(this, ctx, text, this.options);\n g.fillText(layout.text, x, y + (this.options.lineHeight + layout.shift) / 2);\n return false;\n });\n }\n\n hittest(_ctx: Context<C>, _test: { x: number; y: number; type: \"click\" | \"auxclick\" | \"hover\" }): boolean {\n return false;\n }\n}\n","import { forEachNodeAncestor, getNodeRevision } from \"../internal/node-registry\";\nimport type {\n Box,\n Context,\n DynValue,\n FlexLayoutResult,\n HitTest,\n LayoutConstraints,\n Node,\n RendererOptions,\n} from \"../types\";\nimport { shallow } from \"../utils\";\n\nconst MAX_CONSTRAINT_VARIANTS = 8;\n\ntype LayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints): FlexLayoutResult<C> | undefined;\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints): void;\n};\n\ntype TextLayoutCacheAccess<C extends CanvasRenderingContext2D> = {\n getTextLayout<T>(node: Node<C>, key: string): T | undefined;\n setTextLayout<T>(node: Node<C>, key: string, layout: T): void;\n};\n\ntype BoxCacheEntry = {\n revision: number;\n box: Box;\n};\n\ntype LayoutCacheEntry<C extends CanvasRenderingContext2D> = {\n revision: number;\n layout: FlexLayoutResult<C>;\n};\n\ntype TextLayoutCacheEntry = {\n revision: number;\n layout: unknown;\n};\n\ntype RendererContext<C extends CanvasRenderingContext2D> = Context<C> & LayoutCacheAccess<C> & TextLayoutCacheAccess<C>;\n\nfunction constraintKey(constraints: LayoutConstraints | undefined): string {\n if (constraints == null) return \"\";\n return `${constraints.minWidth ?? \"\"},${constraints.maxWidth ?? \"\"},${constraints.minHeight ?? \"\"},${constraints.maxHeight ?? \"\"}`;\n}\n\n/**\n * Base renderer that provides measurement, layout caching, and drawing helpers.\n */\nexport class BaseRenderer<C extends CanvasRenderingContext2D, O extends {} = {}> {\n /** Canvas rendering context used by this renderer. */\n graphics: C;\n #ctx: RendererContext<C>;\n #lastWidth: number;\n #cache = new WeakMap<Node<C>, Map<string, BoxCacheEntry>>();\n #layoutCache = new WeakMap<Node<C>, Map<string, LayoutCacheEntry<C>>>();\n #textLayoutCache = new WeakMap<Node<C>, Map<string, TextLayoutCacheEntry>>();\n\n protected get context(): Context<C> {\n return shallow(this.#ctx);\n }\n\n /**\n * @param graphics Canvas rendering context used for all layout and drawing.\n * @param options Renderer-specific options.\n */\n constructor(\n graphics: C,\n readonly options: RendererOptions & O,\n ) {\n this.graphics = graphics;\n this.graphics.textRendering = \"optimizeLegibility\";\n const self = this;\n this.#ctx = {\n graphics: this.graphics,\n measureNode(node: Node<C>, constraints?: LayoutConstraints) {\n return self.measureNode(node, constraints);\n },\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints) {\n return self.getLayoutResult(node, constraints);\n },\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints) {\n self.setLayoutResult(node, result, constraints);\n },\n getTextLayout<T>(node: Node<C>, key: string) {\n return self.getTextLayout<T>(node, key);\n },\n setTextLayout<T>(node: Node<C>, key: string, layout: T) {\n self.setTextLayout(node, key, layout);\n },\n invalidateNode: this.invalidateNode.bind(this),\n resolveDynValue<T>(value: DynValue<C, T>): T {\n if (typeof value === \"function\") {\n return value(this.graphics);\n }\n return value as T;\n },\n with<T>(cb: (g: C) => T): T {\n this.graphics.save();\n try {\n return cb(this.graphics);\n } finally {\n this.graphics.restore();\n }\n },\n };\n this.#lastWidth = this.graphics.canvas.clientWidth;\n }\n\n #clearAllCaches(): void {\n this.#cache = new WeakMap<Node<C>, Map<string, BoxCacheEntry>>();\n this.#layoutCache = new WeakMap<Node<C>, Map<string, LayoutCacheEntry<C>>>();\n this.#textLayoutCache = new WeakMap<Node<C>, Map<string, TextLayoutCacheEntry>>();\n }\n\n #syncCachesToViewportWidth(): void {\n const width = this.graphics.canvas.clientWidth;\n if (this.#lastWidth === width) {\n return;\n }\n this.#clearAllCaches();\n this.#lastWidth = width;\n }\n\n protected getRootConstraints(): LayoutConstraints {\n return {\n maxWidth: this.graphics.canvas.clientWidth,\n };\n }\n\n protected getRootContext(): Context<C> {\n const ctx = this.context;\n ctx.constraints = this.getRootConstraints();\n return ctx;\n }\n\n protected measureRootNode(node: Node<C>): Box {\n return this.measureNode(node, this.getRootConstraints());\n }\n\n protected drawRootNode(node: Node<C>, x = 0, y = 0): boolean {\n this.measureRootNode(node);\n return node.draw(this.getRootContext(), x, y);\n }\n\n protected hittestRootNode(node: Node<C>, test: HitTest): boolean {\n this.measureRootNode(node);\n return node.hittest(this.getRootContext(), test);\n }\n\n /**\n * Drops cached measurements for a node and every ancestor that depends on it.\n */\n invalidateNode(node: Node<C>): void {\n this.#syncCachesToViewportWidth();\n this.#cache.delete(node);\n this.#layoutCache.delete(node);\n this.#textLayoutCache.delete(node);\n forEachNodeAncestor(node, (ancestor) => {\n this.#cache.delete(ancestor);\n this.#layoutCache.delete(ancestor);\n this.#textLayoutCache.delete(ancestor);\n });\n }\n\n /**\n * Returns the cached layout result for a node under the given constraints, if available.\n */\n getLayoutResult(node: Node<C>, constraints?: LayoutConstraints): FlexLayoutResult<C> | undefined {\n this.#syncCachesToViewportWidth();\n const nodeCache = this.#layoutCache.get(node);\n if (nodeCache == null) {\n return undefined;\n }\n const key = constraintKey(constraints);\n const cached = nodeCache.get(key);\n if (cached == null) {\n return undefined;\n }\n if (cached.revision !== getNodeRevision(node)) {\n nodeCache.delete(key);\n return undefined;\n }\n return cached.layout;\n }\n\n /**\n * Stores a layout result for later draw and hit-test passes.\n */\n setLayoutResult(node: Node<C>, result: FlexLayoutResult<C>, constraints?: LayoutConstraints): void {\n this.#syncCachesToViewportWidth();\n let nodeCache = this.#layoutCache.get(node);\n if (nodeCache == null) {\n nodeCache = new Map();\n this.#layoutCache.set(node, nodeCache);\n } else if (nodeCache.size >= MAX_CONSTRAINT_VARIANTS) {\n const firstKey = nodeCache.keys().next().value!;\n nodeCache.delete(firstKey);\n }\n nodeCache.set(constraintKey(constraints), {\n revision: getNodeRevision(node),\n layout: result,\n });\n }\n\n protected getTextLayout<T>(node: Node<C>, key: string): T | undefined {\n this.#syncCachesToViewportWidth();\n const nodeCache = this.#textLayoutCache.get(node);\n if (nodeCache == null) {\n return undefined;\n }\n const cached = nodeCache.get(key);\n if (cached == null) {\n return undefined;\n }\n if (cached.revision !== getNodeRevision(node)) {\n nodeCache.delete(key);\n return undefined;\n }\n return cached.layout as T;\n }\n\n protected setTextLayout<T>(node: Node<C>, key: string, layout: T): void {\n this.#syncCachesToViewportWidth();\n let nodeCache = this.#textLayoutCache.get(node);\n if (nodeCache == null) {\n nodeCache = new Map();\n this.#textLayoutCache.set(node, nodeCache);\n } else if (nodeCache.size >= MAX_CONSTRAINT_VARIANTS) {\n const firstKey = nodeCache.keys().next().value!;\n nodeCache.delete(firstKey);\n }\n nodeCache.set(key, {\n revision: getNodeRevision(node),\n layout,\n });\n }\n\n /**\n * Measures a node under optional constraints, using cached results when possible.\n */\n measureNode(node: Node<C>, constraints?: LayoutConstraints): Box {\n this.#syncCachesToViewportWidth();\n {\n const nodeCache = this.#cache.get(node);\n if (nodeCache != null) {\n const key = constraintKey(constraints);\n const cached = nodeCache.get(key);\n if (cached != null) {\n if (cached.revision === getNodeRevision(node)) {\n return cached.box;\n }\n nodeCache.delete(key);\n }\n }\n }\n const ctx = this.context;\n if (constraints != null) {\n ctx.constraints = constraints;\n }\n const result = node.measure(ctx);\n const key = constraintKey(constraints);\n let nodeCache = this.#cache.get(node);\n if (nodeCache == null) {\n nodeCache = new Map();\n this.#cache.set(node, nodeCache);\n } else if (nodeCache.size >= MAX_CONSTRAINT_VARIANTS) {\n const firstKey = nodeCache.keys().next().value!;\n nodeCache.delete(firstKey);\n }\n nodeCache.set(key, {\n revision: getNodeRevision(node),\n box: result,\n });\n return result;\n }\n}\n\n/**\n * Immediate-mode renderer for a single root node.\n */\nexport class DebugRenderer<C extends CanvasRenderingContext2D> extends BaseRenderer<C> {\n /**\n * Clears the viewport and draws the provided root node.\n */\n draw(node: Node<C>): boolean {\n const { clientWidth: viewportWidth, clientHeight: viewportHeight } = this.graphics.canvas;\n this.graphics.clearRect(0, 0, viewportWidth, viewportHeight);\n return this.drawRootNode(node);\n }\n\n /**\n * Hit-tests the provided root node using viewport-relative coordinates.\n */\n hittest(node: Node<C>, test: HitTest): boolean {\n return this.hittestRootNode(node, test);\n }\n}\n","/**\n * Mutable list state shared with virtualized renderers.\n */\nexport interface ReplaceListItemAnimationOptions {\n /** Animation duration in milliseconds. */\n duration?: number;\n}\n\ntype ListReplaceChange<T extends {}> = {\n type: \"replace\";\n index: number;\n prevItem: T;\n nextItem: T;\n animation: ReplaceListItemAnimationOptions | undefined;\n};\n\ntype ListUnshiftChange = {\n type: \"unshift\";\n count: number;\n};\n\ntype ListPushChange = {\n type: \"push\";\n count: number;\n};\n\ntype ListResetChange = {\n type: \"reset\";\n};\n\ntype ListSetChange = {\n type: \"set\";\n};\n\nexport type ListStateChange<T extends {}> =\n | ListReplaceChange<T>\n | ListUnshiftChange\n | ListPushChange\n | ListResetChange\n | ListSetChange;\n\nexport type ListStateChangeListener<T extends {}> = (change: ListStateChange<T>) => void;\n\nconst listStateListeners = new WeakMap<ListState<{}>, Set<ListStateChangeListener<{}>>>();\n\nfunction emitListStateChange<T extends {}>(list: ListState<T>, change: ListStateChange<T>): void {\n const listeners = listStateListeners.get(list as unknown as ListState<{}>);\n if (listeners == null || listeners.size === 0) {\n return;\n }\n for (const listener of [...listeners]) {\n (listener as ListStateChangeListener<T>)(change);\n }\n}\n\nexport function subscribeListState<T extends {}>(list: ListState<T>, listener: ListStateChangeListener<T>): () => void {\n const key = list as unknown as ListState<{}>;\n let listeners = listStateListeners.get(key);\n if (listeners == null) {\n listeners = new Set();\n listStateListeners.set(key, listeners);\n }\n listeners.add(listener as ListStateChangeListener<{}>);\n return () => {\n const current = listStateListeners.get(key);\n if (current == null) {\n return;\n }\n current.delete(listener as ListStateChangeListener<{}>);\n if (current.size === 0) {\n listStateListeners.delete(key);\n }\n };\n}\n\nexport class ListState<T extends {}> {\n #items: T[];\n\n /** Pixel offset from the anchored item edge. */\n offset = 0;\n /** Anchor item index, or `undefined` to use the renderer default. */\n position: number | undefined;\n\n /** Items currently managed by the renderer. */\n get items(): T[] {\n return this.#items;\n }\n\n /** Replaces the full item collection while preserving scroll state. */\n set items(value: T[]) {\n this.#items = [...value];\n emitListStateChange(this, { type: \"set\" });\n }\n\n /**\n * @param items Initial list items.\n */\n constructor(items: T[] = []) {\n this.#items = [...items];\n }\n\n /** Prepends one or more items. */\n unshift(...items: T[]): void {\n this.unshiftAll(items);\n }\n\n /** Prepends an array of items. */\n unshiftAll(items: T[]): void {\n if (items.length === 0) {\n return;\n }\n if (this.position != null) {\n this.position += items.length;\n }\n this.#items = items.concat(this.#items);\n emitListStateChange(this, {\n type: \"unshift\",\n count: items.length,\n });\n }\n\n /** Appends one or more items. */\n push(...items: T[]): void {\n this.pushAll(items);\n }\n\n /** Appends an array of items. */\n pushAll(items: T[]): void {\n if (items.length === 0) {\n return;\n }\n this.#items.push(...items);\n emitListStateChange(this, {\n type: \"push\",\n count: items.length,\n });\n }\n\n /**\n * Replaces an existing item by index.\n */\n replace(index: number, item: T, animation?: ReplaceListItemAnimationOptions): void {\n const normalizedIndex = Number.isFinite(index) ? Math.trunc(index) : Number.NaN;\n if (!Number.isInteger(normalizedIndex) || normalizedIndex < 0 || normalizedIndex >= this.#items.length) {\n throw new RangeError(`replace() index ${index} is out of range for list length ${this.#items.length}.`);\n }\n\n const prevItem = this.#items[normalizedIndex]!;\n this.#items[normalizedIndex] = item;\n emitListStateChange(this, {\n type: \"replace\",\n index: normalizedIndex,\n prevItem,\n nextItem: item,\n animation:\n animation != null && Number.isFinite(animation.duration)\n ? { duration: animation.duration }\n : animation == null\n ? undefined\n : {},\n });\n }\n\n /**\n * Sets the current anchor item and pixel offset.\n */\n setAnchor(position: number, offset = 0): void {\n this.position = Number.isFinite(position) ? Math.trunc(position) : undefined;\n this.offset = Number.isFinite(offset) ? offset : 0;\n }\n\n /**\n * Replaces all items and clears scroll state.\n */\n reset(items: T[] = []): void {\n this.#items = [...items];\n this.offset = 0;\n this.position = undefined;\n emitListStateChange(this, { type: \"reset\" });\n }\n\n /** Clears the current scroll anchor while keeping the items. */\n resetScroll(): void {\n this.offset = 0;\n this.position = undefined;\n }\n\n /** Applies a relative pixel scroll delta. */\n applyScroll(delta: number): void {\n this.offset += delta;\n }\n}\n","import type { Node } from \"../types\";\n\nfunction isWeakMapKey(value: unknown): value is object {\n return (typeof value === \"object\" && value !== null) || typeof value === \"function\";\n}\n\n/**\n * Memoizes `renderItem` by object identity.\n */\nexport function memoRenderItem<C extends CanvasRenderingContext2D, T extends object>(\n renderItem: (item: T) => Node<C>,\n): ((item: T) => Node<C>) & { reset: (key: T) => boolean } {\n const cache = new WeakMap<object, Node<C>>();\n\n function fn(item: T): Node<C> {\n if (!isWeakMapKey(item)) {\n throw new TypeError(\"memoRenderItem() only supports object items. Use memoRenderItemBy() for primitive keys.\");\n }\n const key = item as unknown as object;\n const cached = cache.get(key);\n if (cached != null) {\n return cached;\n }\n const result = renderItem(item);\n cache.set(key, result);\n return result;\n }\n\n return Object.assign(fn, {\n reset: (key: T) => cache.delete(key as unknown as object),\n });\n}\n\n/**\n * Memoizes `renderItem` by a caller-provided cache key.\n */\nexport function memoRenderItemBy<C extends CanvasRenderingContext2D, T, K>(\n keyOf: (item: T) => K,\n renderItem: (item: T) => Node<C>,\n): ((item: T) => Node<C>) & { reset: (item: T) => boolean; resetKey: (key: K) => boolean } {\n const cache = new Map<K, Node<C>>();\n\n function fn(item: T): Node<C> {\n const key = keyOf(item);\n const cached = cache.get(key);\n if (cached != null) {\n return cached;\n }\n const result = renderItem(item);\n cache.set(key, result);\n return result;\n }\n\n return Object.assign(fn, {\n reset: (item: T) => cache.delete(keyOf(item)),\n resetKey: (key: K) => cache.delete(key),\n });\n}\n","import type { HitTest, Node, RenderFeedback } from \"../../types\";\nimport { BaseRenderer } from \"../base\";\nimport {\n ListState,\n subscribeListState,\n type ListStateChange,\n} from \"../list-state\";\nimport type { NormalizedListState, VisibleListState, VisibleWindow } from \"./solver\";\n\n/**\n * Options for programmatic scrolling to a target item.\n */\nexport interface JumpToOptions {\n /** Whether to animate the jump. Defaults to `true`. */\n animated?: boolean;\n /** Which edge of the item should align with the viewport. */\n block?: \"start\" | \"center\" | \"end\";\n /** Animation duration in milliseconds. */\n duration?: number;\n /** Called after the jump completes or finishes animating. */\n onComplete?: () => void;\n}\n\ntype ControlledState = {\n position?: number;\n offset: number;\n};\n\ntype JumpAnimation = {\n startAnchor: number;\n targetAnchor: number;\n startTime: number;\n duration: number;\n needsMoreFrames: boolean;\n onComplete: (() => void) | undefined;\n};\n\ntype ReplacementLayer<C extends CanvasRenderingContext2D> = {\n key: number;\n node: Node<C>;\n fromAlpha: number;\n toAlpha: number;\n startTime: number;\n duration: number;\n};\n\ntype ReplacementAnimation<C extends CanvasRenderingContext2D> = {\n currentLayerKey: number;\n layers: ReplacementLayer<C>[];\n fromHeight: number;\n toHeight: number;\n startTime: number;\n duration: number;\n};\n\ntype VirtualizedResolvedItem<C extends CanvasRenderingContext2D> = {\n draw: (y: number) => boolean;\n hittest: (test: HitTest, y: number) => boolean;\n};\n\nconst ALPHA_EPSILON = 1e-3;\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\nfunction sameState(state: ControlledState, position: number | undefined, offset: number): boolean {\n return Object.is(state.position, position) && Object.is(state.offset, offset);\n}\n\nfunction smoothstep(value: number): number {\n return value * value * (3 - 2 * value);\n}\n\nfunction getProgress(startTime: number, duration: number, now: number): number {\n if (!(duration > 0)) {\n return 1;\n }\n return clamp((now - startTime) / duration, 0, 1);\n}\n\nfunction interpolate(from: number, to: number, startTime: number, duration: number, now: number): number {\n const progress = getProgress(startTime, duration, now);\n const eased = progress >= 1 ? 1 : smoothstep(progress);\n return from + (to - from) * eased;\n}\n\nfunction getNow(): number {\n return globalThis.performance?.now() ?? Date.now();\n}\n\n/**\n * Shared base class for virtualized list renderers.\n */\nexport abstract class VirtualizedRenderer<C extends CanvasRenderingContext2D, T extends {}> extends BaseRenderer<\n C,\n {\n renderItem: (item: T) => Node<C>;\n list: ListState<T>;\n }\n> {\n static readonly MIN_JUMP_DURATION = 160;\n static readonly MAX_JUMP_DURATION = 420;\n static readonly JUMP_DURATION_PER_ITEM = 28;\n\n #controlledState: ControlledState | undefined;\n #jumpAnimation: JumpAnimation | undefined;\n #replacementAnimations = new Map<number, ReplacementAnimation<C>>();\n #nextReplacementLayerKey = 0;\n #unsubscribeListState: () => void;\n\n constructor(\n graphics: C,\n options: {\n renderItem: (item: T) => Node<C>;\n list: ListState<T>;\n },\n ) {\n super(graphics, options);\n this.#unsubscribeListState = subscribeListState(options.list, (change) => {\n this.#handleListStateChange(change);\n });\n }\n\n /** Current anchor item index. */\n get position(): number | undefined {\n return this.options.list.position;\n }\n\n /** Updates the current anchor item index. */\n set position(value: number | undefined) {\n this.options.list.position = value;\n }\n\n /** Pixel offset from the anchored item edge. */\n get offset(): number {\n return this.options.list.offset;\n }\n\n /** Updates the pixel offset from the anchored item edge. */\n set offset(value: number) {\n this.options.list.offset = value;\n }\n\n /** Items currently available to the renderer. */\n get items(): T[] {\n return this.options.list.items;\n }\n\n /** Replaces the current item collection. */\n set items(value: T[]) {\n this.options.list.items = value;\n }\n\n /** Renders the current visible window. */\n abstract render(feedback?: RenderFeedback): boolean;\n /** Hit-tests the current visible window. */\n abstract hittest(test: { x: number; y: number; type: \"click\" | \"auxclick\" | \"hover\" }): boolean;\n\n protected _readListState(): VisibleListState {\n return {\n position: this.position,\n offset: this.offset,\n };\n }\n\n protected _commitListState(state: NormalizedListState): void {\n this.position = state.position;\n this.offset = state.offset;\n }\n\n /**\n * Scrolls the viewport to the requested item index.\n */\n jumpTo(index: number, options: JumpToOptions = {}): void {\n if (this.items.length === 0) {\n this.#cancelJumpAnimation();\n return;\n }\n\n const targetIndex = this._clampItemIndex(index);\n const currentState = this._normalizeListState(this._readListState());\n const targetBlock = options.block ?? this._getDefaultJumpBlock();\n const targetAnchor = this._getTargetAnchor(targetIndex, targetBlock);\n\n const animated = options.animated ?? true;\n if (!animated) {\n this.#cancelJumpAnimation();\n this._applyAnchor(targetAnchor);\n options.onComplete?.();\n return;\n }\n\n const startAnchor = this._readAnchor(currentState);\n if (!Number.isFinite(startAnchor)) {\n this.#cancelJumpAnimation();\n this._applyAnchor(targetAnchor);\n options.onComplete?.();\n return;\n }\n\n const duration = clamp(\n options.duration ??\n VirtualizedRenderer.MIN_JUMP_DURATION +\n Math.abs(targetAnchor - startAnchor) * VirtualizedRenderer.JUMP_DURATION_PER_ITEM,\n 0,\n VirtualizedRenderer.MAX_JUMP_DURATION,\n );\n\n if (duration <= 0 || Math.abs(targetAnchor - startAnchor) <= Number.EPSILON) {\n this.#cancelJumpAnimation();\n this._applyAnchor(targetAnchor);\n options.onComplete?.();\n return;\n }\n\n this.#jumpAnimation = {\n startAnchor,\n targetAnchor,\n startTime: getNow(),\n duration,\n needsMoreFrames: true,\n onComplete: options.onComplete,\n };\n this.#controlledState = this._readListState();\n }\n\n protected _resetRenderFeedback(feedback?: RenderFeedback): void {\n if (feedback == null) {\n return;\n }\n feedback.minIdx = Number.NaN;\n feedback.maxIdx = Number.NaN;\n feedback.min = Number.NaN;\n feedback.max = Number.NaN;\n }\n\n protected _accumulateRenderFeedback(feedback: RenderFeedback, idx: number, top: number, height: number): void {\n if (!Number.isFinite(top) || !Number.isFinite(height) || height <= 0) {\n return;\n }\n\n const viewportHeight = this.graphics.canvas.clientHeight;\n const visibleTop = clamp(-top, 0, height);\n const visibleBottom = clamp(viewportHeight - top, 0, height);\n if (visibleBottom <= visibleTop) {\n return;\n }\n\n const itemMin = idx + visibleTop / height;\n const itemMax = idx + visibleBottom / height;\n feedback.minIdx = Number.isNaN(feedback.minIdx) ? idx : Math.min(idx, feedback.minIdx);\n feedback.maxIdx = Number.isNaN(feedback.maxIdx) ? idx : Math.max(idx, feedback.maxIdx);\n feedback.min = Number.isNaN(feedback.min) ? itemMin : Math.min(itemMin, feedback.min);\n feedback.max = Number.isNaN(feedback.max) ? itemMax : Math.max(itemMax, feedback.max);\n }\n\n protected _renderDrawList(\n list: VisibleWindow<VirtualizedResolvedItem<C>>[\"drawList\"],\n shift: number,\n feedback?: RenderFeedback,\n ): boolean {\n let result = false;\n const viewportHeight = this.graphics.canvas.clientHeight;\n\n for (const { idx, value: item, offset, height } of list) {\n const y = offset + shift;\n if (feedback != null) {\n this._accumulateRenderFeedback(feedback, idx, y, height);\n }\n if (y + height < 0 || y > viewportHeight) {\n continue;\n }\n if (item.draw(y)) {\n result = true;\n }\n }\n\n return result;\n }\n\n protected _renderVisibleWindow(window: VisibleWindow<VirtualizedResolvedItem<C>>, feedback?: RenderFeedback): boolean {\n this._resetRenderFeedback(feedback);\n return this._renderDrawList(window.drawList, window.shift, feedback);\n }\n\n protected _hittestVisibleWindow(window: VisibleWindow<VirtualizedResolvedItem<C>>, test: HitTest): boolean {\n for (const { value: item, offset, height } of window.drawList) {\n const y = offset + window.shift;\n if (test.y < y || test.y >= y + height) {\n continue;\n }\n return item.hittest(test, y);\n }\n return false;\n }\n\n protected _prepareRender(): boolean {\n const now = getNow();\n const keepReplacing = this.#prepareReplacementAnimations(now);\n const animation = this.#jumpAnimation;\n if (animation == null) {\n return keepReplacing;\n }\n if (this.items.length === 0) {\n this.#cancelJumpAnimation();\n return keepReplacing;\n }\n if (this.#controlledState != null && !sameState(this.#controlledState, this.position, this.offset)) {\n this.#cancelJumpAnimation();\n return keepReplacing;\n }\n\n const anchor = interpolate(animation.startAnchor, animation.targetAnchor, animation.startTime, animation.duration, now);\n const progress = getProgress(animation.startTime, animation.duration, now);\n this._applyAnchor(anchor);\n animation.needsMoreFrames = progress < 1;\n return keepReplacing || animation.needsMoreFrames;\n }\n\n protected _finishRender(requestRedraw: boolean): boolean {\n const animation = this.#jumpAnimation;\n if (animation == null) {\n return requestRedraw;\n }\n\n if (animation.needsMoreFrames) {\n this.#controlledState = this._readListState();\n return true;\n }\n\n const onComplete = animation.onComplete;\n this.#cancelJumpAnimation();\n onComplete?.();\n return requestRedraw || this.#jumpAnimation != null;\n }\n\n protected _clampItemIndex(index: number): number {\n return clamp(Number.isFinite(index) ? Math.trunc(index) : 0, 0, this.items.length - 1);\n }\n\n protected _getItemHeight(index: number): number {\n const now = getNow();\n const replacement = this.#readReplacementAnimation(index, now);\n if (replacement != null) {\n return this.#sampleReplacementHeight(replacement, now);\n }\n const item = this.items[index];\n const node = this.options.renderItem(item);\n return this.measureRootNode(node).height;\n }\n\n protected _resolveItem(item: T, index: number, now: number): { value: VirtualizedResolvedItem<C>; height: number } {\n const replacement = this.#readReplacementAnimation(index, now);\n if (replacement == null) {\n const node = this.options.renderItem(item);\n return {\n value: {\n draw: (y) => this.drawRootNode(node, 0, y),\n hittest: (test, y) =>\n node.hittest(this.getRootContext(), {\n ...test,\n y: test.y - y,\n }),\n },\n height: this.measureRootNode(node).height,\n };\n }\n\n const slotHeight = this.#sampleReplacementHeight(replacement, now);\n const layers = replacement.layers\n .map((layer) => ({\n alpha: this.#sampleLayerAlpha(layer, now),\n node: layer.node,\n nodeHeight: this.measureRootNode(layer.node).height,\n }))\n .filter((layer) => layer.alpha > ALPHA_EPSILON);\n\n return {\n value: {\n draw: (y) => this.#drawReplacementLayers(layers, slotHeight, y),\n hittest: () => false,\n },\n height: slotHeight,\n };\n }\n\n protected _getAnchorAtOffset(index: number, offset: number): number {\n if (this.items.length === 0) {\n return 0;\n }\n\n let currentIndex = this._clampItemIndex(index);\n let remaining = Number.isFinite(offset) ? offset : 0;\n while (true) {\n if (remaining < 0) {\n if (currentIndex === 0) {\n return 0;\n }\n currentIndex -= 1;\n const height = this._getItemHeight(currentIndex);\n if (height > 0) {\n remaining += height;\n }\n continue;\n }\n\n const height = this._getItemHeight(currentIndex);\n if (height > 0) {\n if (remaining <= height) {\n return currentIndex + remaining / height;\n }\n remaining -= height;\n } else if (remaining === 0) {\n return currentIndex;\n }\n\n if (currentIndex === this.items.length - 1) {\n return this.items.length;\n }\n currentIndex += 1;\n }\n }\n\n protected abstract _normalizeListState(state: VisibleListState): NormalizedListState;\n protected abstract _readAnchor(state: NormalizedListState): number;\n protected abstract _applyAnchor(anchor: number): void;\n protected abstract _getDefaultJumpBlock(): NonNullable<JumpToOptions[\"block\"]>;\n protected abstract _getTargetAnchor(index: number, block: NonNullable<JumpToOptions[\"block\"]>): number;\n protected abstract _getAnimatedLayerOffset(slotHeight: number, nodeHeight: number): number;\n\n #cancelJumpAnimation(): void {\n this.#jumpAnimation = undefined;\n this.#controlledState = undefined;\n }\n\n #createReplacementLayer(node: Node<C>, fromAlpha: number, toAlpha: number, startTime: number, duration: number): ReplacementLayer<C> {\n return {\n key: ++this.#nextReplacementLayerKey,\n node,\n fromAlpha,\n toAlpha,\n startTime,\n duration,\n };\n }\n\n #sampleLayerAlpha(layer: ReplacementLayer<C>, now: number): number {\n return interpolate(layer.fromAlpha, layer.toAlpha, layer.startTime, layer.duration, now);\n }\n\n #sampleReplacementHeight(animation: ReplacementAnimation<C>, now: number): number {\n return interpolate(animation.fromHeight, animation.toHeight, animation.startTime, animation.duration, now);\n }\n\n #isLayerComplete(layer: ReplacementLayer<C>, now: number): boolean {\n return getProgress(layer.startTime, layer.duration, now) >= 1 && Math.abs(layer.toAlpha - this.#sampleLayerAlpha(layer, now)) <= ALPHA_EPSILON;\n }\n\n #readReplacementAnimation(index: number, now: number): ReplacementAnimation<C> | undefined {\n const animation = this.#replacementAnimations.get(index);\n if (animation == null) {\n return undefined;\n }\n\n const currentLayer = animation.layers.find((layer) => layer.key === animation.currentLayerKey);\n if (currentLayer == null) {\n this.#replacementAnimations.delete(index);\n return undefined;\n }\n\n animation.layers = animation.layers.filter((layer) => layer.key === animation.currentLayerKey || !this.#isLayerComplete(layer, now));\n if (\n getProgress(animation.startTime, animation.duration, now) >= 1 &&\n this.#isLayerComplete(currentLayer, now) &&\n animation.layers.length === 1\n ) {\n this.#replacementAnimations.delete(index);\n return undefined;\n }\n\n return animation;\n }\n\n #prepareReplacementAnimations(now: number): boolean {\n let keepAnimating = false;\n for (const index of [...this.#replacementAnimations.keys()]) {\n if (this.#readReplacementAnimation(index, now) != null) {\n keepAnimating = true;\n }\n }\n return keepAnimating;\n }\n\n #drawReplacementLayers(\n layers: { alpha: number; node: Node<C>; nodeHeight: number }[],\n slotHeight: number,\n y: number,\n ): boolean {\n if (slotHeight <= 0) {\n return false;\n }\n\n let result = false;\n const width = this.graphics.canvas.clientWidth;\n for (const layer of layers) {\n const alpha = clamp(layer.alpha, 0, 1);\n if (alpha <= ALPHA_EPSILON) {\n continue;\n }\n\n this.graphics.save();\n try {\n this.graphics.beginPath?.();\n this.graphics.rect?.(0, y, width, slotHeight);\n this.graphics.clip?.();\n if (typeof this.graphics.globalAlpha === \"number\") {\n this.graphics.globalAlpha *= alpha;\n }\n const layerY = y + this._getAnimatedLayerOffset(slotHeight, layer.nodeHeight);\n if (this.drawRootNode(layer.node, 0, layerY)) {\n result = true;\n }\n } finally {\n this.graphics.restore();\n }\n }\n return result;\n }\n\n #handleListStateChange(change: ListStateChange<T>): void {\n switch (change.type) {\n case \"replace\":\n this.#handleReplace(change.index, change.prevItem, change.nextItem, change.animation?.duration);\n break;\n case \"unshift\": {\n if (change.count <= 0 || this.#replacementAnimations.size === 0) {\n return;\n }\n const shifted = new Map<number, ReplacementAnimation<C>>();\n for (const [index, animation] of this.#replacementAnimations) {\n shifted.set(index + change.count, animation);\n }\n this.#replacementAnimations = shifted;\n break;\n }\n case \"push\":\n break;\n case \"reset\":\n case \"set\":\n this.#replacementAnimations.clear();\n break;\n }\n }\n\n #handleReplace(index: number, prevItem: T, nextItem: T, duration: number | undefined): void {\n const normalizedDuration = Number.isFinite(duration) ? Math.max(0, duration!) : 0;\n if (normalizedDuration <= 0) {\n this.#replacementAnimations.delete(index);\n return;\n }\n\n const now = getNow();\n const nextNode = this.options.renderItem(nextItem);\n const nextHeight = this.measureRootNode(nextNode).height;\n const animation = this.#readReplacementAnimation(index, now);\n if (animation == null) {\n const prevNode = this.options.renderItem(prevItem);\n const outgoing = this.#createReplacementLayer(prevNode, 1, 0, now, normalizedDuration);\n const incoming = this.#createReplacementLayer(nextNode, 0, 1, now, normalizedDuration);\n this.#replacementAnimations.set(index, {\n currentLayerKey: incoming.key,\n layers: [outgoing, incoming],\n fromHeight: this.measureRootNode(prevNode).height,\n toHeight: nextHeight,\n startTime: now,\n duration: normalizedDuration,\n });\n return;\n }\n\n const currentLayer = animation.layers.find((layer) => layer.key === animation.currentLayerKey);\n const currentNode = currentLayer?.node ?? this.options.renderItem(prevItem);\n const currentAlpha = currentLayer == null ? 1 : this.#sampleLayerAlpha(currentLayer, now);\n const layers = animation.layers.filter(\n (layer) => layer.key !== animation.currentLayerKey && !this.#isLayerComplete(layer, now),\n );\n if (currentAlpha > ALPHA_EPSILON) {\n layers.push(this.#createReplacementLayer(currentNode, currentAlpha, 0, now, normalizedDuration));\n }\n const incoming = this.#createReplacementLayer(nextNode, 0, 1, now, normalizedDuration);\n layers.push(incoming);\n this.#replacementAnimations.set(index, {\n currentLayerKey: incoming.key,\n layers,\n fromHeight: this.#sampleReplacementHeight(animation, now),\n toHeight: nextHeight,\n startTime: now,\n duration: normalizedDuration,\n });\n }\n}\n","export interface VisibleListState {\n position?: number;\n offset: number;\n}\n\nexport interface NormalizedListState {\n position: number;\n offset: number;\n}\n\nexport interface VisibleWindowEntry<T> {\n idx: number;\n value: T;\n offset: number;\n height: number;\n}\n\nexport interface VisibleWindow<T> {\n drawList: VisibleWindowEntry<T>[];\n shift: number;\n}\n\nexport interface VisibleWindowResult<T> {\n normalizedState: NormalizedListState;\n window: VisibleWindow<T>;\n}\n\ntype ResolvedItem<T> = {\n value: T;\n height: number;\n};\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\nfunction normalizeOffset(offset: number): number {\n return Number.isFinite(offset) ? offset : 0;\n}\n\nexport function normalizeTimelineState(itemCount: number, state: VisibleListState): NormalizedListState {\n if (itemCount <= 0) {\n return { position: 0, offset: 0 };\n }\n\n const position = state.position;\n if (typeof position !== \"number\" || !Number.isFinite(position)) {\n return {\n position: 0,\n offset: normalizeOffset(state.offset),\n };\n }\n\n return {\n position: clamp(Math.trunc(position), 0, itemCount - 1),\n offset: normalizeOffset(state.offset),\n };\n}\n\nexport function normalizeChatState(itemCount: number, state: VisibleListState): NormalizedListState {\n if (itemCount <= 0) {\n return { position: 0, offset: 0 };\n }\n\n const position = state.position;\n if (typeof position !== \"number\" || !Number.isFinite(position)) {\n return {\n position: itemCount - 1,\n offset: normalizeOffset(state.offset),\n };\n }\n\n return {\n position: clamp(Math.trunc(position), 0, itemCount - 1),\n offset: normalizeOffset(state.offset),\n };\n}\n\nexport function resolveTimelineVisibleWindow<T, V>(\n items: readonly T[],\n state: VisibleListState,\n viewportHeight: number,\n resolveItem: (item: T, idx: number) => ResolvedItem<V>,\n): VisibleWindowResult<V> {\n const normalizedState = normalizeTimelineState(items.length, state);\n if (items.length === 0) {\n return {\n normalizedState,\n window: { drawList: [], shift: 0 },\n };\n }\n\n let { position, offset } = normalizedState;\n let drawLength = 0;\n\n if (offset > 0) {\n if (position === 0) {\n offset = 0;\n } else {\n for (let i = position - 1; i >= 0; i -= 1) {\n const { height } = resolveItem(items[i]!, i);\n position = i;\n offset -= height;\n if (offset <= 0) {\n break;\n }\n }\n if (position === 0 && offset > 0) {\n offset = 0;\n }\n }\n }\n\n let y = offset;\n const drawList: VisibleWindowEntry<V>[] = [];\n for (let i = position; i < items.length; i += 1) {\n const { value, height } = resolveItem(items[i]!, i);\n if (y + height > 0) {\n drawList.push({ idx: i, value, offset: y, height });\n drawLength += height;\n } else {\n offset += height;\n position = i + 1;\n }\n y += height;\n if (y >= viewportHeight) {\n break;\n }\n }\n\n let shift = 0;\n if (y < viewportHeight) {\n if (position === 0 && drawLength < viewportHeight) {\n shift = -offset;\n offset = 0;\n } else {\n shift = viewportHeight - y;\n y = (offset += shift);\n let lastIdx = -1;\n for (let i = position - 1; i >= 0; i -= 1) {\n const { value, height } = resolveItem(items[i]!, i);\n drawLength += height;\n y -= height;\n drawList.push({ idx: i, value, offset: y - shift, height });\n lastIdx = i;\n if (y < 0) {\n break;\n }\n }\n if (lastIdx === 0 && drawLength < viewportHeight) {\n shift = drawList.at(-1)?.offset == null ? 0 : -drawList.at(-1)!.offset;\n position = 0;\n offset = 0;\n }\n }\n }\n\n return {\n normalizedState: { position, offset },\n window: { drawList, shift },\n };\n}\n\nexport function resolveChatVisibleWindow<T, V>(\n items: readonly T[],\n state: VisibleListState,\n viewportHeight: number,\n resolveItem: (item: T, idx: number) => ResolvedItem<V>,\n): VisibleWindowResult<V> {\n const normalizedState = normalizeChatState(items.length, state);\n if (items.length === 0) {\n return {\n normalizedState,\n window: { drawList: [], shift: 0 },\n };\n }\n\n let { position, offset } = normalizedState;\n let drawLength = 0;\n\n if (offset < 0) {\n if (position === items.length - 1) {\n offset = 0;\n } else {\n for (let i = position + 1; i < items.length; i += 1) {\n const { height } = resolveItem(items[i]!, i);\n position = i;\n offset += height;\n if (offset > 0) {\n break;\n }\n }\n }\n }\n\n let y = viewportHeight + offset;\n const drawList: VisibleWindowEntry<V>[] = [];\n for (let i = position; i >= 0; i -= 1) {\n const { value, height } = resolveItem(items[i]!, i);\n y -= height;\n if (y <= viewportHeight) {\n drawList.push({ idx: i, value, offset: y, height });\n drawLength += height;\n } else {\n offset -= height;\n position = i - 1;\n }\n if (y < 0) {\n break;\n }\n }\n\n let shift = 0;\n if (y > 0) {\n shift = -y;\n if (drawLength < viewportHeight) {\n y = drawLength;\n for (let i = position + 1; i < items.length; i += 1) {\n const { value, height } = resolveItem(items[i]!, i);\n drawList.push({ idx: i, value, offset: y - shift, height });\n y = (drawLength += height);\n position = i;\n if (y >= viewportHeight) {\n break;\n }\n }\n offset = drawLength < viewportHeight ? 0 : drawLength - viewportHeight;\n } else {\n offset = drawLength - viewportHeight;\n }\n }\n\n return {\n normalizedState: { position, offset },\n window: { drawList, shift },\n };\n}\n","import type { HitTest, RenderFeedback } from \"../../types\";\nimport { VirtualizedRenderer } from \"./base\";\nimport type { JumpToOptions } from \"./base\";\nimport {\n normalizeChatState,\n resolveChatVisibleWindow,\n type NormalizedListState,\n type VisibleListState,\n} from \"./solver\";\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\n/**\n * Virtualized renderer anchored to the bottom, suitable for chat-style UIs.\n */\nexport class ChatRenderer<C extends CanvasRenderingContext2D, T extends {}> extends VirtualizedRenderer<C, T> {\n #resolveVisibleWindow() {\n const now = globalThis.performance?.now() ?? Date.now();\n return resolveChatVisibleWindow(\n this.items,\n this._readListState(),\n this.graphics.canvas.clientHeight,\n (item, idx) => {\n return this._resolveItem(item, idx, now);\n },\n );\n }\n\n protected _getDefaultJumpBlock(): NonNullable<JumpToOptions[\"block\"]> {\n return \"end\";\n }\n\n protected _normalizeListState(state: VisibleListState): NormalizedListState {\n return normalizeChatState(this.items.length, state);\n }\n\n protected _readAnchor(state: NormalizedListState): number {\n if (this.items.length === 0) {\n return 0;\n }\n const height = this._getItemHeight(state.position);\n return height > 0 ? state.position + 1 - state.offset / height : state.position + 1;\n }\n\n protected _applyAnchor(anchor: number): void {\n if (this.items.length === 0) {\n return;\n }\n const clampedAnchor = clamp(anchor, 0, this.items.length);\n const position = clamp(Math.ceil(clampedAnchor) - 1, 0, this.items.length - 1);\n const height = this._getItemHeight(position);\n const offset = height > 0 ? (position + 1 - clampedAnchor) * height : 0;\n this._commitListState({\n position,\n offset: Object.is(offset, -0) ? 0 : offset,\n });\n }\n\n protected _getTargetAnchor(index: number, block: NonNullable<JumpToOptions[\"block\"]>): number {\n const height = this._getItemHeight(index);\n const viewportHeight = this.graphics.canvas.clientHeight;\n\n switch (block) {\n case \"start\":\n return this._getAnchorAtOffset(index, viewportHeight);\n case \"center\":\n return this._getAnchorAtOffset(index, height / 2 + viewportHeight / 2);\n case \"end\":\n return this._getAnchorAtOffset(index, height);\n }\n }\n\n protected _getAnimatedLayerOffset(slotHeight: number, nodeHeight: number): number {\n return slotHeight - nodeHeight;\n }\n\n render(feedback?: RenderFeedback): boolean {\n const keepAnimating = this._prepareRender();\n const { clientWidth: viewportWidth, clientHeight: viewportHeight } = this.graphics.canvas;\n this.graphics.clearRect(0, 0, viewportWidth, viewportHeight);\n const solution = this.#resolveVisibleWindow();\n const requestRedraw = this._renderVisibleWindow(solution.window, feedback);\n this._commitListState(solution.normalizedState);\n return this._finishRender(keepAnimating || requestRedraw);\n }\n\n hittest(test: HitTest): boolean {\n return this._hittestVisibleWindow(this.#resolveVisibleWindow().window, test);\n }\n}\n","import type { HitTest, RenderFeedback } from \"../../types\";\nimport { VirtualizedRenderer } from \"./base\";\nimport type { JumpToOptions } from \"./base\";\nimport {\n normalizeTimelineState,\n resolveTimelineVisibleWindow,\n type NormalizedListState,\n type VisibleListState,\n} from \"./solver\";\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n\n/**\n * Virtualized renderer anchored to the top, suitable for timeline-style UIs.\n */\nexport class TimelineRenderer<C extends CanvasRenderingContext2D, T extends {}> extends VirtualizedRenderer<C, T> {\n #resolveVisibleWindow() {\n const now = globalThis.performance?.now() ?? Date.now();\n return resolveTimelineVisibleWindow(\n this.items,\n this._readListState(),\n this.graphics.canvas.clientHeight,\n (item, idx) => {\n return this._resolveItem(item, idx, now);\n },\n );\n }\n\n protected _getDefaultJumpBlock(): NonNullable<JumpToOptions[\"block\"]> {\n return \"start\";\n }\n\n protected _normalizeListState(state: VisibleListState): NormalizedListState {\n return normalizeTimelineState(this.items.length, state);\n }\n\n protected _readAnchor(state: NormalizedListState): number {\n if (this.items.length === 0) {\n return 0;\n }\n const height = this._getItemHeight(state.position);\n return height > 0 ? state.position - state.offset / height : state.position;\n }\n\n protected _applyAnchor(anchor: number): void {\n if (this.items.length === 0) {\n return;\n }\n const clampedAnchor = clamp(anchor, 0, this.items.length);\n const position = clamp(Math.floor(clampedAnchor), 0, this.items.length - 1);\n const height = this._getItemHeight(position);\n const offset = height > 0 ? -(clampedAnchor - position) * height : 0;\n this._commitListState({\n position,\n offset: Object.is(offset, -0) ? 0 : offset,\n });\n }\n\n protected _getTargetAnchor(index: number, block: NonNullable<JumpToOptions[\"block\"]>): number {\n const height = this._getItemHeight(index);\n const viewportHeight = this.graphics.canvas.clientHeight;\n\n switch (block) {\n case \"start\":\n return this._getAnchorAtOffset(index, 0);\n case \"center\":\n return this._getAnchorAtOffset(index, height / 2 - viewportHeight / 2);\n case \"end\":\n return this._getAnchorAtOffset(index, height - viewportHeight);\n }\n }\n\n protected _getAnimatedLayerOffset(_slotHeight: number, _nodeHeight: number): number {\n return 0;\n }\n\n render(feedback?: RenderFeedback): boolean {\n const keepAnimating = this._prepareRender();\n const { clientWidth: viewportWidth, clientHeight: viewportHeight } = this.graphics.canvas;\n this.graphics.clearRect(0, 0, viewportWidth, viewportHeight);\n const solution = this.#resolveVisibleWindow();\n const requestRedraw = this._renderVisibleWindow(solution.window, feedback);\n this._commitListState(solution.normalizedState);\n return this._finishRender(keepAnimating || requestRedraw);\n }\n\n hittest(test: HitTest): boolean {\n return this._hittestVisibleWindow(this.#resolveVisibleWindow().window, test);\n }\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -8,7 +8,7 @@
8
8
  "repository": {
9
9
  "url": "https://github.com/codehz/chat-layout.git"
10
10
  },
11
- "version": "1.1.0",
11
+ "version": "1.1.1",
12
12
  "main": "./index.mjs",
13
13
  "types": "./index.d.mts",
14
14
  "exports": {