@kenyaemr/esm-express-workflow-app 5.4.3 → 5.4.4-pre.100
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +7 -12
- package/dist/1074.js +1 -0
- package/dist/1074.js.map +1 -0
- package/dist/12.js +17 -0
- package/dist/12.js.map +1 -0
- package/dist/1311.js +1 -0
- package/dist/1311.js.map +1 -0
- package/dist/1323.js +1 -0
- package/dist/1323.js.map +1 -0
- package/dist/1469.js +1 -0
- package/dist/1469.js.map +1 -0
- package/dist/1506.js +13 -0
- package/dist/1506.js.map +1 -0
- package/dist/1562.js +1 -0
- package/dist/1562.js.map +1 -0
- package/dist/1760.js +1 -0
- package/dist/1760.js.map +1 -0
- package/dist/1780.js +1 -0
- package/dist/1780.js.map +1 -0
- package/dist/1804.js +1 -0
- package/dist/1804.js.map +1 -0
- package/dist/1884.js +1 -0
- package/dist/1884.js.map +1 -0
- package/dist/1972.js +1 -0
- package/dist/1972.js.map +1 -0
- package/dist/1990.js +1 -0
- package/dist/1990.js.map +1 -0
- package/dist/2016.js +1 -0
- package/dist/2016.js.map +1 -0
- package/dist/2024.js +1 -0
- package/dist/2024.js.map +1 -0
- package/dist/2153.js +1 -0
- package/dist/2153.js.map +1 -0
- package/dist/216.js +1 -0
- package/dist/216.js.map +1 -0
- package/dist/2225.js +1 -0
- package/dist/2225.js.map +1 -0
- package/dist/2294.js +1 -0
- package/dist/2294.js.map +1 -0
- package/dist/2345.js +1 -0
- package/dist/2345.js.map +1 -0
- package/dist/2499.js +1 -0
- package/dist/2499.js.map +1 -0
- package/dist/2500.js +1 -0
- package/dist/2500.js.map +1 -0
- package/dist/2586.js +1 -0
- package/dist/2586.js.map +1 -0
- package/dist/2625.js +1 -0
- package/dist/2625.js.map +1 -0
- package/dist/2685.js +1 -0
- package/dist/2685.js.map +1 -0
- package/dist/2809.js +1 -0
- package/dist/2809.js.map +1 -0
- package/dist/2851.js +1 -0
- package/dist/2851.js.map +1 -0
- package/dist/2881.js +1 -0
- package/dist/2881.js.map +1 -0
- package/dist/2948.js +1 -0
- package/dist/2948.js.map +1 -0
- package/dist/2968.js +1 -0
- package/dist/2968.js.map +1 -0
- package/dist/2978.js +1 -0
- package/dist/2978.js.map +1 -0
- package/dist/2998.js +1 -0
- package/dist/2998.js.map +1 -0
- package/dist/3089.js +1 -0
- package/dist/3089.js.map +1 -0
- package/dist/3548.js +1 -0
- package/dist/3548.js.map +1 -0
- package/dist/3567.js +1 -0
- package/dist/3567.js.map +1 -0
- package/dist/3569.js +1 -0
- package/dist/3569.js.map +1 -0
- package/dist/3571.js +1 -0
- package/dist/3571.js.map +1 -0
- package/dist/3691.js +1 -0
- package/dist/3691.js.map +1 -0
- package/dist/3730.js +1 -0
- package/dist/3730.js.map +1 -0
- package/dist/3923.js +1 -0
- package/dist/3923.js.map +1 -0
- package/dist/3963.js +1 -0
- package/dist/3963.js.map +1 -0
- package/dist/4024.js +1 -0
- package/dist/4024.js.map +1 -0
- package/dist/405.js +1 -0
- package/dist/405.js.map +1 -0
- package/dist/4071.js +1 -0
- package/dist/4071.js.map +1 -0
- package/dist/4271.js +1 -0
- package/dist/4271.js.map +1 -0
- package/dist/4296.js +1 -0
- package/dist/4296.js.map +1 -0
- package/dist/4337.js +1 -0
- package/dist/4337.js.map +1 -0
- package/dist/4432.js +1 -0
- package/dist/4432.js.map +1 -0
- package/dist/4581.js +1 -0
- package/dist/4581.js.map +1 -0
- package/dist/4637.js +11 -0
- package/dist/4637.js.map +1 -0
- package/dist/4666.js +1 -0
- package/dist/4666.js.map +1 -0
- package/dist/4680.js +1 -0
- package/dist/4680.js.map +1 -0
- package/dist/4735.js +1 -0
- package/dist/4735.js.map +1 -0
- package/dist/4737.js +1 -0
- package/dist/4737.js.map +1 -0
- package/dist/4744.js +1 -0
- package/dist/4744.js.map +1 -0
- package/dist/4795.js +1 -0
- package/dist/4795.js.map +1 -0
- package/dist/4813.js +2 -0
- package/dist/4813.js.map +1 -0
- package/dist/4818.js +1 -0
- package/dist/4818.js.map +1 -0
- package/dist/4858.js +1 -0
- package/dist/4858.js.map +1 -0
- package/dist/487.js +1 -0
- package/dist/487.js.map +1 -0
- package/dist/4970.js +1 -0
- package/dist/4970.js.map +1 -0
- package/dist/5038.js +1 -0
- package/dist/5038.js.map +1 -0
- package/dist/5202.js +1 -0
- package/dist/5202.js.map +1 -0
- package/dist/5491.js +1 -0
- package/dist/5491.js.map +1 -0
- package/dist/5592.js +1 -0
- package/dist/5592.js.map +1 -0
- package/dist/5669.js +1 -0
- package/dist/5669.js.map +1 -0
- package/dist/586.js +1 -0
- package/dist/586.js.map +1 -0
- package/dist/5932.js +1 -0
- package/dist/5932.js.map +1 -0
- package/dist/5995.js +1 -0
- package/dist/5995.js.map +1 -0
- package/dist/6258.js +1 -0
- package/dist/6258.js.map +1 -0
- package/dist/629.js +1 -0
- package/dist/629.js.map +1 -0
- package/dist/6328.js +1 -0
- package/dist/6328.js.map +1 -0
- package/dist/6355.js +1 -0
- package/dist/6355.js.map +1 -0
- package/dist/6419.js +1 -0
- package/dist/6419.js.map +1 -0
- package/dist/644.js +1 -0
- package/dist/644.js.map +1 -0
- package/dist/6456.js +1 -0
- package/dist/6466.js +3 -0
- package/dist/6466.js.map +1 -0
- package/dist/655.js +1 -0
- package/dist/655.js.map +1 -0
- package/dist/6798.js +66 -0
- package/dist/6798.js.map +1 -0
- package/dist/6910.js +1 -0
- package/dist/6910.js.map +1 -0
- package/dist/6925.js +1 -0
- package/dist/6925.js.map +1 -0
- package/dist/70.js +1 -0
- package/dist/70.js.map +1 -0
- package/dist/7201.js +1 -0
- package/dist/7201.js.map +1 -0
- package/dist/7234.js +1 -0
- package/dist/7234.js.map +1 -0
- package/dist/7261.js +1 -0
- package/dist/7261.js.map +1 -0
- package/dist/7326.js +1 -0
- package/dist/7359.js +1 -0
- package/dist/7487.js +1 -0
- package/dist/7487.js.map +1 -0
- package/dist/7591.js +1 -0
- package/dist/7591.js.map +1 -0
- package/dist/7607.js +1 -0
- package/dist/7701.js +1 -0
- package/dist/7701.js.map +1 -0
- package/dist/7717.js +1 -0
- package/dist/7717.js.map +1 -0
- package/dist/7739.js +1 -0
- package/dist/7739.js.map +1 -0
- package/dist/7788.js +1 -0
- package/dist/7788.js.map +1 -0
- package/dist/7819.js +1 -0
- package/dist/7819.js.map +1 -0
- package/dist/7971.js +1 -0
- package/dist/7971.js.map +1 -0
- package/dist/7983.js +1 -0
- package/dist/7983.js.map +1 -0
- package/dist/807.js +1 -0
- package/dist/807.js.map +1 -0
- package/dist/8159.js +7 -0
- package/dist/8159.js.map +1 -0
- package/dist/8338.js +1 -0
- package/dist/8338.js.map +1 -0
- package/dist/845.js +1 -0
- package/dist/845.js.map +1 -0
- package/dist/8570.js +1 -0
- package/dist/8570.js.map +1 -0
- package/dist/8661.js +1 -0
- package/dist/8661.js.map +1 -0
- package/dist/87.js +1 -0
- package/dist/87.js.map +1 -0
- package/dist/8727.js +1 -0
- package/dist/8766.js +1 -0
- package/dist/8766.js.map +1 -0
- package/dist/8828.js +1 -0
- package/dist/8828.js.map +1 -0
- package/dist/8860.js +1 -0
- package/dist/8860.js.map +1 -0
- package/dist/8911.js +1 -0
- package/dist/8911.js.map +1 -0
- package/dist/8930.js +1 -0
- package/dist/8930.js.map +1 -0
- package/dist/8971.js +1 -0
- package/dist/8971.js.map +1 -0
- package/dist/9124.js +1 -0
- package/dist/9124.js.map +1 -0
- package/dist/9157.js +1 -0
- package/dist/9157.js.map +1 -0
- package/dist/9182.js +1 -0
- package/dist/921.js +1 -0
- package/dist/921.js.map +1 -0
- package/dist/9212.js +1 -0
- package/dist/9212.js.map +1 -0
- package/dist/9255.js +1 -0
- package/dist/9255.js.map +1 -0
- package/dist/9257.js +1 -0
- package/dist/9257.js.map +1 -0
- package/dist/9316.js +1 -0
- package/dist/9316.js.map +1 -0
- package/dist/9333.js +1 -0
- package/dist/9333.js.map +1 -0
- package/dist/9404.js +1 -0
- package/dist/9404.js.map +1 -0
- package/dist/9446.js +1 -0
- package/dist/9446.js.map +1 -0
- package/dist/9447.js +1 -0
- package/dist/9447.js.map +1 -0
- package/dist/9449.js +1 -0
- package/dist/9449.js.map +1 -0
- package/dist/9535.js +1 -0
- package/dist/9535.js.map +1 -0
- package/dist/9606.js +1 -0
- package/dist/9606.js.map +1 -0
- package/dist/973.js +1 -0
- package/dist/973.js.map +1 -0
- package/dist/9845.js +1 -0
- package/dist/9845.js.map +1 -0
- package/dist/kenyaemr-esm-express-workflow-app.js +5 -5
- package/dist/kenyaemr-esm-express-workflow-app.js.buildmanifest.json +3004 -175
- package/dist/kenyaemr-esm-express-workflow-app.js.map +1 -1
- package/dist/main.js +5 -114
- package/dist/main.js.map +1 -1
- package/dist/routes.json +1 -1
- package/package.json +6 -7
- package/rspack.config.js +1 -1
- package/src/components/accounting/index.tsx +11 -10
- package/src/components/admissions/index.tsx +11 -10
- package/src/components/appointments/index.ts +10 -8
- package/src/components/consultation/clinical-encounter/encounter-details.component.tsx +9 -9
- package/src/components/consultation/clinical-encounter/encounter-details.test.tsx +19 -18
- package/src/components/consultation/consultation-context.tsx +19 -0
- package/src/components/consultation/consultation-summary-cards.component.tsx +124 -0
- package/src/components/consultation/consultation.component.tsx +41 -268
- package/src/components/consultation/consultation.resource.ts +67 -24
- package/src/components/consultation/consultation.scss +5 -0
- package/src/components/consultation/consultation.utils.tsx +222 -0
- package/src/components/consultation/dashboard.component.tsx +0 -2
- package/src/components/consultation/index.ts +27 -20
- package/src/components/facility-dashboard/index.tsx +10 -9
- package/src/components/laboratory/index.ts +11 -10
- package/src/components/laboratory/lab-table.component.tsx +22 -8
- package/src/components/laboratory/laboratory-tabs.component.tsx +2 -2
- package/src/components/mch/dashboard.component.tsx +0 -2
- package/src/components/mch/index.tsx +10 -9
- package/src/components/mch/mch.consultation.component.tsx +27 -15
- package/src/components/mch/mch.triage.component.tsx +42 -33
- package/src/components/pharmacy/index.ts +20 -18
- package/src/components/pharmacy/orders/pharmacy-orders.component.tsx +35 -13
- package/src/components/pharmacy/orders/pharmacy-orders.test.tsx +8 -7
- package/src/components/procedures/index.ts +11 -10
- package/src/components/procedures/procedures-table.component.tsx +44 -13
- package/src/components/procedures/procedures-tabs.component.tsx +2 -2
- package/src/components/radiology-and-imaging/index.ts +14 -10
- package/src/components/radiology-and-imaging/radiology-and-imaging-table.component.tsx +35 -15
- package/src/components/radiology-and-imaging/radiology-and-imaging.component.tsx +2 -2
- package/src/components/registration/card/HIE-card/hie-card.component.tsx +32 -44
- package/src/components/registration/card/Local-card/local-card.component.tsx +22 -24
- package/src/components/registration/dependants/dependants.component.tsx +32 -60
- package/src/components/registration/dependants/dependants.resource.ts +79 -50
- package/src/components/registration/helper/index.ts +4 -0
- package/src/components/registration/index.ts +10 -9
- package/src/components/registration/search-bar/search-bar.resource.ts +32 -93
- package/src/components/registration/start-visit-form/hooks/useDefaultFacilityLocation.tsx +15 -0
- package/src/components/registration/start-visit-form/hooks/useDefaultVisitLocation.tsx +24 -0
- package/src/components/registration/start-visit-form/hooks/useOfflineVisitType.tsx +18 -0
- package/src/components/registration/start-visit-form/hooks/useRecommendedVisitTypes.tsx +36 -0
- package/src/components/registration/start-visit-form/hooks/useVisitAttributeType.tsx +102 -0
- package/src/components/registration/start-visit-form/overflow-menu-extension/overflow-menu-item.extension.tsx +55 -0
- package/src/components/registration/start-visit-form/overflow-menu-extension/overflow-menu-item.scss +3 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/base-visit-type.component.tsx +126 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/base-visit-type.scss +75 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/exported-visit-form.workspace.tsx +743 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/location-selector.component.tsx +86 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/recommended-visit-type.component.tsx +32 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/visit-attribute-type.component.tsx +257 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/visit-attribute-type.scss +5 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/visit-date-time.component.tsx +193 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/visit-form.resource.ts +383 -0
- package/src/components/registration/start-visit-form/start-visit-workspace/visit-form.scss +166 -0
- package/src/components/registration/start-visit-form/visit-form-workspace/visit-form.workspace.tsx +55 -0
- package/src/components/reports/index.ts +10 -9
- package/src/components/triage/dashboard.component.tsx +0 -2
- package/src/components/triage/index.ts +10 -9
- package/src/components/triage/triage.component.tsx +92 -42
- package/src/components/triage/triage.resource.ts +56 -50
- package/src/components/triage/triage.scss +6 -0
- package/src/config-schema.ts +91 -0
- package/src/hooks/useServiceQueues.tsx +95 -25
- package/src/index.ts +32 -13
- package/src/routes.json +53 -254
- package/src/shared/orders/OrdersTabs.tsx +8 -3
- package/src/shared/patient-chart/patient-chart.resources.ts +8 -12
- package/src/shared/patient-chart/patient-summary-dashboard/patient-summary-dashboard.component.tsx +1 -2
- package/src/shared/pin-put/pinput.component.test.tsx +7 -6
- package/src/shared/queue/queue-entry/queue-entry-table.component.tsx +99 -88
- package/src/shared/queue/queue-entry/queue-entry-table.scss +4 -0
- package/src/shared/queue/queue-summary-cards.component.tsx +32 -0
- package/src/shared/queue/queue-tab.component.tsx +182 -70
- package/src/shared/queue/queue-tab.scss +1 -0
- package/src/shared/queue/queue-workflow-context.tsx +90 -0
- package/src/shared/tabs/extension-tabs.component.tsx +9 -6
- package/src/shared/utils/index.ts +1 -2
- package/src/types/index.ts +17 -2
- package/translations/am.json +64 -13
- package/translations/en.json +59 -8
- package/translations/sw.json +58 -7
- package/dist/127.js +0 -1
- package/dist/200.js +0 -1
- package/dist/200.js.map +0 -1
- package/dist/24.js +0 -1
- package/dist/24.js.map +0 -1
- package/dist/267.js +0 -1
- package/dist/267.js.map +0 -1
- package/dist/285.js +0 -1
- package/dist/285.js.map +0 -1
- package/dist/329.js +0 -1
- package/dist/329.js.map +0 -1
- package/dist/40.js +0 -1
- package/dist/466.js +0 -1
- package/dist/466.js.map +0 -1
- package/dist/472.js +0 -66
- package/dist/472.js.map +0 -1
- package/dist/490.js +0 -1
- package/dist/490.js.map +0 -1
- package/dist/54.js +0 -1
- package/dist/54.js.map +0 -1
- package/dist/689.js +0 -1
- package/dist/689.js.map +0 -1
- package/dist/697.js +0 -1
- package/dist/697.js.map +0 -1
- package/dist/704.js +0 -1
- package/dist/704.js.map +0 -1
- package/dist/710.js +0 -1
- package/dist/710.js.map +0 -1
- package/dist/729.js +0 -17
- package/dist/729.js.map +0 -1
- package/dist/805.js +0 -37
- package/dist/805.js.map +0 -1
- package/dist/847.js +0 -1
- package/dist/847.js.map +0 -1
- package/dist/85.js +0 -1
- package/dist/85.js.map +0 -1
- package/dist/882.js +0 -1
- package/dist/91.js +0 -1
- package/dist/91.js.map +0 -1
- package/dist/916.js +0 -1
- package/dist/994.js +0 -1
- package/dist/994.js.map +0 -1
- package/dist/998.js +0 -1
- package/dist/998.js.map +0 -1
- package/jest.config.js +0 -3
- package/src/shared/patient-chart/patient-chart.component.tsx +0 -61
- package/src/shared/patient-chart/patient-chart.scss +0 -15
- package/src/shared/patient-chart/useInitialize.ts +0 -24
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import React, { useEffect, useState } from 'react';
|
|
2
|
+
import classNames from 'classnames';
|
|
3
|
+
import isEmpty from 'lodash-es/isEmpty';
|
|
4
|
+
import { ComboBox } from '@carbon/react';
|
|
5
|
+
import { useTranslation } from 'react-i18next';
|
|
6
|
+
import { type Control, Controller } from 'react-hook-form';
|
|
7
|
+
import {
|
|
8
|
+
type Location,
|
|
9
|
+
type OpenmrsResource,
|
|
10
|
+
useConfig,
|
|
11
|
+
useFeatureFlag,
|
|
12
|
+
useLocations,
|
|
13
|
+
useSession,
|
|
14
|
+
} from '@openmrs/esm-framework';
|
|
15
|
+
import { type VisitFormData } from './visit-form.resource';
|
|
16
|
+
import { useDefaultFacilityLocation } from '../hooks/useDefaultFacilityLocation';
|
|
17
|
+
import { useDefaultVisitLocation } from '../hooks/useDefaultVisitLocation';
|
|
18
|
+
import styles from './visit-form.scss';
|
|
19
|
+
import { ExpressWorkflowConfig } from '../../../../config-schema';
|
|
20
|
+
|
|
21
|
+
interface LocationSelectorProps {
|
|
22
|
+
control: Control<VisitFormData>;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
const LocationSelector: React.FC<LocationSelectorProps> = ({ control }) => {
|
|
26
|
+
const { t } = useTranslation();
|
|
27
|
+
const config = useConfig<ExpressWorkflowConfig>();
|
|
28
|
+
const [searchTerm, setSearchTerm] = useState('');
|
|
29
|
+
const sessionLocation = useSession().sessionLocation;
|
|
30
|
+
const isEmrApiModuleInstalled = useFeatureFlag('emrapi-module');
|
|
31
|
+
const defaultVisitLocation = useDefaultVisitLocation(
|
|
32
|
+
sessionLocation,
|
|
33
|
+
config.restrictByVisitLocationTag && isEmrApiModuleInstalled,
|
|
34
|
+
);
|
|
35
|
+
const locations = useLocations(
|
|
36
|
+
config.restrictByVisitLocationTag && isEmrApiModuleInstalled ? 'Visit Location' : null,
|
|
37
|
+
searchTerm,
|
|
38
|
+
);
|
|
39
|
+
const { defaultFacility, isLoading: loadingDefaultFacility } = useDefaultFacilityLocation();
|
|
40
|
+
const disableChangingVisitLocation = config?.disableChangingVisitLocation;
|
|
41
|
+
const locationsToShow: Array<OpenmrsResource> =
|
|
42
|
+
!loadingDefaultFacility && !isEmpty(defaultFacility) ? [defaultFacility] : locations ? locations : [];
|
|
43
|
+
|
|
44
|
+
const handleSearch = (searchString: string) => {
|
|
45
|
+
setSearchTerm(searchString);
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
useEffect(() => {
|
|
49
|
+
if (config.restrictByVisitLocationTag && !isEmrApiModuleInstalled) {
|
|
50
|
+
console.warn('EMR API module is not installed. Visit location will not be restricted by location tag.');
|
|
51
|
+
}
|
|
52
|
+
}, [config.restrictByVisitLocationTag, isEmrApiModuleInstalled]);
|
|
53
|
+
|
|
54
|
+
return (
|
|
55
|
+
<section data-testid="combo">
|
|
56
|
+
<div className={styles.sectionTitle}>{t('visitLocation', 'Visit Location')}</div>
|
|
57
|
+
<div className={classNames(styles.selectContainer, styles.sectionField)}>
|
|
58
|
+
{!disableChangingVisitLocation ? (
|
|
59
|
+
<Controller
|
|
60
|
+
control={control}
|
|
61
|
+
name="visitLocation"
|
|
62
|
+
render={({ field: { onBlur, onChange, value } }) => (
|
|
63
|
+
<ComboBox
|
|
64
|
+
aria-label={t('selectLocation', 'Select a location')}
|
|
65
|
+
id="location"
|
|
66
|
+
invalidText={t('required', 'Required')}
|
|
67
|
+
items={locationsToShow as Array<Location>}
|
|
68
|
+
itemToString={(location: Location | null) => location?.display ?? ''}
|
|
69
|
+
onBlur={onBlur}
|
|
70
|
+
onChange={({ selectedItem }) => onChange(selectedItem)}
|
|
71
|
+
onInputChange={(searchTerm) => handleSearch(searchTerm)}
|
|
72
|
+
readOnly={disableChangingVisitLocation}
|
|
73
|
+
selectedItem={value as Location | null}
|
|
74
|
+
titleText={t('selectLocation', 'Select a location')}
|
|
75
|
+
/>
|
|
76
|
+
)}
|
|
77
|
+
/>
|
|
78
|
+
) : (
|
|
79
|
+
<p className={styles.bodyShort02}>{defaultVisitLocation?.display}</p>
|
|
80
|
+
)}
|
|
81
|
+
</div>
|
|
82
|
+
</section>
|
|
83
|
+
);
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
export default LocationSelector;
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import { StructuredListSkeleton } from '@carbon/react';
|
|
3
|
+
import { type PatientProgram } from '@openmrs/esm-patient-common-lib';
|
|
4
|
+
import { useRecommendedVisitTypes } from '../hooks/useRecommendedVisitTypes';
|
|
5
|
+
import BaseVisitType from './base-visit-type.component';
|
|
6
|
+
|
|
7
|
+
interface RecommendedVisitTypeProp {
|
|
8
|
+
patientUuid: string;
|
|
9
|
+
patientProgramEnrollment: PatientProgram;
|
|
10
|
+
locationUuid: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
const RecommendedVisitType: React.FC<RecommendedVisitTypeProp> = ({
|
|
14
|
+
patientUuid,
|
|
15
|
+
patientProgramEnrollment,
|
|
16
|
+
locationUuid,
|
|
17
|
+
}) => {
|
|
18
|
+
const { recommendedVisitTypes, isLoading } = useRecommendedVisitTypes(
|
|
19
|
+
patientUuid,
|
|
20
|
+
patientProgramEnrollment?.uuid,
|
|
21
|
+
patientProgramEnrollment?.program?.uuid,
|
|
22
|
+
locationUuid,
|
|
23
|
+
);
|
|
24
|
+
|
|
25
|
+
return (
|
|
26
|
+
<div style={{ marginTop: '0.625rem' }}>
|
|
27
|
+
{isLoading ? <StructuredListSkeleton /> : <BaseVisitType visitTypes={recommendedVisitTypes} />}
|
|
28
|
+
</div>
|
|
29
|
+
);
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
export const MemoizedRecommendedVisitType = React.memo(RecommendedVisitType);
|
|
@@ -0,0 +1,257 @@
|
|
|
1
|
+
import React, { useEffect, useId, useMemo } from 'react';
|
|
2
|
+
import {
|
|
3
|
+
Checkbox,
|
|
4
|
+
NumberInput,
|
|
5
|
+
Select,
|
|
6
|
+
SelectItem,
|
|
7
|
+
SelectSkeleton,
|
|
8
|
+
TextArea,
|
|
9
|
+
TextInput,
|
|
10
|
+
TextInputSkeleton,
|
|
11
|
+
} from '@carbon/react';
|
|
12
|
+
import { useTranslation } from 'react-i18next';
|
|
13
|
+
import { Controller, type ControllerFieldState, type ControllerRenderProps, useFormContext } from 'react-hook-form';
|
|
14
|
+
import { OpenmrsDatePicker, useConfig } from '@openmrs/esm-framework';
|
|
15
|
+
import { useConceptAnswersForVisitAttributeType, useVisitAttributeType } from '../hooks/useVisitAttributeType';
|
|
16
|
+
import { type VisitFormData } from './visit-form.resource';
|
|
17
|
+
import styles from './visit-attribute-type.scss';
|
|
18
|
+
import { ExpressWorkflowConfig } from '../../../../config-schema';
|
|
19
|
+
|
|
20
|
+
interface VisitAttributes {
|
|
21
|
+
[uuid: string]: string;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
interface VisitAttributeTypeFieldsProps {
|
|
25
|
+
setErrorFetchingResources: React.Dispatch<
|
|
26
|
+
React.SetStateAction<{
|
|
27
|
+
blockSavingForm: boolean;
|
|
28
|
+
}>
|
|
29
|
+
>;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Evaluates a given expression using the provided visitAttributes.
|
|
34
|
+
*
|
|
35
|
+
* @param {string} expression - The expression to be evaluated. This should be a string of JavaScript code.
|
|
36
|
+
* @param {VisitAttributes} visitAttributes - An object containing visit attributes which will be used in the evaluation of the expression.
|
|
37
|
+
*
|
|
38
|
+
* @returns {boolean} - The boolean value of the result of the evaluated expression.
|
|
39
|
+
*
|
|
40
|
+
*/
|
|
41
|
+
function evaluateExpression(expression: string, visitAttributes: VisitAttributes) {
|
|
42
|
+
const func = new Function('visitAttributes', `return ${expression};`);
|
|
43
|
+
|
|
44
|
+
const result = func(visitAttributes);
|
|
45
|
+
|
|
46
|
+
return Boolean(result);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
const VisitAttributeTypeFields: React.FC<VisitAttributeTypeFieldsProps> = ({ setErrorFetchingResources }) => {
|
|
50
|
+
const { visitAttributeTypes } = useConfig<ExpressWorkflowConfig>();
|
|
51
|
+
const { control, getValues } = useFormContext<VisitFormData>();
|
|
52
|
+
|
|
53
|
+
if (visitAttributeTypes?.length) {
|
|
54
|
+
return (
|
|
55
|
+
<>
|
|
56
|
+
{visitAttributeTypes.map((attributeType) => {
|
|
57
|
+
const { visitAttributes } = getValues();
|
|
58
|
+
|
|
59
|
+
const showAttributeType = attributeType?.showWhenExpression
|
|
60
|
+
? evaluateExpression(attributeType?.showWhenExpression, visitAttributes)
|
|
61
|
+
: true;
|
|
62
|
+
|
|
63
|
+
return (
|
|
64
|
+
showAttributeType && (
|
|
65
|
+
<Controller
|
|
66
|
+
key={attributeType.uuid}
|
|
67
|
+
name={`visitAttributes.${attributeType.uuid}`}
|
|
68
|
+
control={control}
|
|
69
|
+
render={({ field, fieldState }) => (
|
|
70
|
+
<AttributeTypeField
|
|
71
|
+
key={attributeType.uuid}
|
|
72
|
+
attributeType={attributeType}
|
|
73
|
+
setErrorFetchingResources={setErrorFetchingResources}
|
|
74
|
+
fieldProps={field}
|
|
75
|
+
fieldState={fieldState}
|
|
76
|
+
/>
|
|
77
|
+
)}
|
|
78
|
+
/>
|
|
79
|
+
)
|
|
80
|
+
);
|
|
81
|
+
})}
|
|
82
|
+
</>
|
|
83
|
+
);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return null;
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
interface AttributeTypeFieldProps {
|
|
90
|
+
fieldProps: ControllerRenderProps<VisitFormData, `visitAttributes.${string}`>;
|
|
91
|
+
fieldState: ControllerFieldState;
|
|
92
|
+
attributeType: {
|
|
93
|
+
uuid: string;
|
|
94
|
+
required: boolean;
|
|
95
|
+
};
|
|
96
|
+
setErrorFetchingResources: React.Dispatch<
|
|
97
|
+
React.SetStateAction<{
|
|
98
|
+
blockSavingForm: boolean;
|
|
99
|
+
}>
|
|
100
|
+
>;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
const AttributeTypeField: React.FC<AttributeTypeFieldProps> = ({
|
|
104
|
+
attributeType,
|
|
105
|
+
setErrorFetchingResources,
|
|
106
|
+
fieldProps,
|
|
107
|
+
fieldState,
|
|
108
|
+
}) => {
|
|
109
|
+
const { uuid, required } = attributeType;
|
|
110
|
+
const { data, isLoading, error: errorFetchingVisitAttributeType } = useVisitAttributeType(uuid);
|
|
111
|
+
const {
|
|
112
|
+
answers,
|
|
113
|
+
isLoading: isLoadingAnswers,
|
|
114
|
+
error: errorFetchingVisitAttributeAnswers,
|
|
115
|
+
} = useConceptAnswersForVisitAttributeType(data?.datatypeConfig ?? null);
|
|
116
|
+
const { t } = useTranslation();
|
|
117
|
+
const baseId = useId();
|
|
118
|
+
const displayText = data?.display?.trim() || data?.name?.trim() || '--';
|
|
119
|
+
const labelText = !required ? `${displayText} (${t('optional', 'optional')})` : displayText;
|
|
120
|
+
|
|
121
|
+
const {
|
|
122
|
+
formState: { errors },
|
|
123
|
+
} = useFormContext<VisitFormData>();
|
|
124
|
+
|
|
125
|
+
useEffect(() => {
|
|
126
|
+
if (errorFetchingVisitAttributeType || errorFetchingVisitAttributeAnswers) {
|
|
127
|
+
setErrorFetchingResources((prev) => ({
|
|
128
|
+
blockSavingForm: prev?.blockSavingForm || required,
|
|
129
|
+
}));
|
|
130
|
+
}
|
|
131
|
+
}, [errorFetchingVisitAttributeAnswers, errorFetchingVisitAttributeType, required, setErrorFetchingResources]);
|
|
132
|
+
|
|
133
|
+
const fieldToRender = useMemo(() => {
|
|
134
|
+
if (isLoading) {
|
|
135
|
+
return <></>;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
if (errorFetchingVisitAttributeType) {
|
|
139
|
+
return null;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
switch (data?.datatypeClassname) {
|
|
143
|
+
case 'org.openmrs.customdatatype.datatype.ConceptDatatype':
|
|
144
|
+
if (isLoadingAnswers) {
|
|
145
|
+
return <SelectSkeleton />;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
if (errorFetchingVisitAttributeAnswers) {
|
|
149
|
+
return null;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
return (
|
|
153
|
+
<Select
|
|
154
|
+
id={`select-${baseId}`}
|
|
155
|
+
{...fieldProps}
|
|
156
|
+
labelText={labelText}
|
|
157
|
+
invalid={!!fieldState?.error?.message}
|
|
158
|
+
invalidText={fieldState?.error?.message}>
|
|
159
|
+
<SelectItem text={t('selectAnOption', 'Select an option')} value={''} />
|
|
160
|
+
{(answers ?? []).map((ans, indx) => (
|
|
161
|
+
<SelectItem key={indx} text={ans.display} value={ans.uuid} />
|
|
162
|
+
))}
|
|
163
|
+
</Select>
|
|
164
|
+
);
|
|
165
|
+
case 'org.openmrs.customdatatype.datatype.FloatDatatype':
|
|
166
|
+
return (
|
|
167
|
+
<NumberInput
|
|
168
|
+
{...fieldProps}
|
|
169
|
+
id={`number-${baseId}`}
|
|
170
|
+
label={labelText}
|
|
171
|
+
hideSteppers
|
|
172
|
+
invalid={!!fieldState?.error?.message}
|
|
173
|
+
invalidText={fieldState?.error?.message}
|
|
174
|
+
/>
|
|
175
|
+
);
|
|
176
|
+
case 'org.openmrs.customdatatype.datatype.FreeTextDatatype':
|
|
177
|
+
return (
|
|
178
|
+
<TextInput
|
|
179
|
+
{...fieldProps}
|
|
180
|
+
id={`text-${baseId}`}
|
|
181
|
+
labelText={labelText}
|
|
182
|
+
placeholder={labelText}
|
|
183
|
+
invalid={!!fieldState?.error?.message}
|
|
184
|
+
invalidText={fieldState?.error?.message}
|
|
185
|
+
/>
|
|
186
|
+
);
|
|
187
|
+
case 'org.openmrs.customdatatype.datatype.LongFreeTextDatatype':
|
|
188
|
+
return (
|
|
189
|
+
<TextArea
|
|
190
|
+
{...fieldProps}
|
|
191
|
+
id={`textarea-${baseId}`}
|
|
192
|
+
labelText={labelText}
|
|
193
|
+
invalid={!!fieldState?.error?.message}
|
|
194
|
+
invalidText={fieldState?.error?.message}
|
|
195
|
+
/>
|
|
196
|
+
);
|
|
197
|
+
case 'org.openmrs.customdatatype.datatype.BooleanDatatype':
|
|
198
|
+
return (
|
|
199
|
+
<Checkbox
|
|
200
|
+
{...fieldProps}
|
|
201
|
+
id={`checkbox-${baseId}`}
|
|
202
|
+
labelText={labelText}
|
|
203
|
+
invalid={!!fieldState?.error?.message}
|
|
204
|
+
invalidText={fieldState?.error?.message}
|
|
205
|
+
/>
|
|
206
|
+
);
|
|
207
|
+
case 'org.openmrs.customdatatype.datatype.DateDatatype':
|
|
208
|
+
return (
|
|
209
|
+
<OpenmrsDatePicker
|
|
210
|
+
{...fieldProps}
|
|
211
|
+
id={`date-${baseId}`}
|
|
212
|
+
labelText={labelText}
|
|
213
|
+
aria-invalid={!!fieldState?.error?.message}
|
|
214
|
+
invalidText={fieldState?.error?.message}
|
|
215
|
+
/>
|
|
216
|
+
);
|
|
217
|
+
default:
|
|
218
|
+
return (
|
|
219
|
+
<TextInput
|
|
220
|
+
{...fieldProps}
|
|
221
|
+
id={`text-${baseId}`}
|
|
222
|
+
labelText={labelText}
|
|
223
|
+
invalid={!!fieldState?.error?.message}
|
|
224
|
+
invalidText={fieldState?.error?.message}
|
|
225
|
+
/>
|
|
226
|
+
);
|
|
227
|
+
}
|
|
228
|
+
}, [
|
|
229
|
+
isLoading,
|
|
230
|
+
errorFetchingVisitAttributeType,
|
|
231
|
+
data?.datatypeClassname,
|
|
232
|
+
isLoadingAnswers,
|
|
233
|
+
errorFetchingVisitAttributeAnswers,
|
|
234
|
+
baseId,
|
|
235
|
+
fieldProps,
|
|
236
|
+
labelText,
|
|
237
|
+
fieldState?.error?.message,
|
|
238
|
+
t,
|
|
239
|
+
answers,
|
|
240
|
+
]);
|
|
241
|
+
|
|
242
|
+
if (isLoading) {
|
|
243
|
+
return (
|
|
244
|
+
<div className={styles.visitAttributeField}>
|
|
245
|
+
<TextInputSkeleton />
|
|
246
|
+
</div>
|
|
247
|
+
);
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
if (errorFetchingVisitAttributeType) {
|
|
251
|
+
return null;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
return <div className={styles.visitAttributeField}>{fieldToRender}</div>;
|
|
255
|
+
};
|
|
256
|
+
|
|
257
|
+
export default VisitAttributeTypeFields;
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import classNames from 'classnames';
|
|
3
|
+
import dayjs from 'dayjs';
|
|
4
|
+
import { type Control, Controller, type FieldPath, get, useFormContext, useWatch } from 'react-hook-form';
|
|
5
|
+
import { useTranslation } from 'react-i18next';
|
|
6
|
+
import { SelectItem, TimePicker, TimePickerSelect } from '@carbon/react';
|
|
7
|
+
import { type amPm } from '@openmrs/esm-patient-common-lib';
|
|
8
|
+
import { OpenmrsDatePicker, ResponsiveWrapper } from '@openmrs/esm-framework';
|
|
9
|
+
import { convertToDate, type VisitFormData } from './visit-form.resource';
|
|
10
|
+
import styles from './visit-form.scss';
|
|
11
|
+
|
|
12
|
+
const minOf = (...values: Array<number | undefined | null>) => {
|
|
13
|
+
const nums = values.filter((v): v is number => typeof v === 'number' && Number.isFinite(v));
|
|
14
|
+
return nums.length ? Math.min(...nums) : undefined;
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
const maxOf = (...values: Array<number | undefined | null>) => {
|
|
18
|
+
const nums = values.filter((v): v is number => typeof v === 'number' && Number.isFinite(v));
|
|
19
|
+
return nums.length ? Math.max(...nums) : undefined;
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
interface VisitDateTimeSectionProps {
|
|
23
|
+
control: Control<VisitFormData, any>;
|
|
24
|
+
earliestStartDate?: number;
|
|
25
|
+
firstEncounterDateTime: number;
|
|
26
|
+
lastEncounterDateTime: number;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
const VisitDateTimeSection: React.FC<VisitDateTimeSectionProps> = ({
|
|
30
|
+
control,
|
|
31
|
+
earliestStartDate,
|
|
32
|
+
firstEncounterDateTime,
|
|
33
|
+
lastEncounterDateTime,
|
|
34
|
+
}) => {
|
|
35
|
+
const { t } = useTranslation();
|
|
36
|
+
const [
|
|
37
|
+
visitStatus,
|
|
38
|
+
visitStartDate,
|
|
39
|
+
visitStartTime,
|
|
40
|
+
visitStartTimeFormat,
|
|
41
|
+
visitStopDate,
|
|
42
|
+
visitStopTime,
|
|
43
|
+
visitStopTimeFormat,
|
|
44
|
+
] = useWatch({
|
|
45
|
+
control,
|
|
46
|
+
name: [
|
|
47
|
+
'visitStatus',
|
|
48
|
+
'visitStartDate',
|
|
49
|
+
'visitStartTime',
|
|
50
|
+
'visitStartTimeFormat',
|
|
51
|
+
'visitStopDate',
|
|
52
|
+
'visitStopTime',
|
|
53
|
+
'visitStopTimeFormat',
|
|
54
|
+
],
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
const hasStopTime = 'past' === visitStatus;
|
|
58
|
+
const selectedVisitStartDateTime = convertToDate(visitStartDate, visitStartTime, visitStartTimeFormat);
|
|
59
|
+
const selectedVisitStopDateTime = convertToDate(visitStopDate, visitStopTime, visitStopTimeFormat);
|
|
60
|
+
|
|
61
|
+
if (visitStatus === 'new') {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
return (
|
|
66
|
+
<section>
|
|
67
|
+
<div className={styles.sectionTitle}>
|
|
68
|
+
{visitStatus === 'ongoing'
|
|
69
|
+
? t('visitStartDate', 'Visit start date')
|
|
70
|
+
: t('visitStartAndEndDate', 'Visit start and end date')}
|
|
71
|
+
</div>
|
|
72
|
+
<VisitDateTimeField
|
|
73
|
+
dateField={{ name: 'visitStartDate', label: t('startDate', 'Start date') }}
|
|
74
|
+
timeField={{ name: 'visitStartTime', label: t('startTime', 'Start time') }}
|
|
75
|
+
timeFormatField={{ name: 'visitStartTimeFormat', label: t('startTimeFormat', 'Start time format') }}
|
|
76
|
+
minDate={earliestStartDate}
|
|
77
|
+
maxDate={minOf(Date.now(), firstEncounterDateTime, selectedVisitStopDateTime?.getTime())}
|
|
78
|
+
/>
|
|
79
|
+
{hasStopTime && (
|
|
80
|
+
<VisitDateTimeField
|
|
81
|
+
dateField={{ name: 'visitStopDate', label: t('endDate', 'End date') }}
|
|
82
|
+
timeField={{ name: 'visitStopTime', label: t('endTime', 'End time') }}
|
|
83
|
+
timeFormatField={{ name: 'visitStopTimeFormat', label: t('endTimeFormat', 'End time format') }}
|
|
84
|
+
minDate={maxOf(lastEncounterDateTime, selectedVisitStartDateTime?.getTime())}
|
|
85
|
+
maxDate={Date.now()}
|
|
86
|
+
/>
|
|
87
|
+
)}
|
|
88
|
+
</section>
|
|
89
|
+
);
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
interface Field {
|
|
93
|
+
name: FieldPath<VisitFormData>;
|
|
94
|
+
label: string;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
interface VisitDateTimeFieldProps {
|
|
98
|
+
dateField: Field;
|
|
99
|
+
timeField: Field;
|
|
100
|
+
timeFormatField: Field;
|
|
101
|
+
minDate?: dayjs.ConfigType;
|
|
102
|
+
maxDate?: dayjs.ConfigType;
|
|
103
|
+
disabled?: boolean;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
const VisitDateTimeField: React.FC<VisitDateTimeFieldProps> = ({
|
|
107
|
+
dateField,
|
|
108
|
+
timeField,
|
|
109
|
+
timeFormatField,
|
|
110
|
+
minDate,
|
|
111
|
+
maxDate,
|
|
112
|
+
disabled,
|
|
113
|
+
}) => {
|
|
114
|
+
const {
|
|
115
|
+
control,
|
|
116
|
+
formState: { errors },
|
|
117
|
+
} = useFormContext<VisitFormData>();
|
|
118
|
+
const { t } = useTranslation();
|
|
119
|
+
|
|
120
|
+
const minDateObj = minDate ? dayjs(minDate).startOf('day') : null;
|
|
121
|
+
const maxDateObj = maxDate ? dayjs(maxDate).endOf('day') : null;
|
|
122
|
+
|
|
123
|
+
const timeFieldError = get(errors, timeField.name);
|
|
124
|
+
const timeFormatFieldError = get(errors, timeFormatField.name);
|
|
125
|
+
|
|
126
|
+
return (
|
|
127
|
+
<div className={classNames(styles.dateTimeSection, styles.sectionField)}>
|
|
128
|
+
<Controller
|
|
129
|
+
name={dateField.name}
|
|
130
|
+
control={control}
|
|
131
|
+
render={({ field, fieldState }) => (
|
|
132
|
+
<ResponsiveWrapper>
|
|
133
|
+
<OpenmrsDatePicker
|
|
134
|
+
{...field}
|
|
135
|
+
value={field.value as Date}
|
|
136
|
+
className={styles.datePicker}
|
|
137
|
+
id={`${dateField.name}Input`}
|
|
138
|
+
data-testid={`${dateField.name}Input`}
|
|
139
|
+
maxDate={maxDateObj}
|
|
140
|
+
minDate={minDateObj}
|
|
141
|
+
labelText={dateField.label}
|
|
142
|
+
invalid={Boolean(fieldState?.error?.message)}
|
|
143
|
+
invalidText={fieldState?.error?.message}
|
|
144
|
+
/>
|
|
145
|
+
</ResponsiveWrapper>
|
|
146
|
+
)}
|
|
147
|
+
/>
|
|
148
|
+
<ResponsiveWrapper>
|
|
149
|
+
<Controller
|
|
150
|
+
name={timeField.name}
|
|
151
|
+
control={control}
|
|
152
|
+
render={({ field: { onBlur, onChange, value } }) => (
|
|
153
|
+
<div className={styles.timePickerContainer}>
|
|
154
|
+
<TimePicker
|
|
155
|
+
className={styles.timePicker}
|
|
156
|
+
disabled={disabled}
|
|
157
|
+
id={timeField.name}
|
|
158
|
+
invalid={Boolean(timeFieldError?.message)}
|
|
159
|
+
invalidText={timeFieldError?.message}
|
|
160
|
+
labelText={timeField.label}
|
|
161
|
+
onBlur={onBlur}
|
|
162
|
+
onChange={(event) => onChange(event.target.value)}
|
|
163
|
+
pattern="^(0[1-9]|1[0-2]):([0-5][0-9])$"
|
|
164
|
+
value={value as string}>
|
|
165
|
+
<Controller
|
|
166
|
+
name={timeFormatField.name}
|
|
167
|
+
control={control}
|
|
168
|
+
render={({ field: { onChange, value } }) => (
|
|
169
|
+
<TimePickerSelect
|
|
170
|
+
aria-label={timeFormatField.label}
|
|
171
|
+
className={classNames({
|
|
172
|
+
[styles.timePickerSelectError]: Boolean(timeFormatFieldError),
|
|
173
|
+
})}
|
|
174
|
+
disabled={disabled}
|
|
175
|
+
id={`${timeFormatField.name}Input`}
|
|
176
|
+
onChange={(event) => onChange(event.target.value as amPm)}
|
|
177
|
+
value={value as amPm}>
|
|
178
|
+
<SelectItem value="AM" text={t('AM', 'AM')} />
|
|
179
|
+
<SelectItem value="PM" text={t('PM', 'PM')} />
|
|
180
|
+
</TimePickerSelect>
|
|
181
|
+
)}
|
|
182
|
+
/>
|
|
183
|
+
</TimePicker>
|
|
184
|
+
{timeFormatFieldError && <div className={styles.timerPickerError}>{timeFormatFieldError?.message}</div>}
|
|
185
|
+
</div>
|
|
186
|
+
)}
|
|
187
|
+
/>
|
|
188
|
+
</ResponsiveWrapper>
|
|
189
|
+
</div>
|
|
190
|
+
);
|
|
191
|
+
};
|
|
192
|
+
|
|
193
|
+
export default VisitDateTimeSection;
|