Skip to main content
Glama
ResourceInput.test.tsx6.2 kB
// SPDX-FileCopyrightText: Copyright Orangebot, Inc. and Medplum contributors // SPDX-License-Identifier: Apache-2.0 import { formatAddress } from '@medplum/core'; import type { Patient, Resource } from '@medplum/fhirtypes'; import { MockClient } from '@medplum/mock'; import { MedplumProvider } from '@medplum/react-hooks'; import { forwardRef } from 'react'; import type { AsyncAutocompleteOption } from '../AsyncAutocomplete/AsyncAutocomplete'; import { act, fireEvent, render, screen } from '../test-utils/render'; import type { ResourceInputProps } from './ResourceInput'; import { ResourceInput } from './ResourceInput'; const medplum = new MockClient(); function setup(args: ResourceInputProps): void { render( <MedplumProvider medplum={medplum}> <ResourceInput {...args} /> </MedplumProvider> ); } describe('ResourceInput', () => { beforeEach(() => { jest.useFakeTimers(); }); afterEach(async () => { await act(async () => { jest.runOnlyPendingTimers(); }); jest.useRealTimers(); }); test('Renders empty', () => { setup({ resourceType: 'Patient', name: 'foo', placeholder: 'Test', }); expect(screen.getByPlaceholderText('Test')).toBeInTheDocument(); }); test('Renders default value', async () => { await act(async () => { setup({ resourceType: 'Patient', name: 'foo', defaultValue: { reference: 'Patient/123', }, placeholder: 'Test', }); }); expect(await screen.findByText('Homer Simpson')).toBeInTheDocument(); expect(screen.getByText('Homer Simpson')).toBeInTheDocument(); }); test('Use autocomplete', async () => { setup({ resourceType: 'Patient', name: 'foo', placeholder: 'Test', }); const input = screen.getByPlaceholderText('Test') as HTMLInputElement; // Enter "Simpson" await act(async () => { fireEvent.change(input, { target: { value: 'Simpson' } }); }); // Wait for the drop down await act(async () => { jest.advanceTimersByTime(1000); }); expect(screen.getByText('Homer Simpson')).toBeDefined(); }); test('Call onChange', async () => { const onChange = jest.fn(); setup({ resourceType: 'Patient', name: 'foo', placeholder: 'Test', onChange, }); const input = screen.getByPlaceholderText('Test') as HTMLInputElement; // Enter "Simpson" await act(async () => { fireEvent.change(input, { target: { value: 'Simpson' } }); }); // Wait for the drop down await act(async () => { jest.advanceTimersByTime(1000); }); // Press the down arrow await act(async () => { fireEvent.keyDown(input, { key: 'ArrowDown', code: 'ArrowDown' }); }); // Press "Enter" await act(async () => { fireEvent.keyDown(input, { key: 'Enter', code: 'Enter' }); }); expect(screen.getByText('Homer Simpson')).toBeInTheDocument(); expect(onChange).toHaveBeenCalled(); }); test('Handle invalid reference default value', async () => { await act(async () => { setup({ resourceType: 'Patient', name: 'foo', defaultValue: { reference: '', }, placeholder: 'Test', }); }); expect(await screen.findByPlaceholderText('Test')).toBeInTheDocument(); expect(screen.getByPlaceholderText('Test')).toBeInTheDocument(); }); test('Clear button calls onChange', async () => { const onChange = jest.fn(); await act(async () => { setup({ resourceType: 'Patient', name: 'foo', defaultValue: { reference: 'Patient/123' }, placeholder: 'Test', onChange, }); }); expect(await screen.findByText('Homer Simpson')).toBeInTheDocument(); expect(screen.getByText('Homer Simpson')).toBeInTheDocument(); const nameSpan = screen.getByText('Homer Simpson'); const clearButton = nameSpan.parentElement?.childNodes[1] as HTMLImageElement; expect(clearButton).toBeInTheDocument(); await act(async () => { fireEvent.click(clearButton); }); expect(onChange).toHaveBeenCalledWith(undefined); }); test('Clear all button calls onChange', async () => { const onChange = jest.fn(); await act(async () => { setup({ resourceType: 'Patient', name: 'foo', defaultValue: { reference: 'Patient/123' }, placeholder: 'Test', onChange, }); }); expect(await screen.findByText('Homer Simpson')).toBeInTheDocument(); expect(screen.getByText('Homer Simpson')).toBeInTheDocument(); const clearAllButton = screen.getByTitle('Clear all') as HTMLImageElement; expect(clearAllButton).toBeInTheDocument(); await act(async () => { fireEvent.click(clearAllButton); }); expect(onChange).toHaveBeenCalledWith(undefined); }); test('Custom item component', async () => { const MyTestItemComponent = forwardRef<HTMLDivElement, AsyncAutocompleteOption<Resource>>( ({ label, resource, active: _active, ...others }: AsyncAutocompleteOption<Resource>, ref) => { const address = (resource as Patient).address?.[0]; return ( <div ref={ref} {...others}> <div>{label}</div> <div>{address && formatAddress(address)}</div> <div>extra text</div> </div> ); } ); setup({ resourceType: 'Patient', name: 'foo', placeholder: 'Test', itemComponent: MyTestItemComponent, onChange: jest.fn(), }); const input = screen.getByPlaceholderText('Test') as HTMLInputElement; // Enter "Simpson" await act(async () => { fireEvent.change(input, { target: { value: 'Simpson' } }); }); // Wait for the drop down await act(async () => { jest.advanceTimersByTime(1000); }); expect(await screen.findByText('Homer Simpson')).toBeInTheDocument(); expect(await screen.findByText('742 Evergreen Terrace, Springfield, IL, 12345')).toBeInTheDocument(); expect((await screen.findAllByText('extra text'))?.[0]).toBeInTheDocument(); }); });

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