import { fireEvent, render, screen, waitFor } from '@testing-library/react' import * as React from 'react' import { createReactI18nextMock } from '@/test/i18n-mock' import InputWithCopy from './index' // Create a mock function that we can track using vi.hoisted const mockCopyToClipboard = vi.hoisted(() => vi.fn(() => true)) // Mock the copy-to-clipboard library vi.mock('copy-to-clipboard', () => ({ default: mockCopyToClipboard, })) // Mock the i18n hook with custom translations for test assertions vi.mock('react-i18next', () => createReactI18nextMock({ 'operation.copy': 'Copy', 'operation.copied': 'Copied', 'overview.appInfo.embedded.copy': 'Copy', 'overview.appInfo.embedded.copied': 'Copied', })) // Mock es-toolkit/compat debounce vi.mock('es-toolkit/compat', () => ({ debounce: (fn: any) => fn, })) describe('InputWithCopy component', () => { beforeEach(() => { vi.clearAllMocks() mockCopyToClipboard.mockClear() }) it('renders correctly with default props', () => { const mockOnChange = vi.fn() render() const input = screen.getByDisplayValue('test value') const copyButton = screen.getByRole('button') expect(input).toBeInTheDocument() expect(copyButton).toBeInTheDocument() }) it('hides copy button when showCopyButton is false', () => { const mockOnChange = vi.fn() render() const input = screen.getByDisplayValue('test value') const copyButton = screen.queryByRole('button') expect(input).toBeInTheDocument() expect(copyButton).not.toBeInTheDocument() }) it('copies input value when copy button is clicked', async () => { const mockOnChange = vi.fn() render() const copyButton = screen.getByRole('button') fireEvent.click(copyButton) expect(mockCopyToClipboard).toHaveBeenCalledWith('test value') }) it('copies custom value when copyValue prop is provided', async () => { const mockOnChange = vi.fn() render() const copyButton = screen.getByRole('button') fireEvent.click(copyButton) expect(mockCopyToClipboard).toHaveBeenCalledWith('custom copy value') }) it('calls onCopy callback when copy button is clicked', async () => { const onCopyMock = vi.fn() const mockOnChange = vi.fn() render() const copyButton = screen.getByRole('button') fireEvent.click(copyButton) expect(onCopyMock).toHaveBeenCalledWith('test value') }) it('shows copied state after successful copy', async () => { const mockOnChange = vi.fn() render() const copyButton = screen.getByRole('button') fireEvent.click(copyButton) // Hover over the button to trigger tooltip fireEvent.mouseEnter(copyButton) // Check if the tooltip shows "Copied" state await waitFor(() => { expect(screen.getByText('Copied')).toBeInTheDocument() }, { timeout: 2000 }) }) it('passes through all input props correctly', () => { const mockOnChange = vi.fn() render( , ) const input = screen.getByDisplayValue('test value') expect(input).toHaveAttribute('placeholder', 'Custom placeholder') expect(input).toBeDisabled() expect(input).toHaveAttribute('readonly') expect(input).toHaveClass('custom-class') }) it('handles empty value correctly', () => { const mockOnChange = vi.fn() render() const input = screen.getByRole('textbox') const copyButton = screen.getByRole('button') expect(input).toBeInTheDocument() expect(copyButton).toBeInTheDocument() fireEvent.click(copyButton) expect(mockCopyToClipboard).toHaveBeenCalledWith('') }) it('maintains focus on input after copy', async () => { const mockOnChange = vi.fn() render() const input = screen.getByDisplayValue('test value') const copyButton = screen.getByRole('button') input.focus() expect(input).toHaveFocus() fireEvent.click(copyButton) // Input should maintain focus after copy expect(input).toHaveFocus() }) })