import { AlternateScreen, Box, NoSelect, ScrollBox, stringWidth, 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 { 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 { MessageLine } from './messageLine.js' import { QueuedMessages } from './queuedMessages.js' import { LiveTodoPanel, StreamingAssistant } from './streamingAssistant.js' import { TextInput, type TextInputMouseApi } from './textInput.js' 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]) 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.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 promptLabel = `${promptText} ` const promptWidth = Math.max(1, stringWidth(promptLabel)) 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} ) : ( )} {!isBlocked && ( <> {composer.inputBuf.map((line, i) => ( {i === 0 ? promptLabel : ' '.repeat(promptWidth)} {line || ' '} ))} {sh ? ( {promptLabel} ) : ( {composer.inputBuf.length ? ' '.repeat(promptWidth) : promptLabel} )} {/* 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 }