/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import { renderWithProviders } from '../../../test-utils/render.js'; import { createMockSettings } from '../../../test-utils/settings.js'; import { ToolResultDisplay } from './ToolResultDisplay.js'; import { describe, it, expect, vi } from 'vitest'; import { makeFakeConfig, type AnsiOutput } from '@google/gemini-cli-core'; describe('ToolResultDisplay', () => { beforeEach(() => { vi.clearAllMocks(); }); it('uses ScrollableList for ANSI output in alternate buffer mode', async () => { const content = 'ansi content'; const ansiResult: AnsiOutput = [ [ { text: content, fg: 'red', bg: 'black', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], ]; const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: true }), settings: createMockSettings({ ui: { useAlternateBuffer: true } }), }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toContain(content); unmount(); }); it('uses Scrollable for non-ANSI output in alternate buffer mode', async () => { const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: true }), settings: createMockSettings({ ui: { useAlternateBuffer: true } }), }, ); await waitUntilReady(); const output = lastFrame(); // With real components, we check for the content itself expect(output).toContain('Markdown content'); unmount(); }); it('passes hasFocus prop to scrollable components', async () => { const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: true }), settings: createMockSettings({ ui: { useAlternateBuffer: true } }), }, ); await waitUntilReady(); expect(lastFrame()).toContain('Some result'); unmount(); }); it('renders string result as markdown by default', async () => { const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('renders string result as plain text when renderOutputAsMarkdown is false', async () => { const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), uiState: { constrainHeight: true }, }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('truncates very long string results', { timeout: 20000 }, async () => { const longString = 'a'.repeat(1000005); const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), uiState: { constrainHeight: true }, }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('renders file diff result', async () => { const diffResult = { fileDiff: 'diff content', fileName: 'test.ts', }; const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('renders ANSI output result', async () => { const ansiResult: AnsiOutput = [ [ { text: 'ansi content', fg: 'red', bg: 'black', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], ]; const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('renders nothing for todos result', async () => { const todoResult = { todos: [], }; const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), }, ); await waitUntilReady(); const output = lastFrame({ allowEmpty: true }); expect(output).toMatchSnapshot(); unmount(); }); it('does not fall back to plain text if availableHeight is set and not in alternate buffer', async () => { // availableHeight calculation: 20 - 1 - 5 = 14 > 3 const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), uiState: { constrainHeight: true }, }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('keeps markdown if in alternate buffer even with availableHeight', async () => { const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: true }), settings: createMockSettings({ ui: { useAlternateBuffer: true } }), }, ); await waitUntilReady(); const output = lastFrame(); expect(output).toMatchSnapshot(); unmount(); }); it('truncates ANSI output when maxLines is provided', async () => { const ansiResult: AnsiOutput = [ [ { text: 'Line 1', fg: '', bg: '', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], [ { text: 'Line 2', fg: '', bg: '', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], [ { text: 'Line 3', fg: '', bg: '', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], [ { text: 'Line 4', fg: '', bg: '', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], [ { text: 'Line 5', fg: '', bg: '', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ], ]; const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), uiState: { constrainHeight: true }, }, ); await waitUntilReady(); const output = lastFrame(); expect(output).not.toContain('Line 1'); expect(output).not.toContain('Line 2'); expect(output).not.toContain('Line 3'); expect(output).toContain('Line 4'); expect(output).toContain('Line 5'); unmount(); }); it('truncates ANSI output when maxLines is provided, even if availableTerminalHeight is undefined', async () => { const ansiResult: AnsiOutput = Array.from({ length: 50 }, (_, i) => [ { text: `Line ${i + 1}`, fg: '', bg: '', bold: false, italic: false, underline: false, dim: false, inverse: false, }, ]); const { lastFrame, waitUntilReady, unmount } = await renderWithProviders( , { config: makeFakeConfig({ useAlternateBuffer: false }), settings: createMockSettings({ ui: { useAlternateBuffer: false } }), uiState: { constrainHeight: true }, }, ); await waitUntilReady(); const output = lastFrame(); // It SHOULD truncate to 25 lines because maxLines is provided expect(output).not.toContain('Line 1'); expect(output).toContain('Line 50'); unmount(); }); });