import { render } from 'utils/testRenderer'; import { screen, waitFor } from '@testing-library/react'; import { usePersistentTableState } from './usePersistentTableState'; import { Route, Routes } from 'react-router-dom'; import { createLocalStorage } from '../utils/createLocalStorage'; import { ArrayParam, NumberParam, StringParam } from 'use-query-params'; import { FilterItemParam } from '../utils/serializeQueryParams'; type TestComponentProps = { keyName: string; queryParamsDefinition: Record; nonPersistentParams?: string[]; }; function TestComponent({ keyName, queryParamsDefinition, nonPersistentParams, }: TestComponentProps) { const [tableState, setTableState] = usePersistentTableState( keyName, queryParamsDefinition, nonPersistentParams, ); return ( {tableState.query} {Object.keys(tableState).join(',')} } /> ); } describe('usePersistentTableState', () => { it('initializes correctly from URL', async () => { createLocalStorage('testKey', {}); render( , { route: '/my-url?query=initialUrl' }, ); expect(screen.getByTestId('state-value').textContent).toBe( 'initialUrl', ); expect(window.location.href).toContain('my-url?query=initialUrl'); }); it('initializes correctly from localStorage', async () => { createLocalStorage('testKey', {}).setValue({ query: 'initialStorage' }); render( , { route: '/my-url' }, ); expect(screen.getByTestId('state-value').textContent).toBe( 'initialStorage', ); expect(window.location.href).toContain('my-url?query=initialStorage'); }); it('initializes correctly from localStorage with complex decoder', async () => { createLocalStorage('testKey', {}).setValue({ query: 'initialStorage', filterItem: { operator: 'IS', values: ['default'], }, columns: ['a', 'b'], }); render( , { route: '/my-url' }, ); expect(screen.getByTestId('state-value').textContent).toBe( 'initialStorage', ); expect(window.location.href).toContain( 'my-url?query=initialStorage&filterItem=IS%3Adefault&columns=a&columns=b', ); }); it('initializes correctly from localStorage and URL', async () => { createLocalStorage('testKey', {}).setValue({ query: 'initialStorage' }); render( , { route: '/my-url?query=initialUrl' }, ); expect(screen.getByTestId('state-value').textContent).toBe( 'initialUrl', ); expect(window.location.href).toContain('my-url?query=initialUrl'); }); it('partially updates the state on button click', async () => { createLocalStorage('testKey', {}).setValue({ query: 'before', other: 'other', }); render( , { route: '/my-url' }, ); expect(screen.getByTestId('state-value').textContent).toBe('before'); (await screen.findByText('Update State')).click(); expect((await screen.findByTestId('state-value')).textContent).toBe( 'after', ); expect(window.location.href).toContain( 'my-url?query=after&other=other', ); await waitFor(() => { const { value } = createLocalStorage('testKey', {}); expect(value).toStrictEqual({ query: 'after', other: 'other', }); }); }); it('omits offset in local storage', async () => { createLocalStorage('testKey', {}).setValue({ query: 'before' }); render( , { route: '/my-url' }, ); screen.getByText('Update Offset').click(); screen.getByText('Update State').click(); expect(window.location.href).toContain('my-url?query=after&offset=0'); await waitFor(() => { const { value } = createLocalStorage('testKey', {}); expect(value).toStrictEqual({ query: 'after' }); }); }); it('resets offset to 0 on state update', async () => { createLocalStorage('testKey', {}).setValue({ query: 'before' }); render( , { route: '/my-url?query=before&offset=10' }, ); expect(window.location.href).toContain('my-url?query=before&offset=10'); screen.getByText('Update State').click(); await waitFor(() => { expect(window.location.href).toContain( 'my-url?query=after&offset=0', ); expect(window.location.href).not.toContain('offset=10'); }); }); it('does not reset offset to 0 without offset decoder', async () => { createLocalStorage('testKey', {}).setValue({ query: 'before' }); render( , { route: '/my-url?query=before&offset=10' }, ); expect(window.location.href).toContain('my-url?query=before&offset=10'); screen.getByText('Update State').click(); await waitFor(() => { expect(window.location.href).toContain( 'my-url?query=after&offset=10', ); }); }); it('maintains key order', async () => { createLocalStorage('testKey', {}); render( , { route: '/my-url?another=another&query=initialUrl' }, ); expect(screen.getByTestId('state-keys').textContent).toBe( 'another,query,ignore', ); await waitFor(() => { const { value } = createLocalStorage('testKey', {}); expect(Object.keys(value)).toStrictEqual(['another', 'query']); }); }); });