@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,222 @@
1
+ import React from 'react';
2
+ import { IconButton, InlineLoading } from '@carbon/react';
3
+ import type { TFunction } from 'i18next';
4
+ import { Renew } from '@carbon/react/icons';
5
+ import type { QueueSummaryCard } from '../../shared/queue/queue-summary-cards.component';
6
+ import type { QueueFilter, QueueEntry } from '../../types';
7
+ import styles from './consultation.scss';
8
+
9
+ type InvestigationCategorizedEntries = {
10
+ newLabOrders: QueueEntry[];
11
+ newRadiologyOrders: QueueEntry[];
12
+ newProcedureOrders: QueueEntry[];
13
+ completedLabOrders: QueueEntry[];
14
+ completedRadiologyOrders: QueueEntry[];
15
+ completedProcedureOrders: QueueEntry[];
16
+ };
17
+
18
+ type BuildConsultationCardsParams = {
19
+ t: TFunction;
20
+ waitingCount: number;
21
+ emergencyCount: number;
22
+ urgentCount: number;
23
+ notUrgentCount: number;
24
+ awaitingCount: number;
25
+ completedCount: number;
26
+ labAwaiting: number;
27
+ labCompleted: number;
28
+ radiologyAwaiting: number;
29
+ radiologyCompleted: number;
30
+ proceduresAwaiting: number;
31
+ proceduresCompleted: number;
32
+ isRefreshing: boolean;
33
+ isLoadingInvestigations: boolean;
34
+ onRefreshInvestigations: () => void;
35
+ totalVisitsCount?: number;
36
+ setFilters: React.Dispatch<React.SetStateAction<QueueFilter[]>>;
37
+ emergencyPriorityConceptUuid: string;
38
+ urgentPriorityConceptUuid: string;
39
+ notUrgentPriorityConceptUuid: string;
40
+ investigationCategorizedEntries: InvestigationCategorizedEntries;
41
+ };
42
+
43
+ export function buildConsultationCards({
44
+ t,
45
+ waitingCount,
46
+ emergencyCount,
47
+ urgentCount,
48
+ notUrgentCount,
49
+ awaitingCount,
50
+ completedCount,
51
+ labAwaiting,
52
+ labCompleted,
53
+ radiologyAwaiting,
54
+ radiologyCompleted,
55
+ proceduresAwaiting,
56
+ proceduresCompleted,
57
+ isRefreshing,
58
+ isLoadingInvestigations,
59
+ onRefreshInvestigations,
60
+ totalVisitsCount,
61
+ setFilters,
62
+ emergencyPriorityConceptUuid,
63
+ urgentPriorityConceptUuid,
64
+ notUrgentPriorityConceptUuid,
65
+ investigationCategorizedEntries,
66
+ }: BuildConsultationCardsParams): Array<QueueSummaryCard> {
67
+ return [
68
+ {
69
+ title: t('awaitingConsultation', 'Awaiting consultation'),
70
+ value: waitingCount.toString(),
71
+ categories: [
72
+ {
73
+ label: t('emergency', 'Emergency'),
74
+ value: emergencyCount,
75
+ onClick: () => {
76
+ setFilters((prevFilters) => [
77
+ ...prevFilters.filter((f) => f.key !== 'priority'),
78
+ { key: 'priority', value: emergencyPriorityConceptUuid, label: t('emergency', 'Emergency') },
79
+ ]);
80
+ },
81
+ },
82
+ {
83
+ label: t('urgent', 'Urgent'),
84
+ value: urgentCount,
85
+ onClick: () => {
86
+ setFilters((prevFilters) => [
87
+ ...prevFilters.filter((f) => f.key !== 'priority'),
88
+ { key: 'priority', value: urgentPriorityConceptUuid, label: t('urgent', 'Urgent') },
89
+ ]);
90
+ },
91
+ },
92
+ {
93
+ label: t('notUrgent', 'Not Urgent'),
94
+ value: notUrgentCount,
95
+ onClick: () => {
96
+ setFilters((prevFilters) => [
97
+ ...prevFilters.filter((f) => f.key !== 'priority'),
98
+ { key: 'priority', value: notUrgentPriorityConceptUuid, label: t('notUrgent', 'Not Urgent') },
99
+ ]);
100
+ },
101
+ },
102
+ ],
103
+ },
104
+ {
105
+ title: t('investigationAwaiting', 'Investigation Awaiting'),
106
+ value: awaitingCount.toString(),
107
+ categories: [
108
+ {
109
+ label: t('lab', 'Lab'),
110
+ value: labAwaiting,
111
+ onClick: () => {
112
+ setFilters((prevFilters) => [
113
+ ...prevFilters.filter((f) => f.key !== 'service_awaiting'),
114
+ {
115
+ key: 'service_awaiting',
116
+ value: investigationCategorizedEntries.newLabOrders.map((entry) => entry.patient.uuid).join(','),
117
+ label: t('labAwaiting', 'Lab Awaiting'),
118
+ },
119
+ ]);
120
+ },
121
+ },
122
+ {
123
+ label: t('radiology', 'Radiology'),
124
+ value: radiologyAwaiting,
125
+ onClick: () => {
126
+ setFilters((prevFilters) => [
127
+ ...prevFilters.filter((f) => f.key !== 'service_awaiting'),
128
+ {
129
+ key: 'service_awaiting',
130
+ value: investigationCategorizedEntries.newRadiologyOrders.map((entry) => entry.patient.uuid).join(','),
131
+ label: t('radiologyAwaiting', 'Radiology Awaiting'),
132
+ },
133
+ ]);
134
+ },
135
+ },
136
+ {
137
+ label: t('procedures', 'Procedures'),
138
+ value: proceduresAwaiting,
139
+ onClick: () => {
140
+ setFilters((prevFilters) => [
141
+ ...prevFilters.filter((f) => f.key !== 'service_awaiting'),
142
+ {
143
+ key: 'service_awaiting',
144
+ value: investigationCategorizedEntries.newProcedureOrders.map((entry) => entry.patient.uuid).join(','),
145
+ label: t('proceduresAwaiting', 'Procedures Awaiting'),
146
+ },
147
+ ]);
148
+ },
149
+ },
150
+ ],
151
+ refreshButton:
152
+ isRefreshing || isLoadingInvestigations ? (
153
+ <InlineLoading description={t('refreshing', 'Refreshing...')} />
154
+ ) : (
155
+ <IconButton
156
+ label={t('refreshInvestigations', 'Refresh investigations')}
157
+ kind="ghost"
158
+ size="sm"
159
+ onClick={onRefreshInvestigations}
160
+ className={styles.refreshButton}>
161
+ <Renew size={16} />
162
+ </IconButton>
163
+ ),
164
+ },
165
+ {
166
+ title: t('investigationCompleted', 'Investigation Completed'),
167
+ value: completedCount.toString(),
168
+ categories: [
169
+ {
170
+ label: t('lab', 'Lab'),
171
+ value: labCompleted,
172
+ onClick: () => {
173
+ setFilters((prevFilters) => [
174
+ ...prevFilters.filter((f) => f.key !== 'service_completed'),
175
+ {
176
+ key: 'service_completed',
177
+ value: investigationCategorizedEntries.completedLabOrders.map((entry) => entry.patient.uuid).join(','),
178
+ label: t('labCompleted', 'Lab Completed'),
179
+ },
180
+ ]);
181
+ },
182
+ },
183
+ {
184
+ label: t('radiology', 'Radiology'),
185
+ value: radiologyCompleted,
186
+ onClick: () => {
187
+ setFilters((prevFilters) => [
188
+ ...prevFilters.filter((f) => f.key !== 'service_completed'),
189
+ {
190
+ key: 'service_completed',
191
+ value: investigationCategorizedEntries.completedRadiologyOrders
192
+ .map((entry) => entry.patient.uuid)
193
+ .join(','),
194
+ label: t('radiologyCompleted', 'Radiology Completed'),
195
+ },
196
+ ]);
197
+ },
198
+ },
199
+ {
200
+ label: t('procedures', 'Procedures'),
201
+ value: proceduresCompleted,
202
+ onClick: () => {
203
+ setFilters((prevFilters) => [
204
+ ...prevFilters.filter((f) => f.key !== 'service_completed'),
205
+ {
206
+ key: 'service_completed',
207
+ value: investigationCategorizedEntries.completedProcedureOrders
208
+ .map((entry) => entry.patient.uuid)
209
+ .join(','),
210
+ label: t('proceduresCompleted', 'Procedures Completed'),
211
+ },
212
+ ]);
213
+ },
214
+ },
215
+ ],
216
+ },
217
+ {
218
+ title: t('totalVisits', 'Total Visits'),
219
+ value: totalVisitsCount?.toString() ?? '0',
220
+ },
221
+ ];
222
+ }
@@ -1,7 +1,6 @@
1
1
  import React from 'react';
