mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-03-14 18:37:22 +00:00
* feat: integrate Gemini AI agent provider - Core Backend: Ported gemini-cli.js and gemini-response-handler.js to establish the CLI bridge. Registered 'gemini' as an active provider within index.js. - Core Frontend: Extended QuickSettingsPanel.jsx, Settings.jsx, and AgentListItem.jsx to render the Gemini provider option, models (gemini-pro, gemini-flash, etc.), and handle OAuth states. - WebSocket Pipeline: Added support for gemini-command executions in backend and payload processing of gemini-response and gemini-error streams in useChatRealtimeHandlers.ts. Resolved JSON double-stringification and sessionId stripping issues in the transmission handler. - Platform Compatibility: Added scripts/fix-node-pty.js postinstall script and modified posix_spawnp calls with sh -c wrapper to prevent ENOEXEC and MacOS permission errors when spawning the gemini headless binary. - UX & Design: Imported official Google Gemini branding via GeminiLogo.jsx and gemini-ai-icon.svg. Updated translations (chat.json) for en, zh-CN, and ko locales. * fix: propagate gemini permission mode from settings to cli - Added Gemini Permissions UI in Settings to toggle Auto Edit and YOLO modes - Synced gemini permission mode to localStorage - Passed permissionMode in useChatComposerState for Gemini commands - Mapped frontend permission modes to --yolo and --approval-mode options in gemini-cli.js * feat(gemini): Refactor Gemini CLI integration to use stream-json - Replaced regex buffering text-system with NDJSON stream parsing - Added fallback for restricted models like gemini-3.1-pro-preview * feat(gemini): Render tool_use and tool_result UI bubbles - Forwarded gemini tool NDJSON objects to the websocket - Added React state handlers in useChatRealtimeHandlers to match Claude's tool UI behavior * feat(gemini): Add native session resumption and UI token tracking - Captured cliSessionId from init events to map ClaudeCodeUI's chat sessionId directly into Gemini's internal session manager. - Updated gemini-cli.js spawn arguments to append the --resume proxy flag instead of naively dumping the accumulated chat history into the command prompt. - Handled result stream objects by proxying total_tokens back into the frontend's claude-status tracker to natively populate the UI label. - Eliminated gemini-3 model proxy filter entirely. * fix(gemini): Fix static 'Claude' name rendering in chat UI header - Added "gemini": "Gemini" translation strings to messageTypes across English, Korean, and Chinese loc dictionaries. - Updated AssistantThinkingIndicator and MessageComponent ternary checks to identify provider === 'gemini' and render the appropriate brand label instead of statically defaulting to Claude. * feat: Add Gemini session persistence API mapping and Sidebar UI * fix(gemini): Watch ~/.gemini/sessions for live UI updates Added the .gemini/sessions directory to PROVIDER_WATCH_PATHS so that Chokidar emits projects_updated websocket events when new Gemini sessions are created or modified, fixing live sidebar updates. * fix(gemini): Fix Gemini authentication status display in Settings UI - Injected 'checkGeminiAuthStatus' into the Settings.jsx React effect hook so that the UI can poll and render the 'geminiAuthStatus' state. - Updated 'checkGeminiCredentials()' inside server/routes/cli-auth.js to read from '~/.gemini/oauth_creds.json' and '~/.gemini/google_accounts.json', resolving the email address correctly. * Use logo-only icon for gemini * feat(gemini): Add Gemini 3 preview models to UI selection list * Fix Gemini CLI session resume bug and PR #422 review nitpicks * Fix Gemini tool calls disappearing from UI after completion * fix(gemini): resolve outstanding PR #422 feedback and stabilize gemini CLI timeouts * fix(gemini): resolve resume flag and shell session initialization issues This commit addresses the remaining PR comments for the Gemini CLI integration: - Moves the `--resume` flag logic outside the prompt command block, ensuring Gemini sessions correctly resume even when a new prompt isn't passed. - Updates `handleShellConnection` to correctly lookup the native `cliSessionId` from the internal `sessionId` when spawning Gemini sessions in a plain shell. - Refactors dynamic import of `sessionManager.js` back to a native static import for code consistency. * chore: fix TypeScript errors and remove gemini CLI dependency * fix: use cross-spawn on Windows to resolve gemini.cmd correctly --------- Co-authored-by: Haileyesus <118998054+blackmammoth@users.noreply.github.com>
90 lines
3.0 KiB
JavaScript
90 lines
3.0 KiB
JavaScript
import React, { useState, useEffect } from 'react';
|
|
import { cn } from '../lib/utils';
|
|
|
|
function GeminiStatus({ status, onAbort, isLoading }) {
|
|
const [elapsedTime, setElapsedTime] = useState(0);
|
|
const [animationPhase, setAnimationPhase] = useState(0);
|
|
|
|
// Update elapsed time every second
|
|
useEffect(() => {
|
|
if (!isLoading) {
|
|
setElapsedTime(0);
|
|
return;
|
|
}
|
|
|
|
const startTime = Date.now();
|
|
const timer = setInterval(() => {
|
|
const elapsed = Math.floor((Date.now() - startTime) / 1000);
|
|
setElapsedTime(elapsed);
|
|
}, 1000);
|
|
|
|
return () => clearInterval(timer);
|
|
}, [isLoading]);
|
|
|
|
// Animate the status indicator
|
|
useEffect(() => {
|
|
if (!isLoading) return;
|
|
|
|
const timer = setInterval(() => {
|
|
setAnimationPhase(prev => (prev + 1) % 4);
|
|
}, 500);
|
|
|
|
return () => clearInterval(timer);
|
|
}, [isLoading]);
|
|
|
|
if (!isLoading) return null;
|
|
|
|
// Clever action words that cycle
|
|
const actionWords = ['Thinking', 'Processing', 'Analyzing', 'Working', 'Computing', 'Reasoning'];
|
|
const actionIndex = Math.floor(elapsedTime / 3) % actionWords.length;
|
|
|
|
// Parse status data
|
|
const statusText = status?.text || actionWords[actionIndex];
|
|
const canInterrupt = status?.can_interrupt !== false;
|
|
|
|
// Animation characters
|
|
const spinners = ['✻', '✹', '✸', '✶'];
|
|
const currentSpinner = spinners[animationPhase];
|
|
|
|
return (
|
|
<div className="w-full mb-6 animate-in slide-in-from-bottom duration-300">
|
|
<div className="flex items-center justify-between max-w-4xl mx-auto bg-gradient-to-r from-cyan-900 to-blue-900 dark:from-cyan-950 dark:to-blue-950 text-white rounded-lg shadow-lg px-4 py-3">
|
|
<div className="flex-1">
|
|
<div className="flex items-center gap-3">
|
|
{/* Animated spinner */}
|
|
<span className={cn(
|
|
"text-xl transition-all duration-500",
|
|
animationPhase % 2 === 0 ? "text-cyan-400 scale-110" : "text-cyan-300"
|
|
)}>
|
|
{currentSpinner}
|
|
</span>
|
|
|
|
{/* Status text - first line */}
|
|
<div className="flex-1">
|
|
<div className="flex items-center gap-2">
|
|
<span className="font-medium text-sm">{statusText}...</span>
|
|
<span className="text-gray-400 text-sm">({elapsedTime}s)</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Interrupt button */}
|
|
{canInterrupt && onAbort && (
|
|
<button
|
|
type="button"
|
|
onClick={onAbort}
|
|
className="ml-3 text-xs bg-red-600 hover:bg-red-700 text-white px-2.5 py-1 sm:px-3 sm:py-1.5 rounded-md transition-colors flex items-center gap-1.5 flex-shrink-0"
|
|
>
|
|
<svg className="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
<path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
<span className="hidden sm:inline">Stop</span>
|
|
</button>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default GeminiStatus; |