Skip to main content
Glama
CreateVisit.test.tsx6.11 kB
// SPDX-FileCopyrightText: Copyright Orangebot, Inc. and Medplum contributors // SPDX-License-Identifier: Apache-2.0 import { MantineProvider } from '@mantine/core'; import { Notifications } from '@mantine/notifications'; import { act, render, screen, waitFor } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { MedplumProvider } from '@medplum/react'; import { MockClient, HomerSimpson } from '@medplum/mock'; import { MemoryRouter } from 'react-router'; import { describe, expect, test, vi, beforeEach } from 'vitest'; import type { Patient } from '@medplum/fhirtypes'; import type { SlotInfo } from 'react-big-calendar'; import { CreateVisit } from './CreateVisit'; describe('CreateVisit', () => { let medplum: MockClient; let mockPatient: Patient; let mockSlotInfo: SlotInfo; beforeEach(async () => { medplum = new MockClient(); vi.clearAllMocks(); mockPatient = { ...HomerSimpson, id: 'patient-1', name: [{ given: ['Homer'], family: 'Simpson' }], }; const startDate = new Date('2024-01-15T10:00:00Z'); const endDate = new Date('2024-01-15T10:30:00Z'); mockSlotInfo = { start: startDate, end: endDate, slots: [], action: 'select', } as SlotInfo; await medplum.createResource(mockPatient); medplum.getProfile = vi.fn().mockResolvedValue({ resourceType: 'Practitioner', id: 'practitioner-1', name: [{ given: ['Dr.'], family: 'Smith' }], }); }); const setup = (appointmentSlot?: SlotInfo): ReturnType<typeof render> => { return render( <MemoryRouter> <MedplumProvider medplum={medplum}> <MantineProvider> <Notifications /> <CreateVisit appointmentSlot={appointmentSlot} /> </MantineProvider> </MedplumProvider> </MemoryRouter> ); }; describe('Rendering', () => { test('renders form with all required fields', async () => { await act(async () => { setup(mockSlotInfo); }); await waitFor(() => { expect(screen.getByLabelText(/Patient/i)).toBeInTheDocument(); expect(screen.getByLabelText(/Start Time/i)).toBeInTheDocument(); expect(screen.getByLabelText(/End Time/i)).toBeInTheDocument(); expect(screen.getByLabelText(/Class/i)).toBeInTheDocument(); expect(screen.getByLabelText(/Care template/i)).toBeInTheDocument(); expect(screen.getByRole('button', { name: /Create Visit/i })).toBeInTheDocument(); }); }); test('renders without date/time when appointmentSlot is not provided', async () => { await act(async () => { setup(undefined); }); await waitFor(() => { expect(screen.getByRole('button', { name: /Create Visit/i })).toBeInTheDocument(); }); }); }); describe('Form Validation', () => { test('shows error notification when submitting with missing required fields', async () => { const user = userEvent.setup(); await act(async () => { setup(mockSlotInfo); }); await waitFor(() => { expect(screen.getByRole('button', { name: /Create Visit/i })).toBeInTheDocument(); }); await act(async () => { const submitButton = screen.getByRole('button', { name: /Create Visit/i }); await user.click(submitButton); }); await waitFor(() => { expect(screen.getByText(/Please fill out required fields/i)).toBeInTheDocument(); }); }); test('does not proceed with submission when required fields are missing', async () => { const user = userEvent.setup(); await act(async () => { setup(mockSlotInfo); }); await waitFor(() => { expect(screen.getByRole('button', { name: /Create Visit/i })).toBeInTheDocument(); }); await act(async () => { const submitButton = screen.getByRole('button', { name: /Create Visit/i }); await user.click(submitButton); }); // Verify error notification is shown instead of proceeding // Use getAllByText since notifications may persist from previous tests await waitFor(() => { const notifications = screen.getAllByText(/Please fill out required fields/i); expect(notifications.length).toBeGreaterThan(0); }); }); }); describe('PlanDefinition Actions', () => { test('does not display included tasks card initially', async () => { await act(async () => { setup(mockSlotInfo); }); await waitFor(() => { expect(screen.queryByText(/Included Tasks/i)).not.toBeInTheDocument(); }); }); }); describe('Date/Time Formatting', () => { test('updates formatted date/time when appointmentSlot changes', async () => { const { rerender } = await act(async () => { return setup(mockSlotInfo); }); await waitFor(() => { expect(screen.getByText(/Jan.*15.*2024/i)).toBeInTheDocument(); }); const newSlotInfo: SlotInfo = { start: new Date('2024-02-20T14:00:00Z'), end: new Date('2024-02-20T14:30:00Z'), slots: [], action: 'select', } as SlotInfo; await act(async () => { rerender( <MemoryRouter> <MedplumProvider medplum={medplum}> <MantineProvider> <Notifications /> <CreateVisit appointmentSlot={newSlotInfo} /> </MantineProvider> </MedplumProvider> </MemoryRouter> ); }); await waitFor(() => { expect(screen.getByText(/Feb.*20.*2024/i)).toBeInTheDocument(); }); }); }); describe('Button State', () => { test('submit button is initially enabled', async () => { await act(async () => { setup(mockSlotInfo); }); await waitFor(() => { const submitButton = screen.getByRole('button', { name: /Create Visit/i }); expect(submitButton).toBeInTheDocument(); expect(submitButton).not.toBeDisabled(); }); }); }); });

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