Skip to main content
Glama
ValueSetAutocomplete.test.tsx5.32 kB
// SPDX-FileCopyrightText: Copyright Orangebot, Inc. and Medplum contributors // SPDX-License-Identifier: Apache-2.0 import type { ValueSetExpansionContains } from '@medplum/fhirtypes'; import { MockClient } from '@medplum/mock'; import { MedplumProvider } from '@medplum/react-hooks'; import { AsyncAutocompleteTestIds } from '../AsyncAutocomplete/AsyncAutocomplete.utils'; import { act, fireEvent, render, screen, within } from '../test-utils/render'; import { ValueSetAutocomplete } from '../ValueSetAutocomplete/ValueSetAutocomplete'; describe('AsyncAutocomplete', () => { beforeEach(() => { jest.useFakeTimers(); }); afterEach(async () => { await act(async () => { jest.runOnlyPendingTimers(); }); jest.useRealTimers(); }); async function enterSearchString(input: HTMLInputElement, text: string): Promise<void> { await act(async () => { fireEvent.change(input, { target: { value: text } }); }); // Wait for the drop down await act(async () => { jest.advanceTimersByTime(1000); }); } async function selectOption(input: HTMLInputElement, text: string, downCount: number): Promise<void> { await enterSearchString(input, text); // Press the down arrow await act(async () => { for (let i = 0; i < downCount; i++) { fireEvent.keyDown(input, { key: 'ArrowDown', code: 'ArrowDown' }); } }); // Press "Enter" await act(async () => { fireEvent.keyDown(input, { key: 'Enter', code: 'Enter' }); }); } test('select one value', async () => { const onChange = jest.fn(); render( <MedplumProvider medplum={new MockClient()}> <ValueSetAutocomplete binding="x" onChange={onChange} placeholder="Test" maxValues={1} /> </MedplumProvider> ); const input = screen.getByPlaceholderText('Test') as HTMLInputElement; await selectOption(input, 'Display', 1); const selected = within(screen.getByTestId('selected-items')); expect(selected.queryByText('Test Display')).toBeInTheDocument(); expect(selected.queryByText('Test Display 2')).not.toBeInTheDocument(); expect(selected.queryByText('Test Display 3')).not.toBeInTheDocument(); expect(onChange).toHaveBeenCalledTimes(1); expect(onChange.mock.lastCall[0].map((c: ValueSetExpansionContains) => c.code)).toEqual(['test-code']); }); test('select multiple values', async () => { const onChange = jest.fn(); render( <MedplumProvider medplum={new MockClient()}> <ValueSetAutocomplete binding="x" onChange={onChange} placeholder="Test" maxValues={5} /> </MedplumProvider> ); const input = screen.getByPlaceholderText('Test') as HTMLInputElement; await selectOption(input, 'Display', 1); const selected = within(screen.getByTestId(AsyncAutocompleteTestIds.selectedItems)); expect(selected.queryByText('Test Display')).toBeInTheDocument(); expect(selected.queryByText('Test Display 2')).not.toBeInTheDocument(); expect(selected.queryByText('Test Display 3')).not.toBeInTheDocument(); expect(onChange).toHaveBeenCalledTimes(1); expect(onChange.mock.lastCall[0].map((c: ValueSetExpansionContains) => c.code)).toEqual(['test-code']); await act(async () => { fireEvent.keyDown(input, { key: 'ArrowDown', code: 'ArrowDown' }); fireEvent.keyDown(input, { key: 'ArrowDown', code: 'ArrowDown' }); fireEvent.keyDown(input, { key: 'Enter', code: 'Enter' }); }); expect(selected.queryByText('Test Display')).toBeInTheDocument(); expect(selected.queryByText('Test Display 2')).not.toBeInTheDocument(); expect(selected.queryByText('Test Display 3')).toBeInTheDocument(); expect(onChange).toHaveBeenCalledTimes(2); expect(onChange.mock.lastCall[0].map((c: ValueSetExpansionContains) => c.code)).toEqual([ 'test-code', 'test-code-3', ]); // It would be nice to search by name, aka aria-label, but that doesn't work as expected // when hidden is true. See https://github.com/testing-library/dom-testing-library/issues/846 const closeButtons = screen.getAllByRole('button', { hidden: true }); // two items selected and the clear all button expect(closeButtons).toHaveLength(3); // Remove Test Display 3 await act(async () => { fireEvent.click(closeButtons[1]); }); expect(onChange).toHaveBeenCalledTimes(3); expect(onChange.mock.lastCall[0].map((c: ValueSetExpansionContains) => c.code)).toEqual(['test-code']); // Remove Test Display await act(async () => { fireEvent.click(closeButtons[0]); }); expect(onChange).toHaveBeenCalledTimes(4); expect(onChange.mock.lastCall[0].map((c: ValueSetExpansionContains) => c.code)).toEqual([]); }); test('empty search', async () => { const onChange = jest.fn(); render( <MedplumProvider medplum={new MockClient()}> <ValueSetAutocomplete binding="x" onChange={onChange} placeholder="Test" maxValues={1} /> </MedplumProvider> ); const input = screen.getByPlaceholderText('Test') as HTMLInputElement; await enterSearchString(input, ''); const options = screen.getByTestId(AsyncAutocompleteTestIds.options); expect(options).not.toHaveAttribute('hidden'); expect(onChange).toHaveBeenCalledTimes(0); }); });

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/medplum/medplum'

If you have feedback or need assistance with the MCP directory API, please join our Discord server