|
1 | | -import { describe, it, expect } from 'vitest' |
2 | | -import { render, screen } from '@testing-library/react' |
| 1 | +import { describe, it, expect, vi, beforeEach } from 'vitest' |
| 2 | +import { render, screen, fireEvent } from '@testing-library/react' |
3 | 3 | import Home from '../app/page' |
4 | 4 |
|
| 5 | +// Mock the embedding module since tests run in jsdom without real window.location.hash |
| 6 | +vi.mock('../lib/embedding', () => ({ |
| 7 | + getCountryFromHash: () => 'us', |
| 8 | + isEmbedded: () => false, |
| 9 | + updateHash: vi.fn(), |
| 10 | + getShareUrl: () => 'https://policyengine.org/us/va-reform-dashboard', |
| 11 | +})) |
| 12 | + |
| 13 | +// Mock the API client to prevent actual network calls |
| 14 | +vi.mock('../lib/api/client', () => ({ |
| 15 | + submitJob: vi.fn().mockRejectedValue(new Error('No API in tests')), |
| 16 | + pollStatus: vi.fn().mockRejectedValue(new Error('No API in tests')), |
| 17 | +})) |
| 18 | + |
5 | 19 | describe('Home page', () => { |
| 20 | + beforeEach(() => { |
| 21 | + vi.clearAllMocks() |
| 22 | + }) |
| 23 | + |
6 | 24 | it('renders the dashboard title', () => { |
7 | 25 | render(<Home />) |
8 | 26 | expect( |
9 | 27 | screen.getByText('Virginia tax and benefit reform calculator'), |
10 | 28 | ).toBeDefined() |
11 | 29 | }) |
12 | 30 |
|
13 | | - it('renders household configuration section', () => { |
14 | | - render(<Home />) |
15 | | - expect(screen.getByText('Household configuration')).toBeDefined() |
16 | | - }) |
17 | | - |
18 | 31 | it('renders both tab options', () => { |
19 | 32 | render(<Home />) |
20 | 33 | expect(screen.getByText('Household impact')).toBeDefined() |
21 | 34 | expect(screen.getByText('Statewide impact')).toBeDefined() |
22 | 35 | }) |
23 | 36 |
|
24 | | - it('renders reform parameter sections', () => { |
| 37 | + it('renders all sidebar input group labels', () => { |
25 | 38 | render(<Home />) |
| 39 | + expect(screen.getByText('Household configuration')).toBeDefined() |
26 | 40 | expect(screen.getByText('Federal income tax')).toBeDefined() |
27 | 41 | expect(screen.getByText('Federal Child Tax Credit')).toBeDefined() |
28 | 42 | expect(screen.getByText('Federal EITC')).toBeDefined() |
29 | 43 | expect(screen.getByText('Virginia income tax')).toBeDefined() |
30 | 44 | expect(screen.getByText('Virginia EITC')).toBeDefined() |
31 | 45 | }) |
| 46 | + |
| 47 | + it('renders filing status label and options', () => { |
| 48 | + render(<Home />) |
| 49 | + expect(screen.getByText('Filing status')).toBeDefined() |
| 50 | + // The select should have the three options available |
| 51 | + expect(screen.getByText('Single')).toBeDefined() |
| 52 | + }) |
| 53 | + |
| 54 | + it('shows spouse age field only when filing status is joint', () => { |
| 55 | + render(<Home />) |
| 56 | + // Initially single - no spouse field |
| 57 | + expect(screen.queryByText("Spouse's age")).toBeNull() |
| 58 | + |
| 59 | + // Change to joint by finding and changing the select |
| 60 | + const selects = document.querySelectorAll('select') |
| 61 | + const filingStatusSelect = Array.from(selects).find((s) => |
| 62 | + Array.from(s.options).some((o) => o.value === 'joint'), |
| 63 | + ) |
| 64 | + expect(filingStatusSelect).toBeDefined() |
| 65 | + fireEvent.change(filingStatusSelect!, { target: { value: 'joint' } }) |
| 66 | + expect(screen.getByText("Spouse's age")).toBeDefined() |
| 67 | + }) |
| 68 | + |
| 69 | + it('renders dependent age inputs based on count', () => { |
| 70 | + render(<Home />) |
| 71 | + // No dependent age fields initially |
| 72 | + expect(screen.queryByText('Dependent 1')).toBeNull() |
| 73 | + |
| 74 | + // Find the number of dependents input |
| 75 | + const numDepsInput = document.querySelector( |
| 76 | + 'input[type="number"][min="0"][max="10"]', |
| 77 | + ) as HTMLInputElement |
| 78 | + expect(numDepsInput).toBeDefined() |
| 79 | + |
| 80 | + // Set to 2 dependents |
| 81 | + fireEvent.change(numDepsInput, { target: { value: '2' } }) |
| 82 | + expect(screen.getByText('Dependent 1')).toBeDefined() |
| 83 | + expect(screen.getByText('Dependent 2')).toBeDefined() |
| 84 | + }) |
| 85 | + |
| 86 | + it('renders all federal bracket rate sliders', () => { |
| 87 | + render(<Home />) |
| 88 | + expect(screen.getByText('10% bracket rate')).toBeDefined() |
| 89 | + expect(screen.getByText('12% bracket rate')).toBeDefined() |
| 90 | + expect(screen.getByText('22% bracket rate')).toBeDefined() |
| 91 | + expect(screen.getByText('24% bracket rate')).toBeDefined() |
| 92 | + expect(screen.getByText('32% bracket rate')).toBeDefined() |
| 93 | + expect(screen.getByText('35% bracket rate')).toBeDefined() |
| 94 | + expect(screen.getByText('37% bracket rate')).toBeDefined() |
| 95 | + }) |
| 96 | + |
| 97 | + it('renders CTC input fields', () => { |
| 98 | + render(<Home />) |
| 99 | + expect(screen.getByText('CTC amount per child')).toBeDefined() |
| 100 | + expect(screen.getByText('Phase-out threshold (single)')).toBeDefined() |
| 101 | + expect(screen.getByText('Phase-out threshold (joint)')).toBeDefined() |
| 102 | + expect(screen.getByText('Max refundable (ACTC) per child')).toBeDefined() |
| 103 | + expect(screen.getByText('Make fully refundable')).toBeDefined() |
| 104 | + }) |
| 105 | + |
| 106 | + it('renders EITC fields for all child counts', () => { |
| 107 | + render(<Home />) |
| 108 | + expect(screen.getByText('Max EITC (0 children)')).toBeDefined() |
| 109 | + expect(screen.getByText('Max EITC (1 child)')).toBeDefined() |
| 110 | + expect(screen.getByText('Max EITC (2 children)')).toBeDefined() |
| 111 | + expect(screen.getByText('Max EITC (3+ children)')).toBeDefined() |
| 112 | + }) |
| 113 | + |
| 114 | + it('renders VA tax rate sliders', () => { |
| 115 | + render(<Home />) |
| 116 | + expect(screen.getByText('Bracket 1 rate ($0-$3k)')).toBeDefined() |
| 117 | + expect(screen.getByText('Bracket 2 rate ($3k-$5k)')).toBeDefined() |
| 118 | + expect(screen.getByText('Bracket 3 rate ($5k-$17k)')).toBeDefined() |
| 119 | + expect(screen.getByText('Bracket 4 rate ($17k+)')).toBeDefined() |
| 120 | + }) |
| 121 | + |
| 122 | + it('renders VA EITC match rate slider', () => { |
| 123 | + render(<Home />) |
| 124 | + expect(screen.getByText('VA EITC match rate')).toBeDefined() |
| 125 | + }) |
| 126 | + |
| 127 | + it('renders VA standard deduction field', () => { |
| 128 | + render(<Home />) |
| 129 | + expect(screen.getByText('Standard deduction')).toBeDefined() |
| 130 | + }) |
| 131 | + |
| 132 | + it('renders PolicyEngine logo in header', () => { |
| 133 | + render(<Home />) |
| 134 | + expect(screen.getByAltText('PolicyEngine')).toBeDefined() |
| 135 | + }) |
32 | 136 | }) |
0 commit comments