2
2
  import { BrowserRouter, Routes, Route } from 'react-router-dom';
3
3
  import { spaBasePath } from '../../constants';
4
- import PatientChart from '../../shared/patient-chart/patient-chart.component';
5
4
  import Consultation from './consultation.component';
6
5
 
7
6
  type ConsultationDashboardProps = {
@@ -13,7 +12,6 @@ const ConsultationDashboard: React.FC<ConsultationDashboardProps> = ({ dashboard
13
12
  <BrowserRouter basename={`${spaBasePath}/consultation`}>
14
13
  <Routes>
15
14
  <Route path="/" element={<Consultation dashboardTitle={dashboardTitle} />} />
16
- <Route path="/:patientUuid" element={<PatientChart navigationPath="consultation" />} />
17
15
  </Routes>
18
16
  </BrowserRouter>
19
17
  );
@@ -1,37 +1,44 @@
1
- import { getSyncLifecycle } from '@openmrs/esm-framework';
2
- import { createDashboardLink } from '@openmrs/esm-patient-common-lib';
1
+ import { getAsyncLifecycle } from '@openmrs/esm-framework';
3
2
 
4
3
  import { moduleName } from '../../constants';
5
- import { createLeftPanelLink } from '../../shared/dashboard-link/dashboard-link.component';
6
- import ClinicalEncounter from './clinical-encounter/clinical-encounter.component';
7
- import EncounterDetails from './clinical-encounter/encounter-details.component';
8
- import ConsultationDashboard from './dashboard.component';
9
4
 
10
5
  const options = {
11
6
  featureName: 'express-workflow',
12
7
  moduleName,
13
8
  };
14
9
 
15
- export const consultationDashboard = getSyncLifecycle(ConsultationDashboard, options);
10
+ export const consultationDashboard = getAsyncLifecycle(() => import('./dashboard.component'), options);
16
11
  // t('consultation', 'Consultation')
17
- export const consultationLeftPanelLink = getSyncLifecycle(
18
- createLeftPanelLink({
19
- title: 'consultation',
20
- name: 'consultation',
21
- }),
12
+ export const consultationLeftPanelLink = getAsyncLifecycle(
13
+ () =>
14
+ import('../../shared/dashboard-link/dashboard-link.component').then((m) => ({
15
+ default: m.createLeftPanelLink({
16
+ title: 'consultation',
17
+ name: 'consultation',
18
+ }),
19
+ })),
22
20
  options,
23
21
  );
24
22
 
25
- export const clinicalEncounter = getSyncLifecycle(ClinicalEncounter, options);
23
+ export const clinicalEncounter = getAsyncLifecycle(
24
+ () => import('./clinical-encounter/clinical-encounter.component'),
25
+ options,
26
+ );
26
27
  // TODO: register Stethoscope icon in the icon registry
27
28
  // t('clinicalEncounter', 'Clinical Encounter')
28
- export const clinicalEncounterLink = getSyncLifecycle(
29
- createDashboardLink({
30
- title: 'clinicalEncounter',
31
- path: 'clinical-encounter',
32
- icon: 'omrs-icon-syringe',
33
- }),
29
+ export const clinicalEncounterLink = getAsyncLifecycle(
30
+ () =>
31
+ import('@openmrs/esm-patient-common-lib').then((m) => ({
32
+ default: m.createDashboardLink({
33
+ title: 'clinicalEncounter',
34
+ path: 'clinical-encounter',
35
+ icon: 'omrs-icon-syringe',
36
+ }),
37
+ })),
34
38
  options,
35
39
  );
36
40
 
37
- export const encounterDetails = getSyncLifecycle(EncounterDetails, options);
41
+ export const encounterDetails = getAsyncLifecycle(
42
+ () => import('./clinical-encounter/encounter-details.component'),
43
+ options,
44
+ );
@@ -1,21 +1,22 @@
1
- import { getSyncLifecycle } from '@openmrs/esm-framework';
1
+ import { getAsyncLifecycle } from '@openmrs/esm-framework';
2
2
 
3
- import FacilityDashboard from './facility-dashboard.component';
4
3
  import { moduleName } from '../../constants';
5
- import { createLeftPanelLink } from '../../shared/dashboard-link/dashboard-link.component';
6
4
 
7
5
  const options = {
8
6
  featureName: 'express-workflow',
9
7
  moduleName,
10
8
  };
11
9
 
12
- export const facilityDashboard = getSyncLifecycle(FacilityDashboard, options);
10
+ export const facilityDashboard = getAsyncLifecycle(() => import('./facility-dashboard.component'), options);
13
11
 
14
12
  // t('dashboard', 'Dashboard')
15
- export const facilityLeftPanelLink = getSyncLifecycle(
16
- createLeftPanelLink({
17
- title: 'dashboard',
18
- name: 'facility-dashboard',
19
- }),
13
+ export const facilityLeftPanelLink = getAsyncLifecycle(
14
+ () =>
15
+ import('../../shared/dashboard-link/dashboard-link.component').then((m) => ({
16
+ default: m.createLeftPanelLink({
17
+ title: 'dashboard',
18
+ name: 'facility-dashboard',
19
+ }),
20
+ })),
20
21
  options,
21
22
  );
@@ -1,21 +1,22 @@
1
- import { getSyncLifecycle } from '@openmrs/esm-framework';
1
+ import { getAsyncLifecycle } from '@openmrs/esm-framework';
2
2
 
3
- import { createDashboardLink } from '@openmrs/esm-patient-common-lib';
4
3
  import { moduleName } from '../../constants';
5
- import LaboratoryTabs from './laboratory-tabs.component';
6
4
 
7
5
  const options = {
8
6
  featureName: 'express-workflow',
9
7
  moduleName,
10
8
  };
11
9
 
12
- export const laboratoryDashboard = getSyncLifecycle(LaboratoryTabs, options);
10
+ export const laboratoryDashboard = getAsyncLifecycle(() => import('./laboratory-tabs.component'), options);
13
11
  // t('labOrders', 'Lab Orders')
14
- export const laboratoryLeftPanelLink = getSyncLifecycle(
15
- createDashboardLink({
16
- path: 'laboratory',
17
- title: 'labOrders',
18
- icon: 'omrs-icon-microscope',
19
- }),
12
+ export const laboratoryLeftPanelLink = getAsyncLifecycle(
13
+ () =>
14
+ import('@openmrs/esm-patient-common-lib').then((m) => ({
15
+ default: m.createDashboardLink({
16
+ path: 'laboratory',
17
+ title: 'labOrders',
18
+ icon: 'omrs-icon-microscope',
19
+ }),
20
+ })),
20
21
  options,
21
22
  );
@@ -1,22 +1,34 @@
1
- import React from 'react';
1
+ import React, { useMemo } from 'react';
2
2
  import { Order } from '../../types/order/order';
3
3
  import { Layer } from '@carbon/react';
4
4
  import { useTranslation } from 'react-i18next';
5
5
  import { useConfig } from '@openmrs/esm-framework';
6
6
  import styles from './laboratory-tabs.scss';
7
- import { EmptyState, useLaunchWorkspaceRequiringVisit } from '@openmrs/esm-patient-common-lib';
7
+ import { EmptyState, useLaunchWorkspaceRequiringVisit, usePatientChartStore } from '@openmrs/esm-patient-common-lib';
8
8
  import { ExpressWorkflowConfig } from '../../config-schema';
9
9
  import OrderTable from '../../shared/orders/OrderTable';
10
10
 
11
11
  type LabTableProps = {
12
- orders: Order[];
12
+ orders: Array<Order>;
13
+ patientUuid: string;
14
+ mutateOrders: () => void;
13
15
  };
14
16
 
15
- const LabTable: React.FC<LabTableProps> = ({ orders }) => {
17
+ const LabTable: React.FC<LabTableProps> = ({ orders, patientUuid, mutateOrders }) => {
16
18
  const { t } = useTranslation();
19
+ const { patient } = usePatientChartStore(patientUuid);
20
+ const windowProps = useMemo(() => ({ encounterUuid: orders[0]?.encounter?.uuid }), [orders[0]?.encounter?.uuid]);
21
+ const groupProps = useMemo(
22
+ () => ({
23
+ patient,
24
+ patientUuid: patient?.id,
25
+ visitContext: orders[0]?.encounter?.visit,
26
+ mutateVisitContext: mutateOrders,
27
+ }),
28
+ [patient, orders[0]?.encounter?.visit, mutateOrders],
29
+ );
17
30
  const { labOrderTypeUuid, orderableConceptSets } = useConfig<ExpressWorkflowConfig>();
18
-
19
- const launchAddLabOrder = useLaunchWorkspaceRequiringVisit('add-lab-order');
31
+ const launchAddLabOrder = useLaunchWorkspaceRequiringVisit(patientUuid, 'order-basket');
20
32
 
21
33
  if (orders?.length === 0) {
22
34
  return (
@@ -24,7 +36,9 @@ const LabTable: React.FC<LabTableProps> = ({ orders }) => {
24
36
  <EmptyState
25
37
  displayText={t('orders', 'Orders')}
26
38
  headerTitle={t('laboratoryOrders', 'Laboratory Orders')}
27
- launchForm={() => launchAddLabOrder({ orderTypeUuid: labOrderTypeUuid, orderableConceptSets })}
39
+ launchForm={() =>
40
+ launchAddLabOrder({ orderTypeUuid: labOrderTypeUuid, orderableConceptSets }, windowProps, groupProps)
41
+ }
28
42
  />
29
43
  </Layer>
30
44
  );
@@ -34,7 +48,7 @@ const LabTable: React.FC<LabTableProps> = ({ orders }) => {
34
48
  <OrderTable
35
49
  title={t('laboratoryOrders', 'Laboratory Orders')}
36
50
  orders={orders}
37
- onAdd={() => launchAddLabOrder({ orderTypeUuid: labOrderTypeUuid, orderableConceptSets })}
51
+ onAdd={() => launchAddLabOrder(null, { encounterUuid: '' }, groupProps)}
38
52
  containerClassName={styles.labTableContainer}
39
53
  tableCellClassName={styles.tableCell}
40
54
  priorityPillClassName={styles.priorityPill}
@@ -19,8 +19,8 @@ const LaboratoryTabs: React.FC<LaboratoryTabsProps> = ({ patientUuid, patient })
19
19
  basePath="laboratory"
20
20
  resultsSlotName="ewf-laboratory-results-slot"
21
21
  orderTypeUuid={testOrderTypeUuid}
22
- Table={({ orders }) => {
23
- return <LabTable orders={orders} />;
22
+ Table={({ orders, mutateOrders }) => {
23
+ return <LabTable orders={orders} patientUuid={patientUuid} mutateOrders={mutateOrders} />;
24
24
  }}
25
25
  />
26
26
  );
@@ -1,7 +1,6 @@
1
1
  import React from 'react';
2
2
  import { BrowserRouter, Routes, Route } from 'react-router-dom';
3
3
  import { spaBasePath } from '../../constants';
4
- import PatientChart from '../../shared/patient-chart/patient-chart.component';
5
4
  import MCH from './mch.component';
6
5
 
7
6
  type MchDashboardProps = {
@@ -13,7 +12,6 @@ const MCHDashboard: React.FC<MchDashboardProps> = ({ dashboardTitle }) => {
13
12
  <BrowserRouter basename={`${spaBasePath}/mch`}>
14
13
  <Routes>
15
14
  <Route path="/" element={<MCH dashboardTitle={dashboardTitle} />} />
16
- <Route path="/:patientUuid" element={<PatientChart navigationPath="mch" />} />
17
15
  </Routes>
18
16
  </BrowserRouter>
19
17
  );
@@ -1,20 +1,21 @@
1
- import { getSyncLifecycle } from '@openmrs/esm-framework';
1
+ import { getAsyncLifecycle } from '@openmrs/esm-framework';
2
2
 
3
3
  import { moduleName } from '../../constants';
4
- import { createLeftPanelLink } from '../../shared/dashboard-link/dashboard-link.component';
5
- import MCHDashboard from './dashboard.component';
6
4
 
7
5
  const options = {
8
6
  featureName: 'express-workflow',
9
7
  moduleName,
10
8
  };
11
9
 
12
- export const mchDashboard = getSyncLifecycle(MCHDashboard, options);
10
+ export const mchDashboard = getAsyncLifecycle(() => import('./dashboard.component'), options);
13
11
  // t('mch', 'MCH')
14
- export const mchLeftPanelLink = getSyncLifecycle(
15
- createLeftPanelLink({
16
- title: 'mch',
17
- name: 'mch',
18
- }),
12
+ export const mchLeftPanelLink = getAsyncLifecycle(
13
+ () =>
14
+ import('../../shared/dashboard-link/dashboard-link.component').then((m) => ({
15
+ default: m.createLeftPanelLink({
16
+ title: 'mch',
17
+ name: 'mch',
18
+ }),
19
+ })),
19
20
  options,
20
21
  );
@@ -1,12 +1,14 @@
1
+ import React, { useCallback, useMemo, useState } from 'react';
1
2
  import { IconButton, InlineLoading } from '@carbon/react';
2
3
  import { Renew } from '@carbon/react/icons';
3
4
  import { useConfig } from '@openmrs/esm-framework';
4
5
  import { EmptyState } from '@openmrs/esm-patient-common-lib';
5
- import React, { useCallback, useMemo, useState } from 'react';
6
6
  import { useTranslation } from 'react-i18next';
7
+
7
8
  import { ExpressWorkflowConfig } from '../../config-schema';
8
9
  import { useQueues } from '../../hooks/useServiceQueues';
9
10
  import QueueTab from '../../shared/queue/queue-tab.component';
11
+ import QueueSummaryCards, { QueueSummaryCard } from '../../shared/queue/queue-summary-cards.component';
10
12
  import { Queue, QueueFilter } from '../../types';
11
13
  import {
12
14
  useConsultationQueueMetrics,
@@ -17,7 +19,7 @@ import styles from '../consultation/consultation.scss';
17
19
 
18
20
  const MCHConsultation: React.FC = () => {
19
21
  const { t } = useTranslation();
20
- const { queues, isLoading, error } = useQueues();
22
+ const { queues, isLoading: isLoadingQueues } = useQueues();
21
23
  const [currQueue, setCurrQueue] = useState<Queue>();
22
24
  const [filters, setFilters] = useState<Array<QueueFilter>>([]);
23
25
  const [isRefreshing, setIsRefreshing] = useState(false);
@@ -26,7 +28,12 @@ const MCHConsultation: React.FC = () => {
26
28
  priorities: { emergencyPriorityConceptUuid, urgentPriorityConceptUuid, notUrgentPriorityConceptUuid },
27
29
  queueServiceConceptUuids,
28
30
  } = useConfig<ExpressWorkflowConfig>();
29
- const { data: totalVisits, isLoading: isLoadingTotalVisits } = useTotalVisits();
31
+ const {
32
+ data: totalVisits,
33
+ isLoading: isLoadingTotalVisits,
34
+ isValidating: isValidatingTotalVisits,
35
+ } = useTotalVisits();
36
+
30
37
  const consultationQueues = queues.filter(
31
38
  (queue) =>
32
39
  queue.service.uuid === queueServiceConceptUuids.consultationService &&
@@ -38,21 +45,21 @@ const MCHConsultation: React.FC = () => {
38
45
  const {
39
46
  waitingEntries,
40
47
  isLoading: isLoadingQueueMetrics,
41
- error: waitingError,
42
48
  emergencyEntries,
43
49
  urgentEntries,
44
50
  notUrgentEntries,
51
+ isValidating: isValidatingQueueMetrics,
45
52
  } = useConsultationQueueMetrics(activeQueue);
46
53
  const {
47
54
  awaitingCount,
48
55
  completedCount,
49
- totalCount,
50
56
  lab,
51
57
  radiology,
52
58
  procedures,
53
59
  isLoading: isLoadingInvestigations,
54
60
  refresh: refreshInvestigations,
55
61
  investigationCategorizedEntries,
62
+ isValidating: isValidatingInvestigations,
56
63
  } = useInvestigationStats(activeQueue);
57
64
  // Single refresh handler for all investigations
58
65
  const handleRefresh = useCallback(async () => {
@@ -66,7 +73,7 @@ const MCHConsultation: React.FC = () => {
66
73
  }
67
74
  }, [refreshInvestigations]);
68
75
 
69
- const cards = useMemo(
76
+ const cards: Array<QueueSummaryCard> = useMemo(
70
77
  () => [
71
78
  {
72
79
  title: t('awaitingConsultation', 'Awaiting consultation'),
@@ -258,7 +265,10 @@ const MCHConsultation: React.FC = () => {
258
265
  ],
259
266
  );
260
267
 
261
- if (isLoading || isLoadingTotalVisits || isLoadingInvestigations || isLoadingQueueMetrics) {
268
+ const isLoading = isLoadingQueues || isLoadingTotalVisits || isLoadingInvestigations || isLoadingQueueMetrics;
269
+ const isValidating = isValidatingTotalVisits || isValidatingInvestigations || isValidatingQueueMetrics;
270
+
271
+ if (isLoading && !isValidating) {
262
272
  return <InlineLoading description={t('loadingQueues', 'Loading queues...')} />;
263
273
  }
264
274
 
@@ -271,14 +281,16 @@ const MCHConsultation: React.FC = () => {
271
281
  );
272
282
  }
273
283
  return (
274
- <QueueTab
275
- queues={consultationQueues}
276
- navigatePath="mch"
277
- cards={cards}
278
- usePatientChart
279
- filters={filters}
280
- onFiltersChanged={setFilters}
281
- />
284
+ <>
285
+ <QueueSummaryCards cards={cards} />
286
+ <QueueTab
287
+ queues={consultationQueues}
288
+ navigatePath="mch"
289
+ usePatientChart
290
+ filters={filters}
291
+ onFiltersChanged={setFilters}
292
+ />
293
+ </>
282
294
  );
283
295
  };
284
296