Markdown.tsx
components/Markdown.tsx
236
Lines
28160
Bytes
2
Exports
10
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with ui-flow. It contains 236 lines, 10 detected imports, and 2 detected exports.
Important relationships
Detected exports
MarkdownStreamingMarkdown
Keywords
contenttokentokensmarkdownhighlightchildrentextelementsboundarymarked
Detected imports
react/compiler-runtimemarkedreact../hooks/useSettings.js../ink.js../utils/cliHighlight.js../utils/hash.js../utils/markdown.js../utils/messages.js./MarkdownTable.js
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import { c as _c } from "react/compiler-runtime";
import { marked, type Token, type Tokens } from 'marked';
import React, { Suspense, use, useMemo, useRef } from 'react';
import { useSettings } from '../hooks/useSettings.js';
import { Ansi, Box, useTheme } from '../ink.js';
import { type CliHighlight, getCliHighlightPromise } from '../utils/cliHighlight.js';
import { hashContent } from '../utils/hash.js';
import { configureMarked, formatToken } from '../utils/markdown.js';
import { stripPromptXMLTags } from '../utils/messages.js';
import { MarkdownTable } from './MarkdownTable.js';
type Props = {
children: string;
/** When true, render all text content as dim */
dimColor?: boolean;
};
// Module-level token cache — marked.lexer is the hot cost on virtual-scroll
// remounts (~3ms per message). useMemo doesn't survive unmount→remount, so
// scrolling back to a previously-visible message re-parses. Messages are
// immutable in history; same content → same tokens. Keyed by hash to avoid
// retaining full content strings (turn50→turn99 RSS regression, #24180).
const TOKEN_CACHE_MAX = 500;
const tokenCache = new Map<string, Token[]>();
// Characters that indicate markdown syntax. If none are present, skip the
// ~3ms marked.lexer call entirely — render as a single paragraph. Covers
// the majority of short assistant responses and user prompts that are
// plain sentences. Checked via indexOf (not regex) for speed.
// Single regex: matches any MD marker or ordered-list start (N. at line start).
// One pass instead of 10× includes scans.
const MD_SYNTAX_RE = /[#*`|[>\-_~]|\n\n|^\d+\. |\n\d+\. /;
function hasMarkdownSyntax(s: string): boolean {
// Sample first 500 chars — if markdown exists it's usually early (headers,
// code fence, list). Long tool outputs are mostly plain text tails.
return MD_SYNTAX_RE.test(s.length > 500 ? s.slice(0, 500) : s);
}
function cachedLexer(content: string): Token[] {
// Fast path: plain text with no markdown syntax → single paragraph token.
// Skips marked.lexer's full GFM parse (~3ms on long content). Not cached —
// reconstruction is a single object allocation, and caching would retain
// 4× content in raw/text fields plus the hash key for zero benefit.
if (!hasMarkdownSyntax(content)) {
return [{
type: 'paragraph',
raw: content,
text: content,
tokens: [{
type: 'text',
raw: content,
text: content
}]
} as Token];
}
const key = hashContent(content);
const hit = tokenCache.get(key);
if (hit) {
// Promote to MRU — without this the eviction is FIFO (scrolling back to
// an early message evicts the very item you're looking at).
tokenCache.delete(key);
tokenCache.set(key, hit);
return hit;
}
const tokens = marked.lexer(content);
if (tokenCache.size >= TOKEN_CACHE_MAX) {
// LRU-ish: drop oldest. Map preserves insertion order.
const first = tokenCache.keys().next().value;
if (first !== undefined) tokenCache.delete(first);
}
tokenCache.set(key, tokens);
return tokens;
}
/**
* Renders markdown content using a hybrid approach:
* - Tables are rendered as React components with proper flexbox layout
* - Other content is rendered as ANSI strings via formatToken
*/
export function Markdown(props) {
const $ = _c(4);
const settings = useSettings();
if (settings.syntaxHighlightingDisabled) {
let t0;
if ($[0] !== props) {
t0 = <MarkdownBody {...props} highlight={null} />;
$[0] = props;
$[1] = t0;
} else {
t0 = $[1];
}
return t0;
}
let t0;
if ($[2] !== props) {
t0 = <Suspense fallback={<MarkdownBody {...props} highlight={null} />}><MarkdownWithHighlight {...props} /></Suspense>;
$[2] = props;
$[3] = t0;
} else {
t0 = $[3];
}
return t0;
}
function MarkdownWithHighlight(props) {
const $ = _c(4);
let t0;
if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
t0 = getCliHighlightPromise();
$[0] = t0;
} else {
t0 = $[0];
}
const highlight = use(t0);
let t1;
if ($[1] !== highlight || $[2] !== props) {
t1 = <MarkdownBody {...props} highlight={highlight} />;
$[1] = highlight;
$[2] = props;
$[3] = t1;
} else {
t1 = $[3];
}
return t1;
}
function MarkdownBody(t0) {
const $ = _c(7);
const {
children,
dimColor,
highlight
} = t0;
const [theme] = useTheme();
configureMarked();
let elements;
if ($[0] !== children || $[1] !== dimColor || $[2] !== highlight || $[3] !== theme) {
const tokens = cachedLexer(stripPromptXMLTags(children));
elements = [];
let nonTableContent = "";
const flushNonTableContent = function flushNonTableContent() {
if (nonTableContent) {
elements.push(<Ansi key={elements.length} dimColor={dimColor}>{nonTableContent.trim()}</Ansi>);
nonTableContent = "";
}
};
for (const token of tokens) {
if (token.type === "table") {
flushNonTableContent();
elements.push(<MarkdownTable key={elements.length} token={token as Tokens.Table} highlight={highlight} />);
} else {
nonTableContent = nonTableContent + formatToken(token, theme, 0, null, null, highlight);
nonTableContent;
}
}
flushNonTableContent();
$[0] = children;
$[1] = dimColor;
$[2] = highlight;
$[3] = theme;
$[4] = elements;
} else {
elements = $[4];
}
const elements_0 = elements;
let t1;
if ($[5] !== elements_0) {
t1 = <Box flexDirection="column" gap={1}>{elements_0}</Box>;
$[5] = elements_0;
$[6] = t1;
} else {
t1 = $[6];
}
return t1;
}
type StreamingProps = {
children: string;
};
/**
* Renders markdown during streaming by splitting at the last top-level block
* boundary: everything before is stable (memoized, never re-parsed), only the
* final block is re-parsed per delta. marked.lexer() correctly handles
* unclosed code fences as a single token, so block boundaries are always safe.
*
* The stable boundary only advances (monotonic), so ref mutation during render
* is idempotent and safe under StrictMode double-rendering. Component unmounts
* between turns (streamingText → null), resetting the ref.
*/
export function StreamingMarkdown({
children
}: StreamingProps): React.ReactNode {
// React Compiler: this component reads and writes stablePrefixRef.current
// during render by design. The boundary only advances (monotonic), so
// the ref mutation is idempotent under StrictMode double-render — but the
// compiler can't prove that, and memoizing around the ref reads would
// break the algorithm (stale boundary). Opt out.
'use no memo';
configureMarked();
// Strip before boundary tracking so it matches <Markdown>'s stripping
// (line 29). When a closing tag arrives, stripped(N+1) is not a prefix
// of stripped(N), but the startsWith reset below handles that with a
// one-time re-lex on the smaller stripped string.
const stripped = stripPromptXMLTags(children);
const stablePrefixRef = useRef('');
// Reset if text was replaced (defensive; normally unmount handles this)
if (!stripped.startsWith(stablePrefixRef.current)) {
stablePrefixRef.current = '';
}
// Lex only from current boundary — O(unstable length), not O(full text)
const boundary = stablePrefixRef.current.length;
const tokens = marked.lexer(stripped.substring(boundary));
// Last non-space token is the growing block; everything before is final
let lastContentIdx = tokens.length - 1;
while (lastContentIdx >= 0 && tokens[lastContentIdx]!.type === 'space') {
lastContentIdx--;
}
let advance = 0;
for (let i = 0; i < lastContentIdx; i++) {
advance += tokens[i]!.raw.length;
}
if (advance > 0) {
stablePrefixRef.current = stripped.substring(0, boundary + advance);
}
const stablePrefix = stablePrefixRef.current;
const unstableSuffix = stripped.substring(stablePrefix.length);
// stablePrefix is memoized inside <Markdown> via useMemo([children, ...])
// so it never re-parses as the unstable suffix grows
return <Box flexDirection="column" gap={1}>
{stablePrefix && <Markdown>{stablePrefix}</Markdown>}
{unstableSuffix && <Markdown>{unstableSuffix}</Markdown>}
</Box>;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["marked","Token","Tokens","React","Suspense","use","useMemo","useRef","useSettings","Ansi","Box","useTheme","CliHighlight","getCliHighlightPromise","hashContent","configureMarked","formatToken","stripPromptXMLTags","MarkdownTable","Props","children","dimColor","TOKEN_CACHE_MAX","tokenCache","Map","MD_SYNTAX_RE","hasMarkdownSyntax","s","test","length","slice","cachedLexer","content","type","raw","text","tokens","key","hit","get","delete","set","lexer","size","first","keys","next","value","undefined","Markdown","props","$","_c","settings","syntaxHighlightingDisabled","t0","MarkdownWithHighlight","Symbol","for","highlight","t1","MarkdownBody","theme","elements","nonTableContent","flushNonTableContent","push","trim","token","Table","elements_0","StreamingProps","StreamingMarkdown","ReactNode","stripped","stablePrefixRef","startsWith","current","boundary","substring","lastContentIdx","advance","i","stablePrefix","unstableSuffix"],"sources":["Markdown.tsx"],"sourcesContent":["import { marked, type Token, type Tokens } from 'marked'\nimport React, { Suspense, use, useMemo, useRef } from 'react'\nimport { useSettings } from '../hooks/useSettings.js'\nimport { Ansi, Box, useTheme } from '../ink.js'\nimport {\n  type CliHighlight,\n  getCliHighlightPromise,\n} from '../utils/cliHighlight.js'\nimport { hashContent } from '../utils/hash.js'\nimport { configureMarked, formatToken } from '../utils/markdown.js'\nimport { stripPromptXMLTags } from '../utils/messages.js'\nimport { MarkdownTable } from './MarkdownTable.js'\n\ntype Props = {\n  children: string\n  /** When true, render all text content as dim */\n  dimColor?: boolean\n}\n\n// Module-level token cache — marked.lexer is the hot cost on virtual-scroll\n// remounts (~3ms per message). useMemo doesn't survive unmount→remount, so\n// scrolling back to a previously-visible message re-parses. Messages are\n// immutable in history; same content → same tokens. Keyed by hash to avoid\n// retaining full content strings (turn50→turn99 RSS regression, #24180).\nconst TOKEN_CACHE_MAX = 500\nconst tokenCache = new Map<string, Token[]>()\n\n// Characters that indicate markdown syntax. If none are present, skip the\n// ~3ms marked.lexer call entirely — render as a single paragraph. Covers\n// the majority of short assistant responses and user prompts that are\n// plain sentences. Checked via indexOf (not regex) for speed.\n// Single regex: matches any MD marker or ordered-list start (N. at line start).\n// One pass instead of 10× includes scans.\nconst MD_SYNTAX_RE = /[#*`|[>\\-_~]|\\n\\n|^\\d+\\. |\\n\\d+\\. /\nfunction hasMarkdownSyntax(s: string): boolean {\n  // Sample first 500 chars — if markdown exists it's usually early (headers,\n  // code fence, list). Long tool outputs are mostly plain text tails.\n  return MD_SYNTAX_RE.test(s.length > 500 ? s.slice(0, 500) : s)\n}\n\nfunction cachedLexer(content: string): Token[] {\n  // Fast path: plain text with no markdown syntax → single paragraph token.\n  // Skips marked.lexer's full GFM parse (~3ms on long content). Not cached —\n  // reconstruction is a single object allocation, and caching would retain\n  // 4× content in raw/text fields plus the hash key for zero benefit.\n  if (!hasMarkdownSyntax(content)) {\n    return [\n      {\n        type: 'paragraph',\n        raw: content,\n        text: content,\n        tokens: [{ type: 'text', raw: content, text: content }],\n      } as Token,\n    ]\n  }\n  const key = hashContent(content)\n  const hit = tokenCache.get(key)\n  if (hit) {\n    // Promote to MRU — without this the eviction is FIFO (scrolling back to\n    // an early message evicts the very item you're looking at).\n    tokenCache.delete(key)\n    tokenCache.set(key, hit)\n    return hit\n  }\n  const tokens = marked.lexer(content)\n  if (tokenCache.size >= TOKEN_CACHE_MAX) {\n    // LRU-ish: drop oldest. Map preserves insertion order.\n    const first = tokenCache.keys().next().value\n    if (first !== undefined) tokenCache.delete(first)\n  }\n  tokenCache.set(key, tokens)\n  return tokens\n}\n\n/**\n * Renders markdown content using a hybrid approach:\n * - Tables are rendered as React components with proper flexbox layout\n * - Other content is rendered as ANSI strings via formatToken\n */\nexport function Markdown(props: Props): React.ReactNode {\n  const settings = useSettings()\n  if (settings.syntaxHighlightingDisabled) {\n    return <MarkdownBody {...props} highlight={null} />\n  }\n  // Suspense fallback renders with highlight=null — plain markdown shows\n  // for ~50ms on first ever render while cli-highlight loads.\n  return (\n    <Suspense fallback={<MarkdownBody {...props} highlight={null} />}>\n      <MarkdownWithHighlight {...props} />\n    </Suspense>\n  )\n}\n\nfunction MarkdownWithHighlight(props: Props): React.ReactNode {\n  const highlight = use(getCliHighlightPromise())\n  return <MarkdownBody {...props} highlight={highlight} />\n}\n\nfunction MarkdownBody({\n  children,\n  dimColor,\n  highlight,\n}: Props & { highlight: CliHighlight | null }): React.ReactNode {\n  const [theme] = useTheme()\n  configureMarked()\n\n  const elements = useMemo(() => {\n    const tokens = cachedLexer(stripPromptXMLTags(children))\n    const elements: React.ReactNode[] = []\n    let nonTableContent = ''\n\n    function flushNonTableContent(): void {\n      if (nonTableContent) {\n        elements.push(\n          <Ansi key={elements.length} dimColor={dimColor}>\n            {nonTableContent.trim()}\n          </Ansi>,\n        )\n        nonTableContent = ''\n      }\n    }\n\n    for (const token of tokens) {\n      if (token.type === 'table') {\n        flushNonTableContent()\n        elements.push(\n          <MarkdownTable\n            key={elements.length}\n            token={token as Tokens.Table}\n            highlight={highlight}\n          />,\n        )\n      } else {\n        nonTableContent += formatToken(token, theme, 0, null, null, highlight)\n      }\n    }\n\n    flushNonTableContent()\n    return elements\n  }, [children, dimColor, highlight, theme])\n\n  return (\n    <Box flexDirection=\"column\" gap={1}>\n      {elements}\n    </Box>\n  )\n}\n\ntype StreamingProps = {\n  children: string\n}\n\n/**\n * Renders markdown during streaming by splitting at the last top-level block\n * boundary: everything before is stable (memoized, never re-parsed), only the\n * final block is re-parsed per delta. marked.lexer() correctly handles\n * unclosed code fences as a single token, so block boundaries are always safe.\n *\n * The stable boundary only advances (monotonic), so ref mutation during render\n * is idempotent and safe under StrictMode double-rendering. Component unmounts\n * between turns (streamingText → null), resetting the ref.\n */\nexport function StreamingMarkdown({\n  children,\n}: StreamingProps): React.ReactNode {\n  // React Compiler: this component reads and writes stablePrefixRef.current\n  // during render by design. The boundary only advances (monotonic), so\n  // the ref mutation is idempotent under StrictMode double-render — but the\n  // compiler can't prove that, and memoizing around the ref reads would\n  // break the algorithm (stale boundary). Opt out.\n  'use no memo'\n  configureMarked()\n\n  // Strip before boundary tracking so it matches <Markdown>'s stripping\n  // (line 29). When a closing tag arrives, stripped(N+1) is not a prefix\n  // of stripped(N), but the startsWith reset below handles that with a\n  // one-time re-lex on the smaller stripped string.\n  const stripped = stripPromptXMLTags(children)\n\n  const stablePrefixRef = useRef('')\n\n  // Reset if text was replaced (defensive; normally unmount handles this)\n  if (!stripped.startsWith(stablePrefixRef.current)) {\n    stablePrefixRef.current = ''\n  }\n\n  // Lex only from current boundary — O(unstable length), not O(full text)\n  const boundary = stablePrefixRef.current.length\n  const tokens = marked.lexer(stripped.substring(boundary))\n\n  // Last non-space token is the growing block; everything before is final\n  let lastContentIdx = tokens.length - 1\n  while (lastContentIdx >= 0 && tokens[lastContentIdx]!.type === 'space') {\n    lastContentIdx--\n  }\n  let advance = 0\n  for (let i = 0; i < lastContentIdx; i++) {\n    advance += tokens[i]!.raw.length\n  }\n  if (advance > 0) {\n    stablePrefixRef.current = stripped.substring(0, boundary + advance)\n  }\n\n  const stablePrefix = stablePrefixRef.current\n  const unstableSuffix = stripped.substring(stablePrefix.length)\n\n  // stablePrefix is memoized inside <Markdown> via useMemo([children, ...])\n  // so it never re-parses as the unstable suffix grows\n  return (\n    <Box flexDirection=\"column\" gap={1}>\n      {stablePrefix && <Markdown>{stablePrefix}</Markdown>}\n      {unstableSuffix && <Markdown>{unstableSuffix}</Markdown>}\n    </Box>\n  )\n}\n"],"mappings":"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","ignoreList":[]}