import { Box, Text, useInput, useStdout } from '@hermes/ink' import { useEffect, useMemo, useState } from 'react' import { providerDisplayNames } from '../domain/providers.js' import { TUI_SESSION_MODEL_FLAG } from '../domain/slash.js' import type { GatewayClient } from '../gatewayClient.js' import type { ModelOptionProvider, ModelOptionsResponse } from '../gatewayTypes.js' import { asRpcResult, rpcErrorMessage } from '../lib/rpc.js' import type { Theme } from '../theme.js' import { OverlayHint, useOverlayKeys, windowItems, windowOffset } from './overlayControls.js' const VISIBLE = 12 const MIN_WIDTH = 40 const MAX_WIDTH = 90 export function ModelPicker({ gw, onCancel, onSelect, sessionId, t }: ModelPickerProps) { const [providers, setProviders] = useState([]) const [currentModel, setCurrentModel] = useState('') const [err, setErr] = useState('') const [loading, setLoading] = useState(true) const [persistGlobal, setPersistGlobal] = useState(false) const [providerIdx, setProviderIdx] = useState(0) const [modelIdx, setModelIdx] = useState(0) const [stage, setStage] = useState<'model' | 'provider'>('provider') const { stdout } = useStdout() // Pin the picker to a stable width so the FloatBox parent (which shrinks- // to-fit with alignSelf="flex-start") doesn't resize as long provider / // model names scroll into view, and so `wrap="truncate-end"` on each row // has an actual constraint to truncate against. const width = Math.max(MIN_WIDTH, Math.min(MAX_WIDTH, (stdout?.columns ?? 80) - 6)) useEffect(() => { gw.request('model.options', sessionId ? { session_id: sessionId } : {}) .then(raw => { const r = asRpcResult(raw) if (!r) { setErr('invalid response: model.options') setLoading(false) return } const next = r.providers ?? [] setProviders(next) setCurrentModel(String(r.model ?? '')) setProviderIdx( Math.max( 0, next.findIndex(p => p.is_current) ) ) setModelIdx(0) setStage('provider') setErr('') setLoading(false) }) .catch((e: unknown) => { setErr(rpcErrorMessage(e)) setLoading(false) }) }, [gw, sessionId]) const provider = providers[providerIdx] const models = provider?.models ?? [] const names = useMemo(() => providerDisplayNames(providers), [providers]) const back = () => { if (stage === 'model') { setStage('provider') setModelIdx(0) return } onCancel() } useOverlayKeys({ onBack: back, onClose: onCancel }) useInput((ch, key) => { const count = stage === 'provider' ? providers.length : models.length const sel = stage === 'provider' ? providerIdx : modelIdx const setSel = stage === 'provider' ? setProviderIdx : setModelIdx if (key.upArrow && sel > 0) { setSel(v => v - 1) return } if (key.downArrow && sel < count - 1) { setSel(v => v + 1) return } if (key.return) { if (stage === 'provider') { if (!provider) { return } setStage('model') setModelIdx(0) return } const model = models[modelIdx] if (provider && model) { onSelect(`${model} --provider ${provider.slug}${persistGlobal ? ' --global' : ` ${TUI_SESSION_MODEL_FLAG}`}`) } else { setStage('provider') } return } if (ch.toLowerCase() === 'g') { setPersistGlobal(v => !v) return } const n = ch === '0' ? 10 : parseInt(ch, 10) if (!Number.isNaN(n) && n >= 1 && n <= Math.min(10, count)) { const offset = windowOffset(count, sel, VISIBLE) if (stage === 'provider') { const next = offset + n - 1 if (providers[next]) { setProviderIdx(next) } } else if (provider && models[offset + n - 1]) { onSelect( `${models[offset + n - 1]} --provider ${provider.slug}${persistGlobal ? ' --global' : ` ${TUI_SESSION_MODEL_FLAG}`}` ) } } }) if (loading) { return loading models… } if (err) { return ( error: {err} Esc/q cancel ) } if (!providers.length) { return ( no authenticated providers Esc/q cancel ) } if (stage === 'provider') { const rows = providers.map( (p, i) => `${p.is_current ? '*' : ' '} ${names[i]} · ${p.total_models ?? p.models?.length ?? 0} models` ) const { items, offset } = windowItems(rows, providerIdx, VISIBLE) return ( Select provider (step 1/2) Full model IDs on the next step · Enter to continue Current: {currentModel || '(unknown)'} {provider?.warning ? `warning: ${provider.warning}` : ' '} {offset > 0 ? ` ↑ ${offset} more` : ' '} {Array.from({ length: VISIBLE }, (_, i) => { const row = items[i] const idx = offset + i return row ? ( {providerIdx === idx ? '▸ ' : ' '} {i + 1}. {row} ) : ( {' '} ) })} {offset + VISIBLE < rows.length ? ` ↓ ${rows.length - offset - VISIBLE} more` : ' '} persist: {persistGlobal ? 'global' : 'session'} · g toggle ↑/↓ select · Enter choose · 1-9,0 quick · Esc/q cancel ) } const { items, offset } = windowItems(models, modelIdx, VISIBLE) return ( Select model (step 2/2) {names[providerIdx] || '(unknown provider)'} · Esc back {provider?.warning ? `warning: ${provider.warning}` : ' '} {offset > 0 ? ` ↑ ${offset} more` : ' '} {Array.from({ length: VISIBLE }, (_, i) => { const row = items[i] const idx = offset + i if (!row) { return !models.length && i === 0 ? ( no models listed for this provider ) : ( {' '} ) } const prefix = modelIdx === idx ? '▸ ' : row === currentModel ? '* ' : ' ' return ( {prefix} {i + 1}. {row} ) })} {offset + VISIBLE < models.length ? ` ↓ ${models.length - offset - VISIBLE} more` : ' '} persist: {persistGlobal ? 'global' : 'session'} · g toggle {models.length ? '↑/↓ select · Enter switch · 1-9,0 quick · Esc back · q close' : 'Enter/Esc back · q close'} ) } interface ModelPickerProps { gw: GatewayClient onCancel: () => void onSelect: (value: string) => void sessionId: string | null t: Theme }