Files
SimpleFinanceDash/frontend/src/components/InlineEditCell.test.tsx

192 lines
5.5 KiB
TypeScript

import { render, screen, fireEvent } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { describe, it, expect, vi, beforeEach } from 'vitest'
import { InlineEditCell } from './InlineEditCell'
// Wrap in a table/row to satisfy semantic HTML for TableCell
function Wrapper({ children }: { children: React.ReactNode }) {
return (
<table>
<tbody>
<tr>{children}</tr>
</tbody>
</table>
)
}
describe('InlineEditCell', () => {
const defaultProps = {
value: 42.5,
currency: 'EUR',
onSave: vi.fn().mockResolvedValue(undefined),
}
beforeEach(() => {
vi.clearAllMocks()
})
it('renders formatted currency value in display mode', () => {
render(
<Wrapper>
<InlineEditCell {...defaultProps} />
</Wrapper>
)
// Should show formatted value (42.5 in EUR → some formatted string containing 42)
expect(screen.getByText(/42/)).toBeInTheDocument()
})
it('enters edit mode on click', async () => {
const user = userEvent.setup()
render(
<Wrapper>
<InlineEditCell {...defaultProps} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
// After click, an input should be visible
expect(screen.getByRole('spinbutton')).toBeInTheDocument()
})
it('calls onSave with parsed number on blur', async () => {
const user = userEvent.setup()
const onSave = vi.fn().mockResolvedValue(undefined)
render(
<Wrapper>
<InlineEditCell {...defaultProps} onSave={onSave} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
const input = screen.getByRole('spinbutton')
await user.clear(input)
await user.type(input, '100')
fireEvent.blur(input)
expect(onSave).toHaveBeenCalledWith(100)
})
it('does not call onSave when value unchanged', async () => {
const user = userEvent.setup()
const onSave = vi.fn().mockResolvedValue(undefined)
render(
<Wrapper>
<InlineEditCell {...defaultProps} onSave={onSave} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
// Don't change the value, just blur
const input = screen.getByRole('spinbutton')
fireEvent.blur(input)
expect(onSave).not.toHaveBeenCalled()
})
it('calls onSave on Enter key', async () => {
const user = userEvent.setup()
const onSave = vi.fn().mockResolvedValue(undefined)
render(
<Wrapper>
<InlineEditCell {...defaultProps} onSave={onSave} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
const input = screen.getByRole('spinbutton')
await user.clear(input)
await user.type(input, '99')
await user.keyboard('{Enter}')
expect(onSave).toHaveBeenCalledWith(99)
})
// New tests for pencil icon and callbacks
it('renders a pencil icon in display mode', () => {
render(
<Wrapper>
<InlineEditCell {...defaultProps} />
</Wrapper>
)
// Pencil icon from lucide-react renders as an SVG — query by its test-id or aria role
// The Pencil icon should be present in the DOM (opacity-0, shown on hover via CSS)
const pencilIcon = document.querySelector('svg[data-testid="pencil-icon"]') ||
document.querySelector('.lucide-pencil')
expect(pencilIcon).not.toBeNull()
})
it('calls onSaveSuccess callback after successful save', async () => {
const user = userEvent.setup()
const onSave = vi.fn().mockResolvedValue(undefined)
const onSaveSuccess = vi.fn()
render(
<Wrapper>
<InlineEditCell {...defaultProps} onSave={onSave} onSaveSuccess={onSaveSuccess} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
const input = screen.getByRole('spinbutton')
await user.clear(input)
await user.type(input, '100')
fireEvent.blur(input)
// Wait for async operations
await vi.waitFor(() => {
expect(onSaveSuccess).toHaveBeenCalledTimes(1)
})
})
it('calls onSaveError and reverts value when onSave rejects', async () => {
const user = userEvent.setup()
const onSave = vi.fn().mockRejectedValue(new Error('Save failed'))
const onSaveError = vi.fn()
render(
<Wrapper>
<InlineEditCell {...defaultProps} value={42.5} onSave={onSave} onSaveError={onSaveError} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
const input = screen.getByRole('spinbutton')
await user.clear(input)
await user.type(input, '999')
fireEvent.blur(input)
// Wait for async operations
await vi.waitFor(() => {
expect(onSaveError).toHaveBeenCalledTimes(1)
})
// Value should revert back to original — display mode shows original value
await vi.waitFor(() => {
expect(screen.getByText(/42/)).toBeInTheDocument()
})
})
it('does not call onSaveSuccess when value is unchanged', async () => {
const user = userEvent.setup()
const onSave = vi.fn().mockResolvedValue(undefined)
const onSaveSuccess = vi.fn()
render(
<Wrapper>
<InlineEditCell {...defaultProps} onSave={onSave} onSaveSuccess={onSaveSuccess} />
</Wrapper>
)
const span = screen.getByText(/42/)
await user.click(span)
// Don't change the value, just blur
const input = screen.getByRole('spinbutton')
fireEvent.blur(input)
expect(onSave).not.toHaveBeenCalled()
expect(onSaveSuccess).not.toHaveBeenCalled()
})
})