import React, { useState } from 'react'; import { Box, Text } from 'ink'; import type { HistoryItem } from './types.js'; import { useGeminiStream } from './hooks/useGeminiStream.js'; import { useLoadingIndicator } from './hooks/useLoadingIndicator.js'; import { Header } from './components/Header.js'; import { Tips } from './components/Tips.js'; import { HistoryDisplay } from './components/HistoryDisplay.js'; import { LoadingIndicator } from './components/LoadingIndicator.js'; import { InputPrompt } from './components/InputPrompt.js'; import { Footer } from './components/Footer.js'; import { StreamingState } from '../core/gemini-stream.js'; import { PartListUnion } from '@google/genai'; import { ITermDetectionWarning } from './utils/itermDetection.js'; import { useStartupWarnings, useInitializationErrorEffect, } from './hooks/useAppEffects.js'; interface AppProps { directory: string; } export const App = ({ directory }: AppProps) => { const [query, setQuery] = useState(''); const [history, setHistory] = useState([]); const [startupWarnings, setStartupWarnings] = useState([]); const { streamingState, submitQuery, initError } = useGeminiStream(setHistory); const { elapsedTime, currentLoadingPhrase } = useLoadingIndicator(streamingState); useStartupWarnings(setStartupWarnings); useInitializationErrorEffect(initError, history, setHistory); const handleInputSubmit = (value: PartListUnion) => { submitQuery(value) .then(() => { setQuery(''); }) .catch(() => { setQuery(''); }); }; const isWaitingForToolConfirmation = history.some( (item) => item.type === 'tool_group' && item.tools.some((tool) => tool.confirmationDetails !== undefined), ); const isInputActive = streamingState === StreamingState.Idle && !initError; return (
{startupWarnings.length > 0 && ( {startupWarnings.map((warning, index) => ( {warning} ))} )} {initError && streamingState !== StreamingState.Responding && !isWaitingForToolConfirmation && ( {history.find( (item) => item.type === 'error' && item.text?.includes(initError), )?.text ? ( { history.find( (item) => item.type === 'error' && item.text?.includes(initError), )?.text } ) : ( <> Initialization Error: {initError} {' '} Please check API key and configuration. )} )} {!isWaitingForToolConfirmation && isInputActive && ( )} {process.env.TERM_PROGRAM === 'iTerm.app' && ( Note: Flickering may occur in iTerm. )}