2025-04-16 14:16:53 -07:00
import type { ApplyPatchCommand , ApprovalPolicy } from "../../approvals.js" ;
2025-04-16 12:56:08 -04:00
import type { CommandConfirmation } from "../../utils/agent/agent-loop.js" ;
import type { AppConfig } from "../../utils/config.js" ;
import type { ColorName } from "chalk" ;
import type { ResponseItem } from "openai/resources/responses/responses.mjs" ;
import TerminalChatInput from "./terminal-chat-input.js" ;
2025-04-19 07:23:02 -07:00
import { TerminalChatToolCallCommand } from "./terminal-chat-tool-call-command.js" ;
2025-04-16 12:56:08 -04:00
import TerminalMessageHistory from "./terminal-message-history.js" ;
2025-04-16 14:16:53 -07:00
import { formatCommandForDisplay } from "../../format-command.js" ;
2025-04-16 12:56:08 -04:00
import { useConfirmation } from "../../hooks/use-confirmation.js" ;
import { useTerminalSize } from "../../hooks/use-terminal-size.js" ;
import { AgentLoop } from "../../utils/agent/agent-loop.js" ;
2025-04-18 06:28:58 +10:00
import { ReviewDecision } from "../../utils/agent/review.js" ;
2025-04-18 15:48:30 +10:00
import { generateCompactSummary } from "../../utils/compact-summary.js" ;
2025-04-22 10:51:26 -04:00
import { getBaseUrl , getApiKey , saveConfig } from "../../utils/config.js" ;
2025-04-19 16:23:27 -07:00
import { extractAppliedPatches as _extractAppliedPatches } from "../../utils/extract-applied-patches.js" ;
import { getGitDiff } from "../../utils/get-diff.js" ;
2025-04-16 12:56:08 -04:00
import { createInputItem } from "../../utils/input-utils.js" ;
2025-04-21 09:51:34 -04:00
import { log } from "../../utils/logger/log.js" ;
2025-04-21 12:33:57 -04:00
import {
getAvailableModels ,
calculateContextPercentRemaining ,
uniqueById ,
} from "../../utils/model-utils.js" ;
2025-04-16 12:56:08 -04:00
import { CLI_VERSION } from "../../utils/session.js" ;
import { shortCwd } from "../../utils/short-path.js" ;
import { saveRollout } from "../../utils/storage/save-rollout.js" ;
import ApprovalModeOverlay from "../approval-mode-overlay.js" ;
2025-04-19 16:23:27 -07:00
import DiffOverlay from "../diff-overlay.js" ;
2025-04-16 12:56:08 -04:00
import HelpOverlay from "../help-overlay.js" ;
import HistoryOverlay from "../history-overlay.js" ;
import ModelOverlay from "../model-overlay.js" ;
import { Box , Text } from "ink" ;
2025-04-19 18:29:00 -07:00
import { spawn } from "node:child_process" ;
2025-04-18 06:28:58 +10:00
import OpenAI from "openai" ;
2025-04-17 16:19:26 -07:00
import React , { useEffect , useMemo , useRef , useState } from "react" ;
2025-04-16 12:56:08 -04:00
import { inspect } from "util" ;
2025-04-20 09:51:49 -07:00
export type OverlayModeType =
| "none"
| "history"
| "model"
| "approval"
| "help"
| "diff" ;
2025-04-16 12:56:08 -04:00
type Props = {
config : AppConfig ;
prompt? : string ;
imagePaths? : Array < string > ;
approvalPolicy : ApprovalPolicy ;
2025-04-17 15:39:26 -07:00
additionalWritableRoots : ReadonlyArray < string > ;
2025-04-16 12:56:08 -04:00
fullStdout : boolean ;
} ;
const colorsByPolicy : Record < ApprovalPolicy , ColorName | undefined > = {
"suggest" : undefined ,
"auto-edit" : "greenBright" ,
"full-auto" : "green" ,
} ;
2025-04-18 06:28:58 +10:00
/ * *
* Generates an explanation for a shell command using the OpenAI API .
*
* @param command The command to explain
* @param model The model to use for generating the explanation
2025-04-22 10:51:26 -04:00
* @param flexMode Whether to use the flex - mode service tier
* @param config The configuration object
2025-04-18 06:28:58 +10:00
* @returns A human - readable explanation of what the command does
* /
async function generateCommandExplanation (
command : Array < string > ,
model : string ,
2025-04-18 22:15:01 -07:00
flexMode : boolean ,
2025-04-22 10:51:26 -04:00
config : AppConfig ,
2025-04-18 06:28:58 +10:00
) : Promise < string > {
try {
// Create a temporary OpenAI client
const oai = new OpenAI ( {
2025-04-22 10:51:26 -04:00
apiKey : getApiKey ( config . provider ) ,
baseURL : getBaseUrl ( config . provider ) ,
2025-04-18 06:28:58 +10:00
} ) ;
// Format the command for display
const commandForDisplay = formatCommandForDisplay ( command ) ;
// Create a prompt that asks for an explanation with a more detailed system prompt
const response = await oai . chat . completions . create ( {
model ,
2025-04-18 22:15:01 -07:00
. . . ( flexMode ? { service_tier : "flex" } : { } ) ,
2025-04-18 06:28:58 +10:00
messages : [
{
role : "system" ,
content :
"You are an expert in shell commands and terminal operations. Your task is to provide detailed, accurate explanations of shell commands that users are considering executing. Break down each part of the command, explain what it does, identify any potential risks or side effects, and explain why someone might want to run it. Be specific about what files or systems will be affected. If the command could potentially be harmful, make sure to clearly highlight those risks." ,
} ,
{
role : "user" ,
content : ` Please explain this shell command in detail: \` ${ commandForDisplay } \` \ n \ nProvide a structured explanation that includes: \ n1. A brief overview of what the command does \ n2. A breakdown of each part of the command (flags, arguments, etc.) \ n3. What files, directories, or systems will be affected \ n4. Any potential risks or side effects \ n5. Why someone might want to run this command \ n \ nBe specific and technical - this explanation will help the user decide whether to approve or reject the command. ` ,
} ,
] ,
} ) ;
// Extract the explanation from the response
const explanation =
response . choices [ 0 ] ? . message . content || "Unable to generate explanation." ;
return explanation ;
} catch ( error ) {
log ( ` Error generating command explanation: ${ error } ` ) ;
let errorMessage = "Unable to generate explanation due to an error." ;
if ( error instanceof Error ) {
errorMessage = ` Unable to generate explanation: ${ error . message } ` ;
// If it's an API error, check for more specific information
if ( "status" in error && typeof error . status === "number" ) {
// Handle API-specific errors
if ( error . status === 401 ) {
errorMessage =
"Unable to generate explanation: API key is invalid or expired." ;
} else if ( error . status === 429 ) {
errorMessage =
"Unable to generate explanation: Rate limit exceeded. Please try again later." ;
} else if ( error . status >= 500 ) {
errorMessage =
"Unable to generate explanation: OpenAI service is currently unavailable. Please try again later." ;
}
}
}
return errorMessage ;
}
}
2025-04-16 12:56:08 -04:00
export default function TerminalChat ( {
config ,
prompt : _initialPrompt ,
imagePaths : _initialImagePaths ,
approvalPolicy : initialApprovalPolicy ,
2025-04-17 15:39:26 -07:00
additionalWritableRoots ,
2025-04-16 12:56:08 -04:00
fullStdout ,
} : Props ) : React . ReactElement {
2025-04-25 02:08:52 +08:00
const notify = Boolean ( config . notify ) ;
2025-04-16 12:56:08 -04:00
const [ model , setModel ] = useState < string > ( config . model ) ;
2025-04-20 23:59:34 -04:00
const [ provider , setProvider ] = useState < string > ( config . provider || "openai" ) ;
2025-04-16 12:56:08 -04:00
const [ lastResponseId , setLastResponseId ] = useState < string | null > ( null ) ;
const [ items , setItems ] = useState < Array < ResponseItem > > ( [ ] ) ;
const [ loading , setLoading ] = useState < boolean > ( false ) ;
const [ approvalPolicy , setApprovalPolicy ] = useState < ApprovalPolicy > (
initialApprovalPolicy ,
) ;
const [ thinkingSeconds , setThinkingSeconds ] = useState ( 0 ) ;
2025-04-21 12:33:57 -04:00
2025-04-18 15:48:30 +10:00
const handleCompact = async ( ) = > {
setLoading ( true ) ;
try {
2025-04-18 22:15:01 -07:00
const summary = await generateCompactSummary (
items ,
model ,
Boolean ( config . flexMode ) ,
2025-04-22 10:51:26 -04:00
config ,
2025-04-18 22:15:01 -07:00
) ;
2025-04-18 15:48:30 +10:00
setItems ( [
{
id : ` compact- ${ Date . now ( ) } ` ,
type : "message" ,
role : "assistant" ,
content : [ { type : "output_text" , text : summary } ] ,
} as ResponseItem ,
] ) ;
} catch ( err ) {
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` compact-error- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [
{ type : "input_text" , text : ` Failed to compact context: ${ err } ` } ,
] ,
} as ResponseItem ,
] ) ;
} finally {
setLoading ( false ) ;
}
} ;
2025-04-21 12:33:57 -04:00
2025-04-18 06:28:58 +10:00
const {
requestConfirmation ,
confirmationPrompt ,
explanation ,
submitConfirmation ,
} = useConfirmation ( ) ;
2025-04-20 09:51:49 -07:00
const [ overlayMode , setOverlayMode ] = useState < OverlayModeType > ( "none" ) ;
2025-04-16 12:56:08 -04:00
2025-04-19 16:23:27 -07:00
// Store the diff text when opening the diff overlay so the view isn’ t
// recomputed on every re‑ render while it is open.
// diffText is passed down to the DiffOverlay component. The setter is
// currently unused but retained for potential future updates. Prefix with
// an underscore so eslint ignores the unused variable.
const [ diffText , _setDiffText ] = useState < string > ( "" ) ;
2025-04-16 12:56:08 -04:00
const [ initialPrompt , setInitialPrompt ] = useState ( _initialPrompt ) ;
const [ initialImagePaths , setInitialImagePaths ] =
useState ( _initialImagePaths ) ;
const PWD = React . useMemo ( ( ) = > shortCwd ( ) , [ ] ) ;
// Keep a single AgentLoop instance alive across renders;
// recreate only when model/instructions/approvalPolicy change.
const agentRef = React . useRef < AgentLoop > ( ) ;
const [ , forceUpdate ] = React . useReducer ( ( c ) = > c + 1 , 0 ) ; // trigger re‑ render
// ────────────────────────────────────────────────────────────────
// DEBUG: log every render w/ key bits of state
// ────────────────────────────────────────────────────────────────
2025-04-20 09:58:06 -07:00
log (
2025-04-21 12:33:57 -04:00
` render - agent? ${ Boolean ( agentRef . current ) } loading= ${ loading } items= ${
2025-04-20 09:58:06 -07:00
items . length
} ` ,
) ;
2025-04-16 12:56:08 -04:00
useEffect ( ( ) = > {
2025-04-21 12:33:57 -04:00
// Skip recreating the agent if awaiting a decision on a pending confirmation.
2025-04-19 07:21:19 -07:00
if ( confirmationPrompt != null ) {
2025-04-20 09:58:06 -07:00
log ( "skip AgentLoop recreation due to pending confirmationPrompt" ) ;
2025-04-19 07:21:19 -07:00
return ;
}
2025-04-20 09:58:06 -07:00
log ( "creating NEW AgentLoop" ) ;
log (
2025-04-20 23:59:34 -04:00
` model= ${ model } provider= ${ provider } instructions= ${ Boolean (
2025-04-20 09:58:06 -07:00
config . instructions ,
) } approvalPolicy = $ { approvalPolicy } ` ,
) ;
2025-04-16 12:56:08 -04:00
2025-04-21 12:33:57 -04:00
// Tear down any existing loop before creating a new one.
2025-04-16 12:56:08 -04:00
agentRef . current ? . terminate ( ) ;
2025-04-21 09:51:34 -04:00
const sessionId = crypto . randomUUID ( ) ;
2025-04-16 12:56:08 -04:00
agentRef . current = new AgentLoop ( {
model ,
2025-04-20 23:59:34 -04:00
provider ,
2025-04-16 12:56:08 -04:00
config ,
instructions : config.instructions ,
approvalPolicy ,
2025-04-22 01:30:16 -07:00
disableResponseStorage : config.disableResponseStorage ,
2025-04-17 15:39:26 -07:00
additionalWritableRoots ,
2025-04-16 12:56:08 -04:00
onLastResponseId : setLastResponseId ,
onItem : ( item ) = > {
log ( ` onItem: ${ JSON . stringify ( item ) } ` ) ;
setItems ( ( prev ) = > {
const updated = uniqueById ( [ . . . prev , item as ResponseItem ] ) ;
2025-04-21 09:51:34 -04:00
saveRollout ( sessionId , updated ) ;
2025-04-16 12:56:08 -04:00
return updated ;
} ) ;
} ,
onLoading : setLoading ,
getCommandConfirmation : async (
command : Array < string > ,
applyPatch : ApplyPatchCommand | undefined ,
) : Promise < CommandConfirmation > = > {
log ( ` getCommandConfirmation: ${ command } ` ) ;
const commandForDisplay = formatCommandForDisplay ( command ) ;
2025-04-18 06:28:58 +10:00
// First request for confirmation
let { decision : review , customDenyMessage } = await requestConfirmation (
< TerminalChatToolCallCommand commandForDisplay = { commandForDisplay } / > ,
) ;
2025-04-21 12:33:57 -04:00
// If the user wants an explanation, generate one and ask again.
2025-04-18 06:28:58 +10:00
if ( review === ReviewDecision . EXPLAIN ) {
log ( ` Generating explanation for command: ${ commandForDisplay } ` ) ;
2025-04-18 22:15:01 -07:00
const explanation = await generateCommandExplanation (
command ,
model ,
Boolean ( config . flexMode ) ,
2025-04-22 10:51:26 -04:00
config ,
2025-04-18 22:15:01 -07:00
) ;
2025-04-18 06:28:58 +10:00
log ( ` Generated explanation: ${ explanation } ` ) ;
2025-04-21 12:33:57 -04:00
// Ask for confirmation again, but with the explanation.
2025-04-18 06:28:58 +10:00
const confirmResult = await requestConfirmation (
2025-04-16 12:56:08 -04:00
< TerminalChatToolCallCommand
commandForDisplay = { commandForDisplay }
2025-04-18 06:28:58 +10:00
explanation = { explanation }
2025-04-16 12:56:08 -04:00
/ > ,
) ;
2025-04-18 06:28:58 +10:00
2025-04-21 12:33:57 -04:00
// Update the decision based on the second confirmation.
2025-04-18 06:28:58 +10:00
review = confirmResult . decision ;
customDenyMessage = confirmResult . customDenyMessage ;
2025-04-21 12:33:57 -04:00
// Return the final decision with the explanation.
2025-04-18 06:28:58 +10:00
return { review , customDenyMessage , applyPatch , explanation } ;
}
2025-04-16 12:56:08 -04:00
return { review , customDenyMessage , applyPatch } ;
} ,
} ) ;
2025-04-21 12:33:57 -04:00
// Force a render so JSX below can "see" the freshly created agent.
2025-04-16 12:56:08 -04:00
forceUpdate ( ) ;
2025-04-20 09:58:06 -07:00
log ( ` AgentLoop created: ${ inspect ( agentRef . current , { depth : 1 } )} ` ) ;
2025-04-16 12:56:08 -04:00
return ( ) = > {
2025-04-20 09:58:06 -07:00
log ( "terminating AgentLoop" ) ;
2025-04-16 12:56:08 -04:00
agentRef . current ? . terminate ( ) ;
agentRef . current = undefined ;
forceUpdate ( ) ; // re‑ render after teardown too
} ;
2025-04-21 15:01:09 -04:00
// We intentionally omit 'approvalPolicy' and 'confirmationPrompt' from the deps
// so switching modes or showing confirmation dialogs doesn’ t tear down the loop.
// eslint-disable-next-line react-hooks/exhaustive-deps
} , [ model , provider , config , requestConfirmation , additionalWritableRoots ] ) ;
2025-04-16 12:56:08 -04:00
2025-04-21 12:33:57 -04:00
// Whenever loading starts/stops, reset or start a timer — but pause the
2025-04-16 12:56:08 -04:00
// timer while a confirmation overlay is displayed so we don't trigger a
// re‑ render every second during apply_patch reviews.
useEffect ( ( ) = > {
let handle : ReturnType < typeof setInterval > | null = null ;
// Only tick the "thinking…" timer when the agent is actually processing
// a request *and* the user is not being asked to review a command.
if ( loading && confirmationPrompt == null ) {
setThinkingSeconds ( 0 ) ;
handle = setInterval ( ( ) = > {
setThinkingSeconds ( ( s ) = > s + 1 ) ;
} , 1000 ) ;
} else {
if ( handle ) {
clearInterval ( handle ) ;
}
setThinkingSeconds ( 0 ) ;
}
return ( ) = > {
if ( handle ) {
clearInterval ( handle ) ;
}
} ;
} , [ loading , confirmationPrompt ] ) ;
2025-04-21 12:33:57 -04:00
// Notify desktop with a preview when an assistant response arrives.
2025-04-17 16:19:26 -07:00
const prevLoadingRef = useRef < boolean > ( false ) ;
useEffect ( ( ) = > {
2025-04-21 12:33:57 -04:00
// Only notify when notifications are enabled.
2025-04-17 16:19:26 -07:00
if ( ! notify ) {
prevLoadingRef . current = loading ;
return ;
}
2025-04-21 12:33:57 -04:00
2025-04-17 16:19:26 -07:00
if (
prevLoadingRef . current &&
! loading &&
confirmationPrompt == null &&
items . length > 0
) {
if ( process . platform === "darwin" ) {
// find the last assistant message
const assistantMessages = items . filter (
( i ) = > i . type === "message" && i . role === "assistant" ,
) ;
const last = assistantMessages [ assistantMessages . length - 1 ] ;
if ( last ) {
const text = last . content
. map ( ( c ) = > {
if ( c . type === "output_text" ) {
return c . text ;
}
return "" ;
} )
. join ( "" )
. trim ( ) ;
const preview = text . replace ( /\n/g , " " ) . slice ( 0 , 100 ) ;
const safePreview = preview . replace ( /"/g , '\\"' ) ;
const title = "Codex CLI" ;
const cwd = PWD ;
2025-04-19 18:29:00 -07:00
spawn ( "osascript" , [
"-e" ,
` display notification " ${ safePreview } " with title " ${ title } " subtitle " ${ cwd } " sound name "Ping" ` ,
] ) ;
2025-04-17 16:19:26 -07:00
}
}
}
prevLoadingRef . current = loading ;
} , [ notify , loading , confirmationPrompt , items , PWD ] ) ;
2025-04-21 12:33:57 -04:00
// Let's also track whenever the ref becomes available.
2025-04-16 12:56:08 -04:00
const agent = agentRef . current ;
useEffect ( ( ) = > {
2025-04-20 09:58:06 -07:00
log ( ` agentRef.current is now ${ Boolean ( agent ) } ` ) ;
2025-04-16 12:56:08 -04:00
} , [ agent ] ) ;
// ---------------------------------------------------------------------
// Dynamic layout constraints – keep total rendered rows <= terminal rows
// ---------------------------------------------------------------------
const { rows : terminalRows } = useTerminalSize ( ) ;
useEffect ( ( ) = > {
const processInitialInputItems = async ( ) = > {
if (
( ! initialPrompt || initialPrompt . trim ( ) === "" ) &&
( ! initialImagePaths || initialImagePaths . length === 0 )
) {
return ;
}
const inputItems = [
await createInputItem ( initialPrompt || "" , initialImagePaths || [ ] ) ,
] ;
2025-04-21 12:33:57 -04:00
// Clear them to prevent subsequent runs.
2025-04-16 12:56:08 -04:00
setInitialPrompt ( "" ) ;
setInitialImagePaths ( [ ] ) ;
agent ? . run ( inputItems ) ;
} ;
processInitialInputItems ( ) ;
} , [ agent , initialPrompt , initialImagePaths ] ) ;
// ────────────────────────────────────────────────────────────────
// In-app warning if CLI --model isn't in fetched list
// ────────────────────────────────────────────────────────────────
useEffect ( ( ) = > {
( async ( ) = > {
2025-04-20 23:59:34 -04:00
const available = await getAvailableModels ( provider ) ;
2025-04-16 12:56:08 -04:00
if ( model && available . length > 0 && ! available . includes ( model ) ) {
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` unknown-model- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [
{
type : "input_text" ,
2025-04-20 23:59:34 -04:00
text : ` Warning: model " ${ model } " is not in the list of available models for provider " ${ provider } ". ` ,
2025-04-16 12:56:08 -04:00
} ,
] ,
} ,
] ) ;
}
} ) ( ) ;
// run once on mount
// eslint-disable-next-line react-hooks/exhaustive-deps
} , [ ] ) ;
2025-04-21 12:33:57 -04:00
// Just render every item in order, no grouping/collapse.
2025-04-16 12:56:08 -04:00
const lastMessageBatch = items . map ( ( item ) = > ( { item } ) ) ;
const groupCounts : Record < string , number > = { } ;
const userMsgCount = items . filter (
( i ) = > i . type === "message" && i . role === "user" ,
) . length ;
const contextLeftPercent = useMemo (
( ) = > calculateContextPercentRemaining ( items , model ) ,
[ items , model ] ,
) ;
return (
< Box flexDirection = "column" >
< Box flexDirection = "column" >
{ agent ? (
< TerminalMessageHistory
2025-04-20 09:51:49 -07:00
setOverlayMode = { setOverlayMode }
2025-04-16 12:56:08 -04:00
batch = { lastMessageBatch }
groupCounts = { groupCounts }
items = { items }
userMsgCount = { userMsgCount }
confirmationPrompt = { confirmationPrompt }
loading = { loading }
thinkingSeconds = { thinkingSeconds }
fullStdout = { fullStdout }
headerProps = { {
terminalRows ,
version : CLI_VERSION ,
PWD ,
model ,
2025-04-20 23:59:34 -04:00
provider ,
2025-04-16 12:56:08 -04:00
approvalPolicy ,
colorsByPolicy ,
agent ,
initialImagePaths ,
2025-04-18 22:15:01 -07:00
flexModeEnabled : Boolean ( config . flexMode ) ,
2025-04-16 12:56:08 -04:00
} }
/ >
) : (
< Box >
< Text color = "gray" > Initializing agent … < / Text >
< / Box >
) }
fix: allow proper exit from new Switch approval mode dialog (#453)
As described in
https://github.com/openai/codex/issues/392#issuecomment-2817090022
introduced by #400
The testing I'd done worked correctly because I was using the (s)
shortcut, but selecting the same option using arrow‑key → Enter on
“Switch approval mode” was preventing the user from subsequently exiting
the Switch approval mode dialog, requiring a ^C to quit codex entirely.
With this fix, both entry methods work correctly in my testing.
Per codex:
Issue
- When you navigated down (↓) to “Switch approval mode (s)” in the Shell
Command review dialog and pressed Enter, the ApprovalModeOverlay would
open—but because the underlying `TerminalChatCommandReview` component
stayed mounted (albeit disabled), its own Ink input handlers immediately
re‑captured the same key events and re‑opened the overlay as soon as you
hit Esc or Enter again. In practice this made it impossible to exit the
submenu.
Root cause
- We only disabled the SelectInput via `isDisabled`, but never fully
unmounted the review UI when an overlay was shown, so its `useInput` and
`<Select>` hooks were still active and “stealing” keys.
Fix
- In `terminal-chat.tsx` we now only render `<TerminalChatInput>` (and
by extension `TerminalChatCommandReview`) when `overlayMode === "none"`.
That unmounts all of its key handlers whenever any overlay (history,
model, approval, help, diff) is open, so no input leaks through.
Files changed
- **src/components/chat/terminal-chat.tsx**: Wrapped the entire
`<TerminalChatInput>` block in `overlayMode === "none" && agent`
With that in place, arrow‑key → Enter on “Switch approval mode”
correctly opens the overlay, and then you can use Enter/Esc inside the
overlay without getting stuck or immediately re‑opening it.
2025-04-20 22:19:53 -07:00
{ overlayMode === "none" && agent && (
2025-04-16 12:56:08 -04:00
< TerminalChatInput
loading = { loading }
setItems = { setItems }
isNew = { Boolean ( items . length === 0 ) }
setLastResponseId = { setLastResponseId }
confirmationPrompt = { confirmationPrompt }
2025-04-18 06:28:58 +10:00
explanation = { explanation }
2025-04-16 12:56:08 -04:00
submitConfirmation = { (
decision : ReviewDecision ,
customDenyMessage? : string ,
) = >
submitConfirmation ( {
decision ,
customDenyMessage ,
} )
}
contextLeftPercent = { contextLeftPercent }
openOverlay = { ( ) = > setOverlayMode ( "history" ) }
openModelOverlay = { ( ) = > setOverlayMode ( "model" ) }
openApprovalOverlay = { ( ) = > setOverlayMode ( "approval" ) }
openHelpOverlay = { ( ) = > setOverlayMode ( "help" ) }
2025-04-19 16:23:27 -07:00
openDiffOverlay = { ( ) = > {
const { isGitRepo , diff } = getGitDiff ( ) ;
let text : string ;
if ( isGitRepo ) {
text = diff ;
} else {
text = "`/diff` — _not inside a git repository_" ;
}
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` diff- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [ { type : "input_text" , text } ] ,
} ,
] ) ;
// Ensure no overlay is shown.
setOverlayMode ( "none" ) ;
} }
2025-04-18 15:48:30 +10:00
onCompact = { handleCompact }
2025-04-16 12:56:08 -04:00
active = { overlayMode === "none" }
interruptAgent = { ( ) = > {
if ( ! agent ) {
return ;
}
2025-04-20 09:58:06 -07:00
log (
"TerminalChat: interruptAgent invoked – calling agent.cancel()" ,
) ;
2025-04-16 12:56:08 -04:00
agent . cancel ( ) ;
setLoading ( false ) ;
2025-04-17 15:20:19 +10:00
// Add a system message to indicate the interruption
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` interrupt- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [
{
type : "input_text" ,
text : "⏹️ Execution interrupted by user. You can continue typing." ,
} ,
] ,
} ,
] ) ;
2025-04-16 12:56:08 -04:00
} }
submitInput = { ( inputs ) = > {
agent . run ( inputs , lastResponseId || "" ) ;
return { } ;
} }
2025-04-18 14:09:35 -07:00
items = { items }
2025-04-18 18:13:34 -07:00
thinkingSeconds = { thinkingSeconds }
2025-04-16 12:56:08 -04:00
/ >
) }
{ overlayMode === "history" && (
< HistoryOverlay items = { items } onExit = { ( ) = > setOverlayMode ( "none" ) } / >
) }
{ overlayMode === "model" && (
< ModelOverlay
currentModel = { model }
2025-04-23 13:45:56 +08:00
providers = { config . providers }
2025-04-20 23:59:34 -04:00
currentProvider = { provider }
2025-04-16 12:56:08 -04:00
hasLastResponse = { Boolean ( lastResponseId ) }
onSelect = { ( newModel ) = > {
2025-04-20 09:58:06 -07:00
log (
"TerminalChat: interruptAgent invoked – calling agent.cancel()" ,
) ;
if ( ! agent ) {
log ( "TerminalChat: agent is not ready yet" ) ;
2025-04-16 12:56:08 -04:00
}
agent ? . cancel ( ) ;
setLoading ( false ) ;
setModel ( newModel ) ;
setLastResponseId ( ( prev ) = >
prev && newModel !== model ? null : prev ,
) ;
2025-04-20 23:59:34 -04:00
// Save model to config
saveConfig ( {
. . . config ,
model : newModel ,
provider : provider ,
} ) ;
2025-04-16 12:56:08 -04:00
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` switch-model- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [
{
type : "input_text" ,
text : ` Switched model to ${ newModel } ` ,
} ,
] ,
} ,
] ) ;
setOverlayMode ( "none" ) ;
} }
2025-04-20 23:59:34 -04:00
onSelectProvider = { ( newProvider ) = > {
log (
"TerminalChat: interruptAgent invoked – calling agent.cancel()" ,
) ;
if ( ! agent ) {
log ( "TerminalChat: agent is not ready yet" ) ;
}
agent ? . cancel ( ) ;
setLoading ( false ) ;
2025-04-21 12:33:57 -04:00
// Select default model for the new provider.
2025-04-20 23:59:34 -04:00
const defaultModel = model ;
2025-04-21 12:33:57 -04:00
// Save provider to config.
2025-04-20 23:59:34 -04:00
const updatedConfig = {
. . . config ,
provider : newProvider ,
model : defaultModel ,
} ;
saveConfig ( updatedConfig ) ;
setProvider ( newProvider ) ;
setModel ( defaultModel ) ;
setLastResponseId ( ( prev ) = >
prev && newProvider !== provider ? null : prev ,
) ;
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` switch-provider- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [
{
type : "input_text" ,
text : ` Switched provider to ${ newProvider } with model ${ defaultModel } ` ,
} ,
] ,
} ,
] ) ;
// Don't close the overlay so user can select a model for the new provider
// setOverlayMode("none");
} }
2025-04-16 12:56:08 -04:00
onExit = { ( ) = > setOverlayMode ( "none" ) }
/ >
) }
{ overlayMode === "approval" && (
< ApprovalModeOverlay
currentMode = { approvalPolicy }
onSelect = { ( newMode ) = > {
2025-04-21 12:33:57 -04:00
// Update approval policy without cancelling an in-progress session.
2025-04-16 12:56:08 -04:00
if ( newMode === approvalPolicy ) {
return ;
}
2025-04-21 12:33:57 -04:00
2025-04-16 12:56:08 -04:00
setApprovalPolicy ( newMode as ApprovalPolicy ) ;
2025-04-19 07:21:19 -07:00
if ( agentRef . current ) {
2025-04-19 11:22:45 -07:00
(
agentRef . current as unknown as {
approvalPolicy : ApprovalPolicy ;
}
) . approvalPolicy = newMode as ApprovalPolicy ;
2025-04-19 07:21:19 -07:00
}
2025-04-16 12:56:08 -04:00
setItems ( ( prev ) = > [
. . . prev ,
{
id : ` switch-approval- ${ Date . now ( ) } ` ,
type : "message" ,
role : "system" ,
content : [
{
type : "input_text" ,
text : ` Switched approval mode to ${ newMode } ` ,
} ,
] ,
} ,
] ) ;
setOverlayMode ( "none" ) ;
} }
onExit = { ( ) = > setOverlayMode ( "none" ) }
/ >
) }
{ overlayMode === "help" && (
< HelpOverlay onExit = { ( ) = > setOverlayMode ( "none" ) } / >
) }
2025-04-19 16:23:27 -07:00
{ overlayMode === "diff" && (
< DiffOverlay
diffText = { diffText }
onExit = { ( ) = > setOverlayMode ( "none" ) }
/ >
) }
2025-04-16 12:56:08 -04:00
< / Box >
< / Box >
) ;
}