import { AlternateScreen, Box, NoSelect, ScrollBox, Text } from '@hermes/ink' import { useStore } from '@nanostores/react' import { Fragment, memo, useMemo, useRef } from 'react' import { useGateway } from '../app/gatewayContext.js' import type { AppLayoutProps } from '../app/interfaces.js' import { $isBlocked, $overlayState, patchOverlayState } from '../app/overlayStore.js' import { $uiState } from '../app/uiStore.js' import { INLINE_MODE, SHOW_FPS } from '../config/env.js' import { FULL_RENDER_TAIL_ITEMS } from '../config/limits.js' import { PLACEHOLDER } from '../content/placeholders.js' import { COMPOSER_PROMPT_GAP_WIDTH, composerPromptWidth, inputVisualHeight, stableComposerColumns } from '../lib/inputMetrics.js' import { PerfPane } from '../lib/perfPane.js' import { AgentsOverlay } from './agentsOverlay.js' import { GoodVibesHeart, StatusRule, StickyPromptTracker, TranscriptScrollbar } from './appChrome.js' import { FloatingOverlays, PromptZone } from './appOverlays.js' import { Banner, Panel, SessionPanel } from './branding.js' import { FpsOverlay } from './fpsOverlay.js' import { HelpHint } from './helpHint.js' import { MessageLine } from './messageLine.js' import { QueuedMessages } from './queuedMessages.js' import { LiveTodoPanel, StreamingAssistant } from './streamingAssistant.js' import { TextInput, type TextInputMouseApi } from './textInput.js' const PromptPrefix = memo(function PromptPrefix({ bold = false, color, promptText, width }: { bold?: boolean color: string promptText: string width: number }) { const glyphWidth = Math.max(1, width - COMPOSER_PROMPT_GAP_WIDTH) return ( {promptText} ) }) const TranscriptPane = memo(function TranscriptPane({ actions, composer, progress, transcript }: Pick) { const ui = useStore($uiState) // LiveTodoPanel rides as a child of the latest user-message row so it // visually belongs to the prompt and follows it during scroll. -1 when // empty → row.index === -1 is always false → no render. const lastUserIdx = useMemo(() => { const items = transcript.historyItems for (let i = items.length - 1; i >= 0; i--) { if (items[i].role === 'user') { return i } } return -1 }, [transcript.historyItems]) // Index of the first user-role message; every later user message gets a // small dash above it so multi-turn transcripts visually segment by // turn. -1 when no user message has been sent yet → no separator ever // renders. const firstUserIdx = useMemo( () => transcript.historyItems.findIndex(m => m.role === 'user'), [transcript.historyItems] ) return ( <> { if (e.cellIsBlank) { actions.clearSelection() } }} ref={transcript.scrollRef} stickyScroll > {transcript.virtualHistory.topSpacer > 0 ? : null} {transcript.virtualRows.slice(transcript.virtualHistory.start, transcript.virtualHistory.end).map(row => ( {row.msg.role === 'user' && firstUserIdx >= 0 && row.index > firstUserIdx && ( ─── )} {row.msg.kind === 'intro' ? ( {row.msg.info && } ) : row.msg.kind === 'panel' && row.msg.panelData ? ( ) : ( )} {row.index === lastUserIdx && } ))} {transcript.virtualHistory.bottomSpacer > 0 ? : null} ) }) const ComposerPane = memo(function ComposerPane({ actions, composer, status }: Pick) { const ui = useStore($uiState) const isBlocked = useStore($isBlocked) const sh = (composer.inputBuf[0] ?? composer.input).startsWith('!') const promptText = sh ? '$' : ui.theme.brand.prompt const promptWidth = composerPromptWidth(promptText) const promptBlank = ' '.repeat(promptWidth) const inputColumns = stableComposerColumns(composer.cols, promptWidth) const inputHeight = inputVisualHeight(composer.input, inputColumns) const inputMouseRef = useRef(null) const captureInputDrag = (e: GutterMouseEvent) => { if (e.button !== 0) { return } e.stopImmediatePropagation?.() inputMouseRef.current?.startAtBeginning() } // Drag origin matches the input box's top-left, so localRow / localCol // map directly into TextInput coords (after backing out the prompt cell). const dragFromPromptRow = (e: GutterMouseEvent) => { if (e.button !== 0) { return } e.stopImmediatePropagation?.() inputMouseRef.current?.dragAt(e.localRow ?? 0, (e.localCol ?? 0) - promptWidth) } // Spacer rows live on a different vertical origin; only the column is // parent-aligned with the input. Force row=0 so vertical drags can't // jump the cursor to the wrong wrapped line. const dragFromSpacer = (e: GutterMouseEvent) => { if (e.button !== 0) { return } e.stopImmediatePropagation?.() inputMouseRef.current?.dragAt(0, (e.localCol ?? 0) - promptWidth) } const endInputDrag = () => inputMouseRef.current?.end() return ( { if (e.cellIsBlank) { actions.clearSelection() } }} paddingX={1} > {ui.bgTasks.size > 0 && ( {ui.bgTasks.size} background {ui.bgTasks.size === 1 ? 'task' : 'tasks'} running )} {status.showStickyPrompt ? ( {status.stickyPrompt} ) : ( )} {composer.input === '?' && !composer.inputBuf.length && } {!isBlocked && ( <> {composer.inputBuf.map((line, i) => ( {i === 0 ? ( ) : ( {promptBlank} )} {line || ' '} ))} {sh ? ( ) : composer.inputBuf.length ? ( {promptBlank} ) : ( )} {/* Reserve the transcript scrollbar gutter too so typing never rewraps when the scrollbar column repaints. */} )} {!composer.empty && !ui.sid && ⚕ {ui.status}} ) }) const AgentsOverlayPane = memo(function AgentsOverlayPane() { const { gw } = useGateway() const ui = useStore($uiState) const overlay = useStore($overlayState) return ( patchOverlayState({ agents: false, agentsInitialHistoryIndex: 0 })} t={ui.theme} /> ) }) const StatusRulePane = memo(function StatusRulePane({ at, composer, status }: Pick & { at: 'bottom' | 'top' }) { const ui = useStore($uiState) if (ui.statusBar !== at) { return null } return ( ) }) export const AppLayout = memo(function AppLayout({ actions, composer, mouseTracking, progress, status, transcript }: AppLayoutProps) { const overlay = useStore($overlayState) const ui = useStore($uiState) // Inline mode skips AlternateScreen so the host terminal's native // scrollback captures rows scrolled off the top; composer + progress // stay anchored via normal flex-column flow. const Shell = INLINE_MODE ? Fragment : AlternateScreen const shellProps = INLINE_MODE ? {} : { mouseTracking } return ( {overlay.agents ? ( ) : ( )} {!overlay.agents && ( <> {SHOW_FPS && ( )} )} ) }) type GutterMouseEvent = { button: number localCol?: number localRow?: number stopImmediatePropagation?: () => void }