source dump of claude code
at main 116 lines 12 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2import * as React from 'react'; 3import { useLayoutEffect } from 'react'; 4import { PassThrough } from 'stream'; 5import stripAnsi from 'strip-ansi'; 6import { render, useApp } from '../ink.js'; 7 8// This is a workaround for the fact that Ink doesn't support multiple <Static> 9// components in the same render tree. Instead of using a <Static> we just render 10// the component to a string and then print it to stdout 11 12/** 13 * Wrapper component that exits after rendering. 14 * Uses useLayoutEffect to ensure we wait for React's commit phase to complete 15 * before exiting. This is more robust than process.nextTick() for React 19's 16 * async render cycle. 17 */ 18function RenderOnceAndExit(t0) { 19 const $ = _c(5); 20 const { 21 children 22 } = t0; 23 const { 24 exit 25 } = useApp(); 26 let t1; 27 let t2; 28 if ($[0] !== exit) { 29 t1 = () => { 30 const timer = setTimeout(exit, 0); 31 return () => clearTimeout(timer); 32 }; 33 t2 = [exit]; 34 $[0] = exit; 35 $[1] = t1; 36 $[2] = t2; 37 } else { 38 t1 = $[1]; 39 t2 = $[2]; 40 } 41 useLayoutEffect(t1, t2); 42 let t3; 43 if ($[3] !== children) { 44 t3 = <>{children}</>; 45 $[3] = children; 46 $[4] = t3; 47 } else { 48 t3 = $[4]; 49 } 50 return t3; 51} 52 53// DEC synchronized update markers used by terminals 54const SYNC_START = '\x1B[?2026h'; 55const SYNC_END = '\x1B[?2026l'; 56 57/** 58 * Extracts content from the first complete frame in Ink's output. 59 * Ink with non-TTY stdout outputs multiple frames, each wrapped in DEC synchronized 60 * update sequences ([?2026h ... [?2026l). We only want the first frame's content. 61 */ 62function extractFirstFrame(output: string): string { 63 const startIndex = output.indexOf(SYNC_START); 64 if (startIndex === -1) return output; 65 const contentStart = startIndex + SYNC_START.length; 66 const endIndex = output.indexOf(SYNC_END, contentStart); 67 if (endIndex === -1) return output; 68 return output.slice(contentStart, endIndex); 69} 70 71/** 72 * Renders a React node to a string with ANSI escape codes (for terminal output). 73 */ 74export function renderToAnsiString(node: React.ReactNode, columns?: number): Promise<string> { 75 return new Promise(async resolve => { 76 let output = ''; 77 78 // Capture all writes. Set .columns so Ink (ink.tsx:~165) picks up a 79 // chosen width instead of PassThrough's undefined → 80 fallback — 80 // useful for rendering at terminal width for file dumps that should 81 // match what the user sees on screen. 82 const stream = new PassThrough(); 83 if (columns !== undefined) { 84 ; 85 (stream as unknown as { 86 columns: number; 87 }).columns = columns; 88 } 89 stream.on('data', chunk => { 90 output += chunk.toString(); 91 }); 92 93 // Render the component wrapped in RenderOnceAndExit 94 // Non-TTY stdout (PassThrough) gives full-frame output instead of diffs 95 const instance = await render(<RenderOnceAndExit>{node}</RenderOnceAndExit>, { 96 stdout: stream as unknown as NodeJS.WriteStream, 97 patchConsole: false 98 }); 99 100 // Wait for the component to exit naturally 101 await instance.waitUntilExit(); 102 103 // Extract only the first frame's content to avoid duplication 104 // (Ink outputs multiple frames in non-TTY mode) 105 await resolve(extractFirstFrame(output)); 106 }); 107} 108 109/** 110 * Renders a React node to a plain text string (ANSI codes stripped). 111 */ 112export async function renderToString(node: React.ReactNode, columns?: number): Promise<string> { 113 const output = await renderToAnsiString(node, columns); 114 return stripAnsi(output); 115} 116//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useLayoutEffect","PassThrough","stripAnsi","render","useApp","RenderOnceAndExit","t0","$","_c","children","exit","t1","t2","timer","setTimeout","clearTimeout","t3","SYNC_START","SYNC_END","extractFirstFrame","output","startIndex","indexOf","contentStart","length","endIndex","slice","renderToAnsiString","node","ReactNode","columns","Promise","resolve","stream","undefined","on","chunk","toString","instance","stdout","NodeJS","WriteStream","patchConsole","waitUntilExit","renderToString"],"sources":["staticRender.tsx"],"sourcesContent":["import * as React from 'react'\nimport { useLayoutEffect } from 'react'\nimport { PassThrough } from 'stream'\nimport stripAnsi from 'strip-ansi'\nimport { render, useApp } from '../ink.js'\n\n// This is a workaround for the fact that Ink doesn't support multiple <Static>\n// components in the same render tree. Instead of using a <Static> we just render\n// the component to a string and then print it to stdout\n\n/**\n * Wrapper component that exits after rendering.\n * Uses useLayoutEffect to ensure we wait for React's commit phase to complete\n * before exiting. This is more robust than process.nextTick() for React 19's\n * async render cycle.\n */\nfunction RenderOnceAndExit({\n  children,\n}: {\n  children: React.ReactNode\n}): React.ReactNode {\n  const { exit } = useApp()\n\n  // useLayoutEffect runs synchronously after React commits DOM mutations.\n  // setTimeout(0) defers exit to allow Ink to flush output to the stream.\n  useLayoutEffect(() => {\n    const timer = setTimeout(exit, 0)\n    return () => clearTimeout(timer)\n  }, [exit])\n\n  return <>{children}</>\n}\n\n// DEC synchronized update markers used by terminals\nconst SYNC_START = '\\x1B[?2026h'\nconst SYNC_END = '\\x1B[?2026l'\n\n/**\n * Extracts content from the first complete frame in Ink's output.\n * Ink with non-TTY stdout outputs multiple frames, each wrapped in DEC synchronized\n * update sequences ([?2026h ... [?2026l). We only want the first frame's content.\n */\nfunction extractFirstFrame(output: string): string {\n  const startIndex = output.indexOf(SYNC_START)\n  if (startIndex === -1) return output\n\n  const contentStart = startIndex + SYNC_START.length\n  const endIndex = output.indexOf(SYNC_END, contentStart)\n  if (endIndex === -1) return output\n\n  return output.slice(contentStart, endIndex)\n}\n\n/**\n * Renders a React node to a string with ANSI escape codes (for terminal output).\n */\nexport function renderToAnsiString(\n  node: React.ReactNode,\n  columns?: number,\n): Promise<string> {\n  return new Promise(async resolve => {\n    let output = ''\n\n    // Capture all writes. Set .columns so Ink (ink.tsx:~165) picks up a\n    // chosen width instead of PassThrough's undefined → 80 fallback —\n    // useful for rendering at terminal width for file dumps that should\n    // match what the user sees on screen.\n    const stream = new PassThrough()\n    if (columns !== undefined) {\n      ;(stream as unknown as { columns: number }).columns = columns\n    }\n    stream.on('data', chunk => {\n      output += chunk.toString()\n    })\n\n    // Render the component wrapped in RenderOnceAndExit\n    // Non-TTY stdout (PassThrough) gives full-frame output instead of diffs\n    const instance = await render(\n      <RenderOnceAndExit>{node}</RenderOnceAndExit>,\n      {\n        stdout: stream as unknown as NodeJS.WriteStream,\n        patchConsole: false,\n      },\n    )\n\n    // Wait for the component to exit naturally\n    await instance.waitUntilExit()\n\n    // Extract only the first frame's content to avoid duplication\n    // (Ink outputs multiple frames in non-TTY mode)\n    await resolve(extractFirstFrame(output))\n  })\n}\n\n/**\n * Renders a React node to a plain text string (ANSI codes stripped).\n */\nexport async function renderToString(\n  node: React.ReactNode,\n  columns?: number,\n): Promise<string> {\n  const output = await renderToAnsiString(node, columns)\n  return stripAnsi(output)\n}\n"],"mappings":"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","ignoreList":[]}