@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.
Files changed (388) hide show
  1. package/.turbo/turbo-build.log +7 -12
  2. package/dist/1074.js +1 -0
  3. package/dist/1074.js.map +1 -0
  4. package/dist/12.js +17 -0
  5. package/dist/12.js.map +1 -0
  6. package/dist/1311.js +1 -0
  7. package/dist/1311.js.map +1 -0
  8. package/dist/1323.js +1 -0
  9. package/dist/1323.js.map +1 -0
  10. package/dist/1469.js +1 -0
  11. package/dist/1469.js.map +1 -0
  12. package/dist/1506.js +13 -0
  13. package/dist/1506.js.map +1 -0
  14. package/dist/1562.js +1 -0
  15. package/dist/1562.js.map +1 -0
  16. package/dist/1760.js +1 -0
  17. package/dist/1760.js.map +1 -0
  18. package/dist/1780.js +1 -0
  19. package/dist/1780.js.map +1 -0
  20. package/dist/1804.js +1 -0
  21. package/dist/1804.js.map +1 -0
  22. package/dist/1884.js +1 -0
  23. package/dist/1884.js.map +1 -0
  24. package/dist/1972.js +1 -0
  25. package/dist/1972.js.map +1 -0
  26. package/dist/1990.js +1 -0
  27. package/dist/1990.js.map +1 -0
  28. package/dist/2016.js +1 -0
  29. package/dist/2016.js.map +1 -0
  30. package/dist/2024.js +1 -0
  31. package/dist/2024.js.map +1 -0
  32. package/dist/2153.js +1 -0
  33. package/dist/2153.js.map +1 -0
  34. package/dist/216.js +1 -0
  35. package/dist/216.js.map +1 -0
  36. package/dist/2225.js +1 -0
  37. package/dist/2225.js.map +1 -0
  38. package/dist/2294.js +1 -0
  39. package/dist/2294.js.map +1 -0
  40. package/dist/2345.js +1 -0
  41. package/dist/2345.js.map +1 -0
  42. package/dist/2499.js +1 -0
  43. package/dist/2499.js.map +1 -0
  44. package/dist/2500.js +1 -0
  45. package/dist/2500.js.map +1 -0
  46. package/dist/2586.js +1 -0
  47. package/dist/2586.js.map +1 -0
  48. package/dist/2625.js +1 -0
  49. package/dist/2625.js.map +1 -0
  50. package/dist/2685.js +1 -0
  51. package/dist/2685.js.map +1 -0
  52. package/dist/2809.js +1 -0
  53. package/dist/2809.js.map +1 -0
  54. package/dist/2851.js +1 -0
  55. package/dist/2851.js.map +1 -0
  56. package/dist/2881.js +1 -0
  57. package/dist/2881.js.map +1 -0
  58. package/dist/2948.js +1 -0
  59. package/dist/2948.js.map +1 -0
  60. package/dist/2968.js +1 -0
  61. package/dist/2968.js.map +1 -0
  62. package/dist/2978.js +1 -0
  63. package/dist/2978.js.map +1 -0
  64. package/dist/2998.js +1 -0
  65. package/dist/2998.js.map +1 -0
  66. package/dist/3089.js +1 -0
  67. package/dist/3089.js.map +1 -0
  68. package/dist/3548.js +1 -0
  69. package/dist/3548.js.map +1 -0
  70. package/dist/3567.js +1 -0
  71. package/dist/3567.js.map +1 -0
  72. package/dist/3569.js +1 -0
  73. package/dist/3569.js.map +1 -0
  74. package/dist/3571.js +1 -0
  75. package/dist/3571.js.map +1 -0
  76. package/dist/3691.js +1 -0
  77. package/dist/3691.js.map +1 -0
  78. package/dist/3730.js +1 -0
  79. package/dist/3730.js.map +1 -0
  80. package/dist/3923.js +1 -0
  81. package/dist/3923.js.map +1 -0
  82. package/dist/3963.js +1 -0
  83. package/dist/3963.js.map +1 -0
  84. package/dist/4024.js +1 -0
  85. package/dist/4024.js.map +1 -0
  86. package/dist/405.js +1 -0
  87. package/dist/405.js.map +1 -0
  88. package/dist/4071.js +1 -0
  89. package/dist/4071.js.map +1 -0
  90. package/dist/4271.js +1 -0
  91. package/dist/4271.js.map +1 -0
  92. package/dist/4296.js +1 -0
  93. package/dist/4296.js.map +1 -0
  94. package/dist/4337.js +1 -0
  95. package/dist/4337.js.map +1 -0
  96. package/dist/4432.js +1 -0
  97. package/dist/4432.js.map +1 -0
  98. package/dist/4581.js +1 -0
  99. package/dist/4581.js.map +1 -0
  100. package/dist/4637.js +11 -0
  101. package/dist/4637.js.map +1 -0
  102. package/dist/4666.js +1 -0
  103. package/dist/4666.js.map +1 -0
  104. package/dist/4680.js +1 -0
  105. package/dist/4680.js.map +1 -0
  106. package/dist/4735.js +1 -0
  107. package/dist/4735.js.map +1 -0
  108. package/dist/4737.js +1 -0
  109. package/dist/4737.js.map +1 -0
  110. package/dist/4744.js +1 -0
  111. package/dist/4744.js.map +1 -0
  112. package/dist/4795.js +1 -0
  113. package/dist/4795.js.map +1 -0
  114. package/dist/4813.js +2 -0
  115. package/dist/4813.js.map +1 -0
  116. package/dist/4818.js +1 -0
  117. package/dist/4818.js.map +1 -0
  118. package/dist/4858.js +1 -0
  119. package/dist/4858.js.map +1 -0
  120. package/dist/487.js +1 -0
  121. package/dist/487.js.map +1 -0
  122. package/dist/4970.js +1 -0
  123. package/dist/4970.js.map +1 -0
  124. package/dist/5038.js +1 -0
  125. package/dist/5038.js.map +1 -0
  126. package/dist/5202.js +1 -0
  127. package/dist/5202.js.map +1 -0
  128. package/dist/5491.js +1 -0
  129. package/dist/5491.js.map +1 -0
  130. package/dist/5592.js +1 -0
  131. package/dist/5592.js.map +1 -0
  132. package/dist/5669.js +1 -0
  133. package/dist/5669.js.map +1 -0
  134. package/dist/586.js +1 -0
  135. package/dist/586.js.map +1 -0
  136. package/dist/5932.js +1 -0
  137. package/dist/5932.js.map +1 -0
  138. package/dist/5995.js +1 -0
  139. package/dist/5995.js.map +1 -0
  140. package/dist/6258.js +1 -0
  141. package/dist/6258.js.map +1 -0
  142. package/dist/629.js +1 -0
  143. package/dist/629.js.map +1 -0
  144. package/dist/6328.js +1 -0
  145. package/dist/6328.js.map +1 -0
  146. package/dist/6355.js +1 -0
  147. package/dist/6355.js.map +1 -0
  148. package/dist/6419.js +1 -0
  149. package/dist/6419.js.map +1 -0
  150. package/dist/644.js +1 -0
  151. package/dist/644.js.map +1 -0
  152. package/dist/6456.js +1 -0
  153. package/dist/6466.js +3 -0
  154. package/dist/6466.js.map +1 -0
  155. package/dist/655.js +1 -0
  156. package/dist/655.js.map +1 -0
  157. package/dist/6798.js +66 -0
  158. package/dist/6798.js.map +1 -0
  159. package/dist/6910.js +1 -0
  160. package/dist/6910.js.map +1 -0
  161. package/dist/6925.js +1 -0
  162. package/dist/6925.js.map +1 -0
  163. package/dist/70.js +1 -0
  164. package/dist/70.js.map +1 -0
  165. package/dist/7201.js +1 -0
  166. package/dist/7201.js.map +1 -0
  167. package/dist/7234.js +1 -0
  168. package/dist/7234.js.map +1 -0
  169. package/dist/7261.js +1 -0
  170. package/dist/7261.js.map +1 -0
  171. package/dist/7326.js +1 -0
  172. package/dist/7359.js +1 -0
  173. package/dist/7487.js +1 -0
  174. package/dist/7487.js.map +1 -0
  175. package/dist/7591.js +1 -0
  176. package/dist/7591.js.map +1 -0
  177. package/dist/7607.js +1 -0
  178. package/dist/7701.js +1 -0
  179. package/dist/7701.js.map +1 -0
  180. package/dist/7717.js +1 -0
  181. package/dist/7717.js.map +1 -0
  182. package/dist/7739.js +1 -0
  183. package/dist/7739.js.map +1 -0
  184. package/dist/7788.js +1 -0
  185. package/dist/7788.js.map +1 -0
  186. package/dist/7819.js +1 -0
  187. package/dist/7819.js.map +1 -0
  188. package/dist/7971.js +1 -0
  189. package/dist/7971.js.map +1 -0
  190. package/dist/7983.js +1 -0
  191. package/dist/7983.js.map +1 -0
  192. package/dist/807.js +1 -0
  193. package/dist/807.js.map +1 -0
  194. package/dist/8159.js +7 -0
  195. package/dist/8159.js.map +1 -0
  196. package/dist/8338.js +1 -0
  197. package/dist/8338.js.map +1 -0
  198. package/dist/845.js +1 -0
  199. package/dist/845.js.map +1 -0
  200. package/dist/8570.js +1 -0
  201. package/dist/8570.js.map +1 -0
  202. package/dist/8661.js +1 -0
  203. package/dist/8661.js.map +1 -0
  204. package/dist/87.js +1 -0
  205. package/dist/87.js.map +1 -0
  206. package/dist/8727.js +1 -0
  207. package/dist/8766.js +1 -0
  208. package/dist/8766.js.map +1 -0
  209. package/dist/8828.js +1 -0
  210. package/dist/8828.js.map +1 -0
  211. package/dist/8860.js +1 -0
  212. package/dist/8860.js.map +1 -0
  213. package/dist/8911.js +1 -0
  214. package/dist/8911.js.map +1 -0
  215. package/dist/8930.js +1 -0
  216. package/dist/8930.js.map +1 -0
  217. package/dist/8971.js +1 -0
  218. package/dist/8971.js.map +1 -0
  219. package/dist/9124.js +1 -0
  220. package/dist/9124.js.map +1 -0
  221. package/dist/9157.js +1 -0
  222. package/dist/9157.js.map +1 -0
  223. package/dist/9182.js +1 -0
  224. package/dist/921.js +1 -0
  225. package/dist/921.js.map +1 -0
  226. package/dist/9212.js +1 -0
  227. package/dist/9212.js.map +1 -0
  228. package/dist/9255.js +1 -0
  229. package/dist/9255.js.map +1 -0
  230. package/dist/9257.js +1 -0
  231. package/dist/9257.js.map +1 -0
  232. package/dist/9316.js +1 -0
  233. package/dist/9316.js.map +1 -0
  234. package/dist/9333.js +1 -0
  235. package/dist/9333.js.map +1 -0
  236. package/dist/9404.js +1 -0
  237. package/dist/9404.js.map +1 -0
  238. package/dist/9446.js +1 -0
  239. package/dist/9446.js.map +1 -0
  240. package/dist/9447.js +1 -0
  241. package/dist/9447.js.map +1 -0
  242. package/dist/9449.js +1 -0
  243. package/dist/9449.js.map +1 -0
  244. package/dist/9535.js +1 -0
  245. package/dist/9535.js.map +1 -0
  246. package/dist/9606.js +1 -0
  247. package/dist/9606.js.map +1 -0
  248. package/dist/973.js +1 -0
  249. package/dist/973.js.map +1 -0
  250. package/dist/9845.js +1 -0
  251. package/dist/9845.js.map +1 -0
  252. package/dist/kenyaemr-esm-express-workflow-app.js +5 -5
  253. package/dist/kenyaemr-esm-express-workflow-app.js.buildmanifest.json +3004 -175
  254. package/dist/kenyaemr-esm-express-workflow-app.js.map +1 -1
  255. package/dist/main.js +5 -114
  256. package/dist/main.js.map +1 -1
  257. package/dist/routes.json +1 -1
  258. package/package.json +6 -7
  259. package/rspack.config.js +1 -1
  260. package/src/components/accounting/index.tsx +11 -10
  261. package/src/components/admissions/index.tsx +11 -10
  262. package/src/components/appointments/index.ts +10 -8
  263. package/src/components/consultation/clinical-encounter/encounter-details.component.tsx +9 -9
  264. package/src/components/consultation/clinical-encounter/encounter-details.test.tsx +19 -18
  265. package/src/components/consultation/consultation-context.tsx +19 -0
  266. package/src/components/consultation/consultation-summary-cards.component.tsx +124 -0
  267. package/src/components/consultation/consultation.component.tsx +41 -268
  268. package/src/components/consultation/consultation.resource.ts +67 -24
  269. package/src/components/consultation/consultation.scss +5 -0
  270. package/src/components/consultation/consultation.utils.tsx +222 -0
  271. package/src/components/consultation/dashboard.component.tsx +0 -2
  272. package/src/components/consultation/index.ts +27 -20
  273. package/src/components/facility-dashboard/index.tsx +10 -9
  274. package/src/components/laboratory/index.ts +11 -10
  275. package/src/components/laboratory/lab-table.component.tsx +22 -8
  276. package/src/components/laboratory/laboratory-tabs.component.tsx +2 -2
  277. package/src/components/mch/dashboard.component.tsx +0 -2
  278. package/src/components/mch/index.tsx +10 -9
  279. package/src/components/mch/mch.consultation.component.tsx +27 -15
  280. package/src/components/mch/mch.triage.component.tsx +42 -33
  281. package/src/components/pharmacy/index.ts +20 -18
  282. package/src/components/pharmacy/orders/pharmacy-orders.component.tsx +35 -13
  283. package/src/components/pharmacy/orders/pharmacy-orders.test.tsx +8 -7
  284. package/src/components/procedures/index.ts +11 -10
  285. package/src/components/procedures/procedures-table.component.tsx +44 -13
  286. package/src/components/procedures/procedures-tabs.component.tsx +2 -2
  287. package/src/components/radiology-and-imaging/index.ts +14 -10
  288. package/src/components/radiology-and-imaging/radiology-and-imaging-table.component.tsx +35 -15
  289. package/src/components/radiology-and-imaging/radiology-and-imaging.component.tsx +2 -2
  290. package/src/components/registration/card/HIE-card/hie-card.component.tsx +32 -44
  291. package/src/components/registration/card/Local-card/local-card.component.tsx +22 -24
  292. package/src/components/registration/dependants/dependants.component.tsx +32 -60
  293. package/src/components/registration/dependants/dependants.resource.ts +79 -50
  294. package/src/components/registration/helper/index.ts +4 -0
  295. package/src/components/registration/index.ts +10 -9
  296. package/src/components/registration/search-bar/search-bar.resource.ts +32 -93
  297. package/src/components/registration/start-visit-form/hooks/useDefaultFacilityLocation.tsx +15 -0
  298. package/src/components/registration/start-visit-form/hooks/useDefaultVisitLocation.tsx +24 -0
  299. package/src/components/registration/start-visit-form/hooks/useOfflineVisitType.tsx +18 -0
  300. package/src/components/registration/start-visit-form/hooks/useRecommendedVisitTypes.tsx +36 -0
  301. package/src/components/registration/start-visit-form/hooks/useVisitAttributeType.tsx +102 -0
  302. package/src/components/registration/start-visit-form/overflow-menu-extension/overflow-menu-item.extension.tsx +55 -0
  303. package/src/components/registration/start-visit-form/overflow-menu-extension/overflow-menu-item.scss +3 -0
  304. package/src/components/registration/start-visit-form/start-visit-workspace/base-visit-type.component.tsx +126 -0
  305. package/src/components/registration/start-visit-form/start-visit-workspace/base-visit-type.scss +75 -0
  306. package/src/components/registration/start-visit-form/start-visit-workspace/exported-visit-form.workspace.tsx +743 -0
  307. package/src/components/registration/start-visit-form/start-visit-workspace/location-selector.component.tsx +86 -0
  308. package/src/components/registration/start-visit-form/start-visit-workspace/recommended-visit-type.component.tsx +32 -0
  309. package/src/components/registration/start-visit-form/start-visit-workspace/visit-attribute-type.component.tsx +257 -0
  310. package/src/components/registration/start-visit-form/start-visit-workspace/visit-attribute-type.scss +5 -0
  311. package/src/components/registration/start-visit-form/start-visit-workspace/visit-date-time.component.tsx +193 -0
  312. package/src/components/registration/start-visit-form/start-visit-workspace/visit-form.resource.ts +383 -0
  313. package/src/components/registration/start-visit-form/start-visit-workspace/visit-form.scss +166 -0
  314. package/src/components/registration/start-visit-form/visit-form-workspace/visit-form.workspace.tsx +55 -0
  315. package/src/components/reports/index.ts +10 -9
  316. package/src/components/triage/dashboard.component.tsx +0 -2
  317. package/src/components/triage/index.ts +10 -9
  318. package/src/components/triage/triage.component.tsx +92 -42
  319. package/src/components/triage/triage.resource.ts +56 -50
  320. package/src/components/triage/triage.scss +6 -0
  321. package/src/config-schema.ts +91 -0
  322. package/src/hooks/useServiceQueues.tsx +95 -25
  323. package/src/index.ts +32 -13
  324. package/src/routes.json +53 -254
  325. package/src/shared/orders/OrdersTabs.tsx +8 -3
  326. package/src/shared/patient-chart/patient-chart.resources.ts +8 -12
  327. package/src/shared/patient-chart/patient-summary-dashboard/patient-summary-dashboard.component.tsx +1 -2
  328. package/src/shared/pin-put/pinput.component.test.tsx +7 -6
  329. package/src/shared/queue/queue-entry/queue-entry-table.component.tsx +99 -88
  330. package/src/shared/queue/queue-entry/queue-entry-table.scss +4 -0
  331. package/src/shared/queue/queue-summary-cards.component.tsx +32 -0
  332. package/src/shared/queue/queue-tab.component.tsx +182 -70
  333. package/src/shared/queue/queue-tab.scss +1 -0
  334. package/src/shared/queue/queue-workflow-context.tsx +90 -0
  335. package/src/shared/tabs/extension-tabs.component.tsx +9 -6
  336. package/src/shared/utils/index.ts +1 -2
  337. package/src/types/index.ts +17 -2
  338. package/translations/am.json +64 -13
  339. package/translations/en.json +59 -8
  340. package/translations/sw.json +58 -7
  341. package/dist/127.js +0 -1
  342. package/dist/200.js +0 -1
  343. package/dist/200.js.map +0 -1
  344. package/dist/24.js +0 -1
  345. package/dist/24.js.map +0 -1
  346. package/dist/267.js +0 -1
  347. package/dist/267.js.map +0 -1
  348. package/dist/285.js +0 -1
  349. package/dist/285.js.map +0 -1
  350. package/dist/329.js +0 -1
  351. package/dist/329.js.map +0 -1
  352. package/dist/40.js +0 -1
  353. package/dist/466.js +0 -1
  354. package/dist/466.js.map +0 -1
  355. package/dist/472.js +0 -66
  356. package/dist/472.js.map +0 -1
  357. package/dist/490.js +0 -1
  358. package/dist/490.js.map +0 -1
  359. package/dist/54.js +0 -1
  360. package/dist/54.js.map +0 -1
  361. package/dist/689.js +0 -1
  362. package/dist/689.js.map +0 -1
  363. package/dist/697.js +0 -1
  364. package/dist/697.js.map +0 -1
  365. package/dist/704.js +0 -1
  366. package/dist/704.js.map +0 -1
  367. package/dist/710.js +0 -1
  368. package/dist/710.js.map +0 -1
  369. package/dist/729.js +0 -17
  370. package/dist/729.js.map +0 -1
  371. package/dist/805.js +0 -37
  372. package/dist/805.js.map +0 -1
  373. package/dist/847.js +0 -1
  374. package/dist/847.js.map +0 -1
  375. package/dist/85.js +0 -1
  376. package/dist/85.js.map +0 -1
  377. package/dist/882.js +0 -1
  378. package/dist/91.js +0 -1
  379. package/dist/91.js.map +0 -1
  380. package/dist/916.js +0 -1
  381. package/dist/994.js +0 -1
  382. package/dist/994.js.map +0 -1
  383. package/dist/998.js +0 -1
  384. package/dist/998.js.map +0 -1
  385. package/jest.config.js +0 -3
  386. package/src/shared/patient-chart/patient-chart.component.tsx +0 -61
  387. package/src/shared/patient-chart/patient-chart.scss +0 -15
  388. 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,5 @@
1
+ @use '@carbon/layout';
2
+
3
+ .visitAttributeField {
4
+ margin-bottom: layout.$spacing-05;
5
+ }
@@ -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;