import React, { useState, useEffect, useMemo } from 'react'; import { Box, Text, useInput } from 'ink'; import fs from 'fs'; import path from 'path'; import os from 'os'; 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'; const warningsFilePath = path.join(os.tmpdir(), 'gemini-code-cli-warnings.txt'); interface AppProps { directory: string; } const App = ({ directory }: AppProps) => { const [query, setQuery] = useState(''); const [history, setHistory] = useState([]); const [startupWarnings, setStartupWarnings] = useState([]); const [historyIndex, setHistoryIndex] = useState(-1); const [originalQueryBeforeNav, setOriginalQueryBeforeNav] = useState(''); const { streamingState, submitQuery, initError } = useGeminiStream(setHistory); const { elapsedTime, currentLoadingPhrase } = useLoadingIndicator(streamingState); const userMessages = useMemo(() => { return history .filter((item): item is HistoryItem & { type: 'user'; text: string } => item.type === 'user' && typeof item.text === 'string' && item.text.trim() !== '' ) .map(item => item.text); }, [history]); useEffect(() => { try { if (fs.existsSync(warningsFilePath)) { console.log('[App] Found warnings file:', warningsFilePath); const warningsContent = fs.readFileSync(warningsFilePath, 'utf-8'); setStartupWarnings(warningsContent.split('\n').filter(line => line.trim() !== '')); try { fs.unlinkSync(warningsFilePath); } catch (unlinkErr: any) { console.warn(`[App] Warning: Could not delete warnings file: ${unlinkErr.message}`); } } else { console.log('[App] No warnings file found.'); } } catch (err: any) { console.error(`[App] Error checking/reading warnings file: ${err.message}`); } }, []); const handleInputSubmit = (value: PartListUnion) => { setHistoryIndex(-1); setOriginalQueryBeforeNav(''); submitQuery(value) .then(() => { setQuery(''); }) .catch(() => { setQuery(''); }); }; useEffect(() => { if ( initError && !history.some( (item) => item.type === 'error' && item.text?.includes(initError), ) ) { setHistory((prev) => [ ...prev, { id: Date.now(), type: 'error', text: `Initialization Error: ${initError}. Please check API key and configuration.`, } as HistoryItem, ]); } }, [initError, history]); const isWaitingForToolConfirmation = history.some( (item) => item.type === 'tool_group' && item.tools.some((tool) => tool.confirmationDetails !== undefined), ); const isInputActive = streamingState === StreamingState.Idle && !initError; useInput((input, key) => { if (!isInputActive || isWaitingForToolConfirmation) { return; } if (key.upArrow) { if (userMessages.length === 0) return; if (historyIndex === -1) { setOriginalQueryBeforeNav(query); } const nextIndex = Math.min(historyIndex + 1, userMessages.length - 1); if (nextIndex !== historyIndex) { setHistoryIndex(nextIndex); setQuery(userMessages[userMessages.length - 1 - nextIndex]); } } else if (key.downArrow) { if (historyIndex < 0) return; const nextIndex = Math.max(historyIndex - 1, -1); setHistoryIndex(nextIndex); if (nextIndex === -1) { setQuery(originalQueryBeforeNav); } else { setQuery(userMessages[userMessages.length - 1 - nextIndex]); } } else { if (input || key.backspace || key.delete || key.leftArrow || key.rightArrow) { if (historyIndex !== -1) { setHistoryIndex(-1); setOriginalQueryBeforeNav(''); } } } }, { isActive: isInputActive }); 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 && ( )}