2025-11-22 08:17:29 +05:30
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright 2025 Google LLC
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
import { render } from '../../test-utils/render.js';
|
|
|
|
|
import { GeminiRespondingSpinner } from './GeminiRespondingSpinner.js';
|
|
|
|
|
import { describe, it, expect, vi, beforeEach } from 'vitest';
|
|
|
|
|
import { useStreamingContext } from '../contexts/StreamingContext.js';
|
2026-02-25 15:31:35 -08:00
|
|
|
import { Text, useIsScreenReaderEnabled } from 'ink';
|
2025-11-22 08:17:29 +05:30
|
|
|
import { StreamingState } from '../types.js';
|
|
|
|
|
import {
|
|
|
|
|
SCREEN_READER_LOADING,
|
|
|
|
|
SCREEN_READER_RESPONDING,
|
|
|
|
|
} from '../textConstants.js';
|
|
|
|
|
|
|
|
|
|
vi.mock('../contexts/StreamingContext.js');
|
2026-02-18 16:46:50 -08:00
|
|
|
vi.mock('ink', async (importOriginal) => {
|
|
|
|
|
const actual = await importOriginal<typeof import('ink')>();
|
2025-11-22 08:17:29 +05:30
|
|
|
return {
|
|
|
|
|
...actual,
|
|
|
|
|
useIsScreenReaderEnabled: vi.fn(),
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-25 15:31:35 -08:00
|
|
|
vi.mock('./GeminiSpinner.js', () => ({
|
|
|
|
|
GeminiSpinner: ({ altText }: { altText?: string }) => (
|
|
|
|
|
<Text>GeminiSpinner {altText}</Text>
|
|
|
|
|
),
|
2025-12-03 11:07:23 -08:00
|
|
|
}));
|
|
|
|
|
|
2025-11-22 08:17:29 +05:30
|
|
|
describe('GeminiRespondingSpinner', () => {
|
|
|
|
|
const mockUseStreamingContext = vi.mocked(useStreamingContext);
|
|
|
|
|
const mockUseIsScreenReaderEnabled = vi.mocked(useIsScreenReaderEnabled);
|
|
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
|
vi.clearAllMocks();
|
|
|
|
|
mockUseIsScreenReaderEnabled.mockReturnValue(false);
|
|
|
|
|
});
|
|
|
|
|
|
2026-02-18 16:46:50 -08:00
|
|
|
it('renders spinner when responding', async () => {
|
2025-11-22 08:17:29 +05:30
|
|
|
mockUseStreamingContext.mockReturnValue(StreamingState.Responding);
|
2026-02-18 16:46:50 -08:00
|
|
|
const { lastFrame, waitUntilReady, unmount } = render(
|
|
|
|
|
<GeminiRespondingSpinner />,
|
|
|
|
|
);
|
|
|
|
|
await waitUntilReady();
|
2026-02-25 15:31:35 -08:00
|
|
|
expect(lastFrame()).toContain('GeminiSpinner');
|
2026-02-18 16:46:50 -08:00
|
|
|
unmount();
|
2025-11-22 08:17:29 +05:30
|
|
|
});
|
|
|
|
|
|
2026-02-18 16:46:50 -08:00
|
|
|
it('renders screen reader text when responding and screen reader enabled', async () => {
|
2025-11-22 08:17:29 +05:30
|
|
|
mockUseStreamingContext.mockReturnValue(StreamingState.Responding);
|
|
|
|
|
mockUseIsScreenReaderEnabled.mockReturnValue(true);
|
2026-02-18 16:46:50 -08:00
|
|
|
const { lastFrame, waitUntilReady, unmount } = render(
|
|
|
|
|
<GeminiRespondingSpinner />,
|
|
|
|
|
);
|
|
|
|
|
await waitUntilReady();
|
2025-11-22 08:17:29 +05:30
|
|
|
expect(lastFrame()).toContain(SCREEN_READER_RESPONDING);
|
2026-02-18 16:46:50 -08:00
|
|
|
unmount();
|
2025-11-22 08:17:29 +05:30
|
|
|
});
|
|
|
|
|
|
2026-02-18 16:46:50 -08:00
|
|
|
it('renders nothing when not responding and no non-responding display', async () => {
|
2025-11-22 08:17:29 +05:30
|
|
|
mockUseStreamingContext.mockReturnValue(StreamingState.Idle);
|
2026-02-18 16:46:50 -08:00
|
|
|
const { lastFrame, waitUntilReady, unmount } = render(
|
|
|
|
|
<GeminiRespondingSpinner />,
|
|
|
|
|
);
|
|
|
|
|
await waitUntilReady();
|
|
|
|
|
expect(lastFrame({ allowEmpty: true })).toBe('');
|
|
|
|
|
unmount();
|
2025-11-22 08:17:29 +05:30
|
|
|
});
|
|
|
|
|
|
2026-02-18 16:46:50 -08:00
|
|
|
it('renders non-responding display when provided', async () => {
|
2025-11-22 08:17:29 +05:30
|
|
|
mockUseStreamingContext.mockReturnValue(StreamingState.Idle);
|
2026-02-18 16:46:50 -08:00
|
|
|
const { lastFrame, waitUntilReady, unmount } = render(
|
2025-11-22 08:17:29 +05:30
|
|
|
<GeminiRespondingSpinner nonRespondingDisplay="Waiting..." />,
|
|
|
|
|
);
|
2026-02-18 16:46:50 -08:00
|
|
|
await waitUntilReady();
|
2025-11-22 08:17:29 +05:30
|
|
|
expect(lastFrame()).toContain('Waiting...');
|
2026-02-18 16:46:50 -08:00
|
|
|
unmount();
|
2025-11-22 08:17:29 +05:30
|
|
|
});
|
|
|
|
|
|
2026-02-18 16:46:50 -08:00
|
|
|
it('renders screen reader loading text when non-responding display provided and screen reader enabled', async () => {
|
2025-11-22 08:17:29 +05:30
|
|
|
mockUseStreamingContext.mockReturnValue(StreamingState.Idle);
|
|
|
|
|
mockUseIsScreenReaderEnabled.mockReturnValue(true);
|
2026-02-18 16:46:50 -08:00
|
|
|
const { lastFrame, waitUntilReady, unmount } = render(
|
2025-11-22 08:17:29 +05:30
|
|
|
<GeminiRespondingSpinner nonRespondingDisplay="Waiting..." />,
|
|
|
|
|
);
|
2026-02-18 16:46:50 -08:00
|
|
|
await waitUntilReady();
|
2025-11-22 08:17:29 +05:30
|
|
|
expect(lastFrame()).toContain(SCREEN_READER_LOADING);
|
2026-02-18 16:46:50 -08:00
|
|
|
unmount();
|
2025-11-22 08:17:29 +05:30
|
|
|
});
|
|
|
|
|
});
|