/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import { ThoughtSummary } from '@google/gemini-cli-core'; import React from 'react'; import { Box, Text } from 'ink'; import { theme } from '../semantic-colors.js'; import { useStreamingContext } from '../contexts/StreamingContext.js'; import { StreamingState } from '../types.js'; import { GeminiRespondingSpinner } from './GeminiRespondingSpinner.js'; import { formatDuration } from '../utils/formatters.js'; import { useTerminalSize } from '../hooks/useTerminalSize.js'; import { isNarrowWidth } from '../utils/isNarrowWidth.js'; interface LoadingIndicatorProps { currentLoadingPhrase?: string; elapsedTime: number; rightContent?: React.ReactNode; thought?: ThoughtSummary | null; } export const LoadingIndicator: React.FC = ({ currentLoadingPhrase, elapsedTime, rightContent, thought, }) => { const streamingState = useStreamingContext(); const { columns: terminalWidth } = useTerminalSize(); const isNarrow = isNarrowWidth(terminalWidth); if (streamingState === StreamingState.Idle) { return null; } const primaryText = thought?.subject || currentLoadingPhrase; const cancelAndTimerContent = streamingState !== StreamingState.WaitingForConfirmation ? `(esc to cancel, ${elapsedTime < 60 ? `${elapsedTime}s` : formatDuration(elapsedTime * 1000)})` : null; return ( {/* Main loading line */} {primaryText && {primaryText}} {!isNarrow && cancelAndTimerContent && ( {cancelAndTimerContent} )} {!isNarrow && {/* Spacer */}} {!isNarrow && rightContent && {rightContent}} {isNarrow && cancelAndTimerContent && ( {cancelAndTimerContent} )} {isNarrow && rightContent && {rightContent}} ); };