diff --git a/src/settings/RealTimeUpdatesSettings.tsx b/src/settings/RealTimeUpdatesSettings.tsx index a4218b5a..f587baa8 100644 --- a/src/settings/RealTimeUpdatesSettings.tsx +++ b/src/settings/RealTimeUpdatesSettings.tsx @@ -5,6 +5,7 @@ import { SimpleCard } from '../utils/SimpleCard'; import { FormText } from '../utils/forms/FormText'; import { LabeledFormGroup } from '../utils/forms/LabeledFormGroup'; import { Settings } from './reducers/settings'; +import { useDomId } from '../utils/helpers/hooks'; interface RealTimeUpdatesProps { settings: Settings; @@ -16,39 +17,46 @@ const intervalValue = (interval?: number) => (!interval ? '' : `${interval}`); export const RealTimeUpdatesSettings = ( { settings: { realTimeUpdates }, toggleRealTimeUpdates, setRealTimeUpdatesInterval }: RealTimeUpdatesProps, -) => ( - - - - Enable or disable real-time updates. - - Real-time updates are currently being {realTimeUpdates.enabled ? 'processed' : 'ignored'}. - - - - - setRealTimeUpdatesInterval(Number(target.value))} - /> - {realTimeUpdates.enabled && ( - - {realTimeUpdates.interval !== undefined && realTimeUpdates.interval > 0 && ( - - Updates will be reflected in the UI every {realTimeUpdates.interval} minute{realTimeUpdates.interval > 1 && 's'}. - - )} - {!realTimeUpdates.interval && 'Updates will be reflected in the UI as soon as they happen.'} - - )} - - -); +) => { + const inputId = useDomId(); + + return ( + + + + Enable or disable real-time updates. + + Real-time updates are currently being {realTimeUpdates.enabled ? 'processed' : 'ignored'}. + + + + + setRealTimeUpdatesInterval(Number(target.value))} + /> + {realTimeUpdates.enabled && ( + + {realTimeUpdates.interval !== undefined && realTimeUpdates.interval > 0 && ( + + Updates will be reflected in the UI + every {realTimeUpdates.interval} minute{realTimeUpdates.interval > 1 && 's'}. + + )} + {!realTimeUpdates.interval && 'Updates will be reflected in the UI as soon as they happen.'} + + )} + + + ); +}; diff --git a/test/settings/RealTimeUpdatesSettings.test.tsx b/test/settings/RealTimeUpdatesSettings.test.tsx index 170c1de9..947e27e4 100644 --- a/test/settings/RealTimeUpdatesSettings.test.tsx +++ b/test/settings/RealTimeUpdatesSettings.test.tsx @@ -1,64 +1,54 @@ -import { shallow, ShallowWrapper } from 'enzyme'; +import userEvent from '@testing-library/user-event'; +import { render, screen } from '@testing-library/react'; import { Mock } from 'ts-mockery'; -import { Input } from 'reactstrap'; -import { FormText } from '../../src/utils/forms/FormText'; import { RealTimeUpdatesSettings as RealTimeUpdatesSettingsOptions, Settings, } from '../../src/settings/reducers/settings'; import { RealTimeUpdatesSettings } from '../../src/settings/RealTimeUpdatesSettings'; -import { ToggleSwitch } from '../../src/utils/ToggleSwitch'; -import { LabeledFormGroup } from '../../src/utils/forms/LabeledFormGroup'; describe('', () => { const toggleRealTimeUpdates = jest.fn(); const setRealTimeUpdatesInterval = jest.fn(); - let wrapper: ShallowWrapper; - const createWrapper = (realTimeUpdates: Partial = {}) => { - const settings = Mock.of({ realTimeUpdates }); - - wrapper = shallow( + const setUp = (realTimeUpdates: Partial = {}) => ({ + user: userEvent.setup(), + ...render( ({ realTimeUpdates })} toggleRealTimeUpdates={toggleRealTimeUpdates} setRealTimeUpdatesInterval={setRealTimeUpdatesInterval} />, - ); - - return wrapper; - }; + ), + }); afterEach(jest.clearAllMocks); - afterEach(() => wrapper?.unmount()); it('renders enabled real time updates as expected', () => { - const wrapper = createWrapper({ enabled: true }); - const toggle = wrapper.find(ToggleSwitch); - const label = wrapper.find(LabeledFormGroup); - const input = wrapper.find(Input); - const formText = wrapper.find(FormText); + setUp({ enabled: true }); - expect(toggle.prop('checked')).toEqual(true); - expect(toggle.html()).toContain('processed'); - expect(toggle.html()).not.toContain('ignored'); - expect(label.prop('labelClassName')).not.toContain('text-muted'); - expect(input.prop('disabled')).toEqual(false); - expect(formText).toHaveLength(2); + expect(screen.getByLabelText(/^Enable or disable real-time updates./)).toBeChecked(); + expect(screen.getByText(/^Real-time updates are currently being/)).toHaveTextContent('processed'); + expect(screen.getByText(/^Real-time updates are currently being/)).not.toHaveTextContent('ignored'); + expect(screen.getByText('Real-time updates frequency (in minutes):')).not.toHaveAttribute( + 'class', + expect.stringContaining('text-muted'), + ); + expect(screen.getByLabelText('Real-time updates frequency (in minutes):')).not.toHaveAttribute('disabled'); + expect(screen.getByText('Updates will be reflected in the UI as soon as they happen.')).toBeInTheDocument(); }); it('renders disabled real time updates as expected', () => { - const wrapper = createWrapper({ enabled: false }); - const toggle = wrapper.find(ToggleSwitch); - const label = wrapper.find(LabeledFormGroup); - const input = wrapper.find(Input); - const formText = wrapper.find(FormText); + setUp({ enabled: false }); - expect(toggle.prop('checked')).toEqual(false); - expect(toggle.html()).not.toContain('processed'); - expect(toggle.html()).toContain('ignored'); - expect(label.prop('labelClassName')).toContain('text-muted'); - expect(input.prop('disabled')).toEqual(true); - expect(formText).toHaveLength(1); + expect(screen.getByLabelText(/^Enable or disable real-time updates./)).not.toBeChecked(); + expect(screen.getByText(/^Real-time updates are currently being/)).not.toHaveTextContent('processed'); + expect(screen.getByText(/^Real-time updates are currently being/)).toHaveTextContent('ignored'); + expect(screen.getByText('Real-time updates frequency (in minutes):')).toHaveAttribute( + 'class', + expect.stringContaining('text-muted'), + ); + expect(screen.getByLabelText('Real-time updates frequency (in minutes):')).toHaveAttribute('disabled'); + expect(screen.queryByText('Updates will be reflected in the UI as soon as they happen.')).not.toBeInTheDocument(); }); it.each([ @@ -67,43 +57,35 @@ describe('', () => { [10, 'minutes'], [100, 'minutes'], ])('shows expected children when interval is greater than 0', (interval, minutesWord) => { - const wrapper = createWrapper({ enabled: true, interval }); - const span = wrapper.find('span'); - const input = wrapper.find(Input); + setUp({ enabled: true, interval }); - expect(span).toHaveLength(1); - expect(span.html()).toEqual( - `Updates will be reflected in the UI every ${interval} ${minutesWord}.`, + expect(screen.getByText(/^Updates will be reflected in the UI every/)).toHaveTextContent( + `${interval} ${minutesWord}`, ); - expect(input.prop('value')).toEqual(`${interval}`); + expect(screen.getByLabelText('Real-time updates frequency (in minutes):')).toHaveValue(interval); + expect(screen.queryByText('Updates will be reflected in the UI as soon as they happen.')).not.toBeInTheDocument(); }); it.each([[undefined], [0]])('shows expected children when interval is 0 or undefined', (interval) => { - const wrapper = createWrapper({ enabled: true, interval }); - const span = wrapper.find('span'); - const formText = wrapper.find(FormText).at(1); - const input = wrapper.find(Input); + setUp({ enabled: true, interval }); - expect(span).toHaveLength(0); - expect(formText.html()).toContain('Updates will be reflected in the UI as soon as they happen.'); - expect(input.prop('value')).toEqual(''); + expect(screen.queryByText(/^Updates will be reflected in the UI every/)).not.toBeInTheDocument(); + expect(screen.getByText('Updates will be reflected in the UI as soon as they happen.')).toBeInTheDocument(); }); - it('updates real time updates on input change', () => { - const wrapper = createWrapper(); - const input = wrapper.find(Input); + it('updates real time updates when typing on input', async () => { + const { user } = setUp({ enabled: true }); expect(setRealTimeUpdatesInterval).not.toHaveBeenCalled(); - input.simulate('change', { target: { value: '10' } }); - expect(setRealTimeUpdatesInterval).toHaveBeenCalledWith(10); + await user.type(screen.getByLabelText('Real-time updates frequency (in minutes):'), '5'); + expect(setRealTimeUpdatesInterval).toHaveBeenCalledWith(5); }); - it('toggles real time updates on switch change', () => { - const wrapper = createWrapper(); - const toggle = wrapper.find(ToggleSwitch); + it('toggles real time updates on switch change', async () => { + const { user } = setUp({ enabled: true }); expect(toggleRealTimeUpdates).not.toHaveBeenCalled(); - toggle.simulate('change'); + await user.click(screen.getByText(/^Enable or disable real-time updates./)); expect(toggleRealTimeUpdates).toHaveBeenCalled(); }); }); diff --git a/test/settings/ShortUrlsListSettings.test.tsx b/test/settings/ShortUrlsListSettings.test.tsx index c167ea4e..104fd6f3 100644 --- a/test/settings/ShortUrlsListSettings.test.tsx +++ b/test/settings/ShortUrlsListSettings.test.tsx @@ -1,52 +1,43 @@ -import { shallow, ShallowWrapper } from 'enzyme'; +import userEvent from '@testing-library/user-event'; +import { render, screen } from '@testing-library/react'; import { Mock } from 'ts-mockery'; -import { - DEFAULT_SHORT_URLS_ORDERING, - Settings, - ShortUrlsListSettings as ShortUrlsSettings, -} from '../../src/settings/reducers/settings'; +import { Settings, ShortUrlsListSettings as ShortUrlsSettings } from '../../src/settings/reducers/settings'; import { ShortUrlsListSettings } from '../../src/settings/ShortUrlsListSettings'; -import { OrderingDropdown } from '../../src/utils/OrderingDropdown'; import { ShortUrlsOrder } from '../../src/short-urls/data'; describe('', () => { - let wrapper: ShallowWrapper; const setSettings = jest.fn(); - const createWrapper = (shortUrlsList?: ShortUrlsSettings) => { - wrapper = shallow( + const setUp = (shortUrlsList?: ShortUrlsSettings) => ({ + user: userEvent.setup(), + ...render( ({ shortUrlsList })} setShortUrlsListSettings={setSettings} />, - ); + ), + }); - return wrapper; - }; - - afterEach(() => wrapper?.unmount()); afterEach(jest.clearAllMocks); it.each([ - [undefined, DEFAULT_SHORT_URLS_ORDERING], - [{}, DEFAULT_SHORT_URLS_ORDERING], - [{ defaultOrdering: {} }, {}], - [{ defaultOrdering: { field: 'longUrl', dir: 'DESC' } as ShortUrlsOrder }, { field: 'longUrl', dir: 'DESC' }], - [{ defaultOrdering: { field: 'visits', dir: 'ASC' } as ShortUrlsOrder }, { field: 'visits', dir: 'ASC' }], + [undefined, 'Order by: Created at - DESC'], + [{}, 'Order by: Created at - DESC'], + [{ defaultOrdering: {} }, 'Order by...'], + [{ defaultOrdering: { field: 'longUrl', dir: 'DESC' } as ShortUrlsOrder }, 'Order by: Long URL - DESC'], + [{ defaultOrdering: { field: 'visits', dir: 'ASC' } as ShortUrlsOrder }, 'Order by: Visits - ASC'], ])('shows expected ordering', (shortUrlsList, expectedOrder) => { - const wrapper = createWrapper(shortUrlsList); - const dropdown = wrapper.find(OrderingDropdown); - - expect(dropdown.prop('order')).toEqual(expectedOrder); + setUp(shortUrlsList); + expect(screen.getByRole('button')).toHaveTextContent(expectedOrder); }); it.each([ - [undefined, undefined], - ['longUrl', 'ASC'], - ['visits', undefined], - ['title', 'DESC'], - ])('invokes setSettings when ordering changes', (field, dir) => { - const wrapper = createWrapper(); - const dropdown = wrapper.find(OrderingDropdown); + ['Clear selection', undefined, undefined], + ['Long URL', 'longUrl', 'ASC'], + ['Visits', 'visits', 'ASC'], + ['Title', 'title', 'ASC'], + ])('invokes setSettings when ordering changes', async (name, field, dir) => { + const { user } = setUp(); expect(setSettings).not.toHaveBeenCalled(); - dropdown.simulate('change', field, dir); + await user.click(screen.getByRole('button')); + await user.click(screen.getByRole('menuitem', { name })); expect(setSettings).toHaveBeenCalledWith({ defaultOrdering: { field, dir } }); }); });