@bathiran212/esm-patient-chart-app 7.0.0

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 (382) hide show
  1. package/README.md +3 -0
  2. package/dist/108.js +1 -0
  3. package/dist/108.js.map +1 -0
  4. package/dist/1339.js +1 -0
  5. package/dist/1339.js.map +1 -0
  6. package/dist/1480.js +1 -0
  7. package/dist/1480.js.map +1 -0
  8. package/dist/1543.js +1 -0
  9. package/dist/1543.js.map +1 -0
  10. package/dist/1582.js +1 -0
  11. package/dist/1582.js.map +1 -0
  12. package/dist/1646.js +1 -0
  13. package/dist/1646.js.map +1 -0
  14. package/dist/1797.js +1 -0
  15. package/dist/1797.js.map +1 -0
  16. package/dist/1869.js +1 -0
  17. package/dist/1869.js.map +1 -0
  18. package/dist/1877.js +1 -0
  19. package/dist/1877.js.map +1 -0
  20. package/dist/2020.js +1 -0
  21. package/dist/2020.js.map +1 -0
  22. package/dist/2246.js +1 -0
  23. package/dist/2246.js.map +1 -0
  24. package/dist/2317.js +1 -0
  25. package/dist/2317.js.map +1 -0
  26. package/dist/2416.js +1 -0
  27. package/dist/2416.js.map +1 -0
  28. package/dist/2790.js +1 -0
  29. package/dist/2790.js.map +1 -0
  30. package/dist/282.js +1 -0
  31. package/dist/282.js.map +1 -0
  32. package/dist/2881.js +1 -0
  33. package/dist/2881.js.map +1 -0
  34. package/dist/3137.js +1 -0
  35. package/dist/3137.js.map +1 -0
  36. package/dist/3378.js +1 -0
  37. package/dist/3378.js.map +1 -0
  38. package/dist/3390.js +1 -0
  39. package/dist/3390.js.map +1 -0
  40. package/dist/3536.js +1 -0
  41. package/dist/3536.js.map +1 -0
  42. package/dist/3720.js +1 -0
  43. package/dist/3720.js.map +1 -0
  44. package/dist/3857.js +1 -0
  45. package/dist/3857.js.map +1 -0
  46. package/dist/3925.js +1 -0
  47. package/dist/3925.js.map +1 -0
  48. package/dist/3963.js +1 -0
  49. package/dist/3963.js.map +1 -0
  50. package/dist/3989.js +1 -0
  51. package/dist/3989.js.map +1 -0
  52. package/dist/4092.js +1 -0
  53. package/dist/4092.js.map +1 -0
  54. package/dist/4106.js +1 -0
  55. package/dist/4106.js.map +1 -0
  56. package/dist/4111.js +1 -0
  57. package/dist/4111.js.map +1 -0
  58. package/dist/4145.js +1 -0
  59. package/dist/4145.js.map +1 -0
  60. package/dist/434.js +1 -0
  61. package/dist/434.js.map +1 -0
  62. package/dist/4348.js +1 -0
  63. package/dist/4348.js.map +1 -0
  64. package/dist/4383.js +1 -0
  65. package/dist/4383.js.map +1 -0
  66. package/dist/4540.js +1 -0
  67. package/dist/4540.js.map +1 -0
  68. package/dist/4658.js +1 -0
  69. package/dist/4658.js.map +1 -0
  70. package/dist/466.js +1 -0
  71. package/dist/466.js.map +1 -0
  72. package/dist/4913.js +1 -0
  73. package/dist/4913.js.map +1 -0
  74. package/dist/4928.js +1 -0
  75. package/dist/4928.js.map +1 -0
  76. package/dist/5069.js +1 -0
  77. package/dist/5069.js.map +1 -0
  78. package/dist/5117.js +1 -0
  79. package/dist/5117.js.map +1 -0
  80. package/dist/5132.js +1 -0
  81. package/dist/5132.js.map +1 -0
  82. package/dist/5145.js +1 -0
  83. package/dist/5145.js.map +1 -0
  84. package/dist/52.js +1 -0
  85. package/dist/52.js.map +1 -0
  86. package/dist/5422.js +1 -0
  87. package/dist/5422.js.map +1 -0
  88. package/dist/5503.js +1 -0
  89. package/dist/5503.js.map +1 -0
  90. package/dist/5549.js +1 -0
  91. package/dist/5549.js.map +1 -0
  92. package/dist/556.js +1 -0
  93. package/dist/556.js.map +1 -0
  94. package/dist/5644.js +1 -0
  95. package/dist/5644.js.map +1 -0
  96. package/dist/5697.js +1 -0
  97. package/dist/5697.js.map +1 -0
  98. package/dist/5793.js +1 -0
  99. package/dist/5793.js.map +1 -0
  100. package/dist/5940.js +1 -0
  101. package/dist/5940.js.map +1 -0
  102. package/dist/5952.js +1 -0
  103. package/dist/5952.js.map +1 -0
  104. package/dist/6047.js +1 -0
  105. package/dist/6047.js.map +1 -0
  106. package/dist/6371.js +1 -0
  107. package/dist/6371.js.map +1 -0
  108. package/dist/6377.js +1 -0
  109. package/dist/6377.js.map +1 -0
  110. package/dist/6444.js +1 -0
  111. package/dist/6444.js.map +1 -0
  112. package/dist/6479.js +1 -0
  113. package/dist/6479.js.map +1 -0
  114. package/dist/6508.js +1 -0
  115. package/dist/6508.js.map +1 -0
  116. package/dist/6724.js +1 -0
  117. package/dist/6724.js.map +1 -0
  118. package/dist/6759.js +27 -0
  119. package/dist/6759.js.map +1 -0
  120. package/dist/689.js +1 -0
  121. package/dist/689.js.map +1 -0
  122. package/dist/6904.js +1 -0
  123. package/dist/6904.js.map +1 -0
  124. package/dist/7045.js +1 -0
  125. package/dist/7045.js.map +1 -0
  126. package/dist/7175.js +1 -0
  127. package/dist/7175.js.map +1 -0
  128. package/dist/7182.js +1 -0
  129. package/dist/7182.js.map +1 -0
  130. package/dist/7302.js +1 -0
  131. package/dist/7302.js.map +1 -0
  132. package/dist/7646.js +17 -0
  133. package/dist/7646.js.map +1 -0
  134. package/dist/7742.js +1 -0
  135. package/dist/7742.js.map +1 -0
  136. package/dist/7912.js +1 -0
  137. package/dist/7912.js.map +1 -0
  138. package/dist/8105.js +21 -0
  139. package/dist/8105.js.map +1 -0
  140. package/dist/8202.js +1 -0
  141. package/dist/8202.js.map +1 -0
  142. package/dist/8349.js +1 -0
  143. package/dist/8349.js.map +1 -0
  144. package/dist/8358.js +1 -0
  145. package/dist/8358.js.map +1 -0
  146. package/dist/8359.js +1 -0
  147. package/dist/8359.js.map +1 -0
  148. package/dist/8695.js +1 -0
  149. package/dist/8695.js.map +1 -0
  150. package/dist/8702.js +1 -0
  151. package/dist/8702.js.map +1 -0
  152. package/dist/8894.js +1 -0
  153. package/dist/8894.js.map +1 -0
  154. package/dist/8958.js +1 -0
  155. package/dist/8958.js.map +1 -0
  156. package/dist/903.js +1 -0
  157. package/dist/903.js.map +1 -0
  158. package/dist/9061.js +1 -0
  159. package/dist/9061.js.map +1 -0
  160. package/dist/9072.js +1 -0
  161. package/dist/9072.js.map +1 -0
  162. package/dist/9105.js +1 -0
  163. package/dist/9105.js.map +1 -0
  164. package/dist/9107.js +1 -0
  165. package/dist/9107.js.map +1 -0
  166. package/dist/9456.js +1 -0
  167. package/dist/9456.js.map +1 -0
  168. package/dist/9586.js +1 -0
  169. package/dist/9586.js.map +1 -0
  170. package/dist/9712.js +1 -0
  171. package/dist/9712.js.map +1 -0
  172. package/dist/9771.js +1 -0
  173. package/dist/9771.js.map +1 -0
  174. package/dist/9806.js +1 -0
  175. package/dist/9806.js.map +1 -0
  176. package/dist/9873.js +1 -0
  177. package/dist/9873.js.map +1 -0
  178. package/dist/9927.js +1 -0
  179. package/dist/9927.js.map +1 -0
  180. package/dist/main.js +6 -0
  181. package/dist/main.js.map +1 -0
  182. package/dist/openmrs-esm-patient-chart-app.js +6 -0
  183. package/dist/openmrs-esm-patient-chart-app.js.buildmanifest.json +2386 -0
  184. package/dist/openmrs-esm-patient-chart-app.js.map +1 -0
  185. package/dist/routes.json +1 -0
  186. package/package.json +63 -0
  187. package/rspack.config.js +1 -0
  188. package/src/actions-buttons/action-button.scss +3 -0
  189. package/src/actions-buttons/delete-visit.component.tsx +41 -0
  190. package/src/actions-buttons/delete-visit.test.tsx +26 -0
  191. package/src/actions-buttons/mark-patient-alive.component.tsx +42 -0
  192. package/src/actions-buttons/mark-patient-deceased.component.tsx +35 -0
  193. package/src/actions-buttons/start-visit.component.tsx +41 -0
  194. package/src/actions-buttons/start-visit.test.tsx +44 -0
  195. package/src/actions-buttons/stop-visit.component.tsx +39 -0
  196. package/src/actions-buttons/stop-visit.test.tsx +27 -0
  197. package/src/clinical-views/encounter-list/encounter-list-tabs.extension.tsx +78 -0
  198. package/src/clinical-views/encounter-list/encounter-list-tabs.scss +7 -0
  199. package/src/clinical-views/encounter-list/encounter-list.component.tsx +306 -0
  200. package/src/clinical-views/encounter-list/encounter-list.scss +36 -0
  201. package/src/clinical-views/encounter-list/table.component.tsx +63 -0
  202. package/src/clinical-views/encounter-list/table.scss +11 -0
  203. package/src/clinical-views/encounter-list/tag.component.test.tsx +307 -0
  204. package/src/clinical-views/encounter-list/tag.component.tsx +43 -0
  205. package/src/clinical-views/encounter-tile/clinical-views-summary.component.tsx +40 -0
  206. package/src/clinical-views/encounter-tile/encounter-tile.component.tsx +94 -0
  207. package/src/clinical-views/encounter-tile/tile.scss +82 -0
  208. package/src/clinical-views/hooks/index.ts +3 -0
  209. package/src/clinical-views/hooks/useEncounterRows.ts +60 -0
  210. package/src/clinical-views/hooks/useEncountersByVisit.ts +13 -0
  211. package/src/clinical-views/hooks/useFormsJson.ts +15 -0
  212. package/src/clinical-views/hooks/useLastEncounter.ts +29 -0
  213. package/src/clinical-views/types.ts +305 -0
  214. package/src/clinical-views/utils/concept-utils.ts +24 -0
  215. package/src/clinical-views/utils/encounter-list-config-builder.ts +160 -0
  216. package/src/clinical-views/utils/encounter-list.resource.ts +26 -0
  217. package/src/clinical-views/utils/helpers.ts +226 -0
  218. package/src/clinical-views/utils/index.ts +90 -0
  219. package/src/config-schema.ts +235 -0
  220. package/src/constants.ts +11 -0
  221. package/src/dashboard.meta.ts +15 -0
  222. package/src/data.resource.ts +117 -0
  223. package/src/declarations.d.ts +4 -0
  224. package/src/index.ts +204 -0
  225. package/src/loader/loader.component.tsx +11 -0
  226. package/src/loader/loader.scss +9 -0
  227. package/src/mark-patient-alive/mark-patient-alive.modal.tsx +54 -0
  228. package/src/mark-patient-deceased/mark-patient-deceased-form.scss +175 -0
  229. package/src/mark-patient-deceased/mark-patient-deceased-form.test.tsx +203 -0
  230. package/src/mark-patient-deceased/mark-patient-deceased-form.workspace.tsx +295 -0
  231. package/src/offline.ts +41 -0
  232. package/src/patient-banner-tags/visit-attribute-tags.extension.tsx +62 -0
  233. package/src/patient-banner-tags/visit-attribute-tags.scss +8 -0
  234. package/src/patient-chart/chart-review/chart-review.component.tsx +138 -0
  235. package/src/patient-chart/chart-review/chart-review.test.tsx +77 -0
  236. package/src/patient-chart/chart-review/dashboard-view.component.tsx +85 -0
  237. package/src/patient-chart/chart-review/dashboard-view.scss +84 -0
  238. package/src/patient-chart/patient-chart.component.tsx +71 -0
  239. package/src/patient-chart/patient-chart.resources.test.ts +238 -0
  240. package/src/patient-chart/patient-chart.resources.ts +231 -0
  241. package/src/patient-chart/patient-chart.scss +65 -0
  242. package/src/patient-details-tile/patient-details-tile.component.tsx +25 -0
  243. package/src/patient-details-tile/patient-details-tile.scss +24 -0
  244. package/src/root.component.tsx +35 -0
  245. package/src/root.scss +54 -0
  246. package/src/routes.json +267 -0
  247. package/src/side-nav/side-menu.component.tsx +10 -0
  248. package/src/side-nav/side-menu.scss +38 -0
  249. package/src/side-nav/side-menu.test.tsx +27 -0
  250. package/src/utils.test.ts +17 -0
  251. package/src/utils.ts +5 -0
  252. package/src/visit/hooks/useDefaultFacilityLocation.tsx +15 -0
  253. package/src/visit/hooks/useDefaultVisitLocation.tsx +24 -0
  254. package/src/visit/hooks/useDeleteVisit.test.tsx +267 -0
  255. package/src/visit/hooks/useDeleteVisit.tsx +103 -0
  256. package/src/visit/hooks/useOfflineVisitType.tsx +18 -0
  257. package/src/visit/hooks/useRecommendedVisitTypes.tsx +34 -0
  258. package/src/visit/hooks/useVisitAttributeType.tsx +102 -0
  259. package/src/visit/start-visit-button.component.tsx +47 -0
  260. package/src/visit/start-visit-button.test.tsx +32 -0
  261. package/src/visit/visit-action-items/delete-visit-action-item.component.tsx +60 -0
  262. package/src/visit/visit-action-items/delete-visit-action-item.test.tsx +48 -0
  263. package/src/visit/visit-action-items/edit-visit-details.component.tsx +79 -0
  264. package/src/visit/visit-form/base-visit-type.component.tsx +121 -0
  265. package/src/visit/visit-form/base-visit-type.scss +75 -0
  266. package/src/visit/visit-form/base-visit-type.test.tsx +153 -0
  267. package/src/visit/visit-form/exported-visit-form.workspace.tsx +755 -0
  268. package/src/visit/visit-form/location-selector.component.tsx +86 -0
  269. package/src/visit/visit-form/location-selector.test.tsx +146 -0
  270. package/src/visit/visit-form/recommended-visit-type.component.tsx +32 -0
  271. package/src/visit/visit-form/visit-attribute-type.component.tsx +258 -0
  272. package/src/visit/visit-form/visit-attribute-type.scss +5 -0
  273. package/src/visit/visit-form/visit-date-time.component.tsx +206 -0
  274. package/src/visit/visit-form/visit-form.resource.ts +401 -0
  275. package/src/visit/visit-form/visit-form.scss +167 -0
  276. package/src/visit/visit-form/visit-form.test.tsx +1233 -0
  277. package/src/visit/visit-form/visit-form.workspace.tsx +61 -0
  278. package/src/visit/visit-form/visit-type.test.tsx +88 -0
  279. package/src/visit/visit-history-table/visit-actions-cell.component.tsx +20 -0
  280. package/src/visit/visit-history-table/visit-actions-cell.scss +4 -0
  281. package/src/visit/visit-history-table/visit-date-cell.component.tsx +19 -0
  282. package/src/visit/visit-history-table/visit-diagnoses-cell.component.tsx +18 -0
  283. package/src/visit/visit-history-table/visit-history-table.component.tsx +145 -0
  284. package/src/visit/visit-history-table/visit-history-table.scss +25 -0
  285. package/src/visit/visit-history-table/visit-type-cell.component.tsx +15 -0
  286. package/src/visit/visit-prompt/delete-visit-dialog.modal.tsx +46 -0
  287. package/src/visit/visit-prompt/delete-visit-dialog.test.tsx +79 -0
  288. package/src/visit/visit-prompt/end-visit-dialog.modal.tsx +82 -0
  289. package/src/visit/visit-prompt/end-visit-dialog.scss +7 -0
  290. package/src/visit/visit-prompt/end-visit-dialog.test.tsx +131 -0
  291. package/src/visit/visit-prompt/modify-visit-date.modal.tsx +40 -0
  292. package/src/visit/visit-prompt/start-visit-dialog.modal.tsx +64 -0
  293. package/src/visit/visit-prompt/start-visit-dialog.scss +10 -0
  294. package/src/visit/visit-prompt/start-visit-dialog.test.tsx +40 -0
  295. package/src/visit/visits-widget/active-visit-buttons/active-visit-buttons.scss +7 -0
  296. package/src/visit/visits-widget/active-visit-buttons/active-visit-buttons.tsx +178 -0
  297. package/src/visit/visits-widget/current-visit-summary.extension.tsx +48 -0
  298. package/src/visit/visits-widget/current-visit-summary.scss +10 -0
  299. package/src/visit/visits-widget/current-visit-summary.test.tsx +85 -0
  300. package/src/visit/visits-widget/encounter-observations/encounter-observations.component.tsx +67 -0
  301. package/src/visit/visits-widget/encounter-observations/index.ts +3 -0
  302. package/src/visit/visits-widget/encounter-observations/styles.scss +22 -0
  303. package/src/visit/visits-widget/past-visits-components/delete-encounter.modal.tsx +47 -0
  304. package/src/visit/visits-widget/past-visits-components/delete-encounter.scss +9 -0
  305. package/src/visit/visits-widget/past-visits-components/encounters-table/all-encounters-table.component.tsx +49 -0
  306. package/src/visit/visits-widget/past-visits-components/encounters-table/completed-forms-table.component.tsx +67 -0
  307. package/src/visit/visits-widget/past-visits-components/encounters-table/completed-forms-table.test.tsx +146 -0
  308. package/src/visit/visits-widget/past-visits-components/encounters-table/encounters-table.component.tsx +452 -0
  309. package/src/visit/visits-widget/past-visits-components/encounters-table/encounters-table.resource.test.ts +156 -0
  310. package/src/visit/visits-widget/past-visits-components/encounters-table/encounters-table.resource.ts +215 -0
  311. package/src/visit/visits-widget/past-visits-components/encounters-table/encounters-table.scss +113 -0
  312. package/src/visit/visits-widget/past-visits-components/encounters-table/encounters-table.test.tsx +432 -0
  313. package/src/visit/visits-widget/past-visits-components/encounters-table/visit-completed-forms-table.component.tsx +61 -0
  314. package/src/visit/visits-widget/past-visits-components/encounters-table/visit-completed-forms-table.test.tsx +125 -0
  315. package/src/visit/visits-widget/past-visits-components/encounters-table/visit-encounters-table.component.tsx +47 -0
  316. package/src/visit/visits-widget/past-visits-components/medications-summary.component.tsx +163 -0
  317. package/src/visit/visits-widget/past-visits-components/notes-summary.component.tsx +66 -0
  318. package/src/visit/visits-widget/past-visits-components/patient-notes-summary.component.tsx +318 -0
  319. package/src/visit/visits-widget/past-visits-components/tests-summary.component.tsx +16 -0
  320. package/src/visit/visits-widget/past-visits-components/visit-summary.component.tsx +192 -0
  321. package/src/visit/visits-widget/past-visits-components/visit-summary.scss +72 -0
  322. package/src/visit/visits-widget/past-visits-components/visit-summary.test.tsx +105 -0
  323. package/src/visit/visits-widget/single-visit-details/visit-timeline/visit-timeline.component.tsx +94 -0
  324. package/src/visit/visits-widget/single-visit-details/visit-timeline/visit-timeline.scss +60 -0
  325. package/src/visit/visits-widget/visit-context/retrospective-data-date-time-picker/restrospective-date-time-picker.scss +35 -0
  326. package/src/visit/visits-widget/visit-context/retrospective-data-date-time-picker/retrospective-date-time-picker.component.tsx +140 -0
  327. package/src/visit/visits-widget/visit-context/visit-context-header.extension.tsx +61 -0
  328. package/src/visit/visits-widget/visit-context/visit-context-header.scss +45 -0
  329. package/src/visit/visits-widget/visit-context/visit-context-header.test.tsx +59 -0
  330. package/src/visit/visits-widget/visit-context/visit-context-info.component.tsx +37 -0
  331. package/src/visit/visits-widget/visit-context/visit-context-info.scss +12 -0
  332. package/src/visit/visits-widget/visit-context/visit-context-switcher.modal.tsx +166 -0
  333. package/src/visit/visits-widget/visit-context/visit-context-switcher.scss +83 -0
  334. package/src/visit/visits-widget/visit-context/visit-context-switcher.test.tsx +79 -0
  335. package/src/visit/visits-widget/visit-detail-overview.component.tsx +67 -0
  336. package/src/visit/visits-widget/visit-detail-overview.scss +301 -0
  337. package/src/visit/visits-widget/visit-detail-overview.test.tsx +205 -0
  338. package/src/visit/visits-widget/visit.resource.tsx +146 -0
  339. package/translations/am.json +209 -0
  340. package/translations/ar.json +209 -0
  341. package/translations/ar_SY.json +209 -0
  342. package/translations/bn.json +209 -0
  343. package/translations/cs.json +209 -0
  344. package/translations/de.json +209 -0
  345. package/translations/en.json +209 -0
  346. package/translations/en_US.json +209 -0
  347. package/translations/es.json +209 -0
  348. package/translations/es_MX.json +209 -0
  349. package/translations/fr.json +209 -0
  350. package/translations/he.json +209 -0
  351. package/translations/hi.json +209 -0
  352. package/translations/hi_IN.json +209 -0
  353. package/translations/id.json +209 -0
  354. package/translations/it.json +209 -0
  355. package/translations/ka.json +209 -0
  356. package/translations/km.json +209 -0
  357. package/translations/ku.json +209 -0
  358. package/translations/ky.json +209 -0
  359. package/translations/lg.json +209 -0
  360. package/translations/ne.json +209 -0
  361. package/translations/pl.json +209 -0
  362. package/translations/pt.json +209 -0
  363. package/translations/pt_BR.json +209 -0
  364. package/translations/qu.json +209 -0
  365. package/translations/ro_RO.json +209 -0
  366. package/translations/ru_RU.json +209 -0
  367. package/translations/si.json +209 -0
  368. package/translations/sq.json +209 -0
  369. package/translations/sw.json +209 -0
  370. package/translations/sw_KE.json +209 -0
  371. package/translations/tr.json +209 -0
  372. package/translations/tr_TR.json +209 -0
  373. package/translations/uk.json +209 -0
  374. package/translations/uz.json +209 -0
  375. package/translations/uz@Latn.json +209 -0
  376. package/translations/uz_UZ.json +209 -0
  377. package/translations/vi.json +209 -0
  378. package/translations/zh.json +209 -0
  379. package/translations/zh_CN.json +209 -0
  380. package/translations/zh_TW.json +209 -0
  381. package/tsconfig.json +4 -0
  382. package/vitest.config.ts +4 -0
@@ -0,0 +1,85 @@
1
+ import React, { useEffect, useMemo, useState } from 'react';
2
+ import classNames from 'classnames';
3
+ import { useMatch } from 'react-router-dom';
4
+ import { useTranslation } from 'react-i18next';
5
+ import { launchWorkspace2, Extension, ExtensionSlot, useExtensionSlotMeta } from '@openmrs/esm-framework';
6
+ import { launchStartVisitPrompt } from '@openmrs/esm-patient-common-lib';
7
+ import { dashboardPath } from '../../constants';
8
+ import styles from './dashboard-view.scss';
9
+
10
+ /**
11
+ * The layout mode dictates the width occuppied by the chart dashboard widgets.
12
+ * - In 'contained' mode, the dashboard widgets are displayed in a centered
13
+ * container with a fixed width (max-width: 60rem).
14
+ * - In 'anchored' mode, the dashboard widgets expand to occupy the entire width
15
+ * of the chart dashboard
16
+ */
17
+ export type LayoutMode = 'contained' | 'anchored';
18
+
19
+ export interface DashboardConfig {
20
+ slot: string;
21
+ title: string | (() => string | Promise<string>);
22
+ path: string;
23
+ hideDashboardTitle?: boolean;
24
+ layoutMode?: LayoutMode;
25
+ moduleName: string;
26
+ }
27
+
28
+ interface DashboardViewProps {
29
+ dashboard: DashboardConfig;
30
+ patientUuid: string;
31
+ patient: fhir.Patient;
32
+ }
33
+
34
+ export function DashboardView({ dashboard, patientUuid, patient }: DashboardViewProps) {
35
+ const widgetMetas = useExtensionSlotMeta(dashboard.slot);
36
+ const { t } = useTranslation();
37
+ const {
38
+ params: { view },
39
+ } = useMatch(dashboardPath);
40
+
41
+ const state = useMemo(
42
+ () => ({
43
+ basePath: view,
44
+ patient,
45
+ patientUuid,
46
+ launchWorkspace2,
47
+ launchStartVisitPrompt,
48
+ launchAppointmentForm: () => {
49
+ launchWorkspace2('patient-chart-appointments-form-workspace', {}, {}, { patientUuid, patient });
50
+ },
51
+ }),
52
+ [patient, patientUuid, view],
53
+ );
54
+
55
+ const [resolvedTitle, setResolvedTitle] = useState<string | undefined>();
56
+
57
+ useEffect(() => {
58
+ if (typeof dashboard?.title === 'function') {
59
+ Promise.resolve(dashboard.title()).then(setResolvedTitle);
60
+ } else if (typeof dashboard?.title === 'string') {
61
+ setResolvedTitle(dashboard.title);
62
+ } else {
63
+ setResolvedTitle(undefined);
64
+ }
65
+ }, [dashboard]);
66
+
67
+ return (
68
+ <>
69
+ <ExtensionSlot state={state} name="top-of-all-patient-dashboards-slot" />
70
+ {!dashboard.hideDashboardTitle && resolvedTitle && <h1 className={styles.dashboardTitle}>{t(resolvedTitle)}</h1>}
71
+ <div className={styles.dashboardContainer}>
72
+ <ExtensionSlot key={dashboard.slot} name={dashboard.slot} className={styles.dashboard}>
73
+ {(extension) => {
74
+ const { fullWidth = false } = widgetMetas[extension.id] || {};
75
+ return (
76
+ <div className={classNames(styles.extension, fullWidth && styles.fullWidth)}>
77
+ <Extension state={state} className={styles.extensionWrapper} />
78
+ </div>
79
+ );
80
+ }}
81
+ </ExtensionSlot>
82
+ </div>
83
+ </>
84
+ );
85
+ }
@@ -0,0 +1,84 @@
1
+ @use '@carbon/layout';
2
+ @use '@carbon/type';
3
+
4
+ .dashboardTitle {
5
+ @include type.type-style('heading-03');
6
+ margin: layout.$spacing-05 0 layout.$spacing-05 1.3125rem;
7
+ }
8
+
9
+ .dashboard {
10
+ display: grid;
11
+ grid-template-columns: repeat(2, minmax(0, 1fr));
12
+ grid-auto-rows: auto;
13
+ grid-gap: 1.3125rem;
14
+ margin: 1.3125rem;
15
+ // Hide empty extension wrappers when components return null
16
+ // This prevents empty space in the grid when supplementary widgets have no data
17
+ > :has([data-extension-id='immunization-detailed-history-card']:empty) {
18
+ display: none;
19
+ }
20
+ }
21
+
22
+ // See https://zpl.io/lrlmdq0 for the Visits dashboard design
23
+ [data-extension-slot-name='patient-chart-encounters-dashboard-slot'],
24
+ // See https://zpl.io/RmMXrDE for the Test Results dashboard design
25
+ [data-extension-slot-name='patient-chart-test-results-dashboard-slot'] {
26
+ margin: 0 layout.$spacing-05;
27
+ }
28
+
29
+ .dashboardContainer:not(:has([data-extension-slot-name='patient-chart-attachments-dashboard-slot'])) {
30
+ container-name: dashboard;
31
+ container-type: inline-size;
32
+ }
33
+
34
+ // When the dashboard's width is less than or equal to 68.25rem (1100px),
35
+ // the layout should switch to a single column. This adjustment ensures
36
+ // proper rendering when the workspace area is opened.
37
+ @container dashboard (width <= 68.25rem) {
38
+ .dashboard {
39
+ grid-template-columns: 1fr;
40
+ }
41
+ }
42
+
43
+ .extensionWrapper {
44
+ > * {
45
+ height: 100%;
46
+ display: flex;
47
+ flex-direction: column;
48
+ }
49
+ }
50
+
51
+ .extension:only-child {
52
+ grid-column: 1 / -1;
53
+ }
54
+
55
+ .fullWidth {
56
+ grid-column: 1 / -1;
57
+ }
58
+
59
+ :global(.omrs-breakpoint-lt-desktop) .dashboard {
60
+ grid-template-columns: 1fr;
61
+ }
62
+
63
+ :global(.omrs-breakpoint-lt-tablet) .container {
64
+ margin: 5px;
65
+ overflow-x: auto;
66
+ justify-content: center;
67
+ }
68
+
69
+ :global(.omrs-breakpoint-lt-tablet) .dashboard {
70
+ display: flex;
71
+ flex-direction: column;
72
+ flex-wrap: wrap;
73
+ }
74
+
75
+ :global(.omrs-breakpoint-lt-tablet) .dashboard > div {
76
+ margin: layout.$spacing-01 0px;
77
+ }
78
+
79
+ // Overriding styles for RTL support
80
+ html[dir='rtl'] {
81
+ .dashboardTitle {
82
+ margin: layout.$spacing-05 1.3125rem layout.$spacing-05 0;
83
+ }
84
+ }
@@ -0,0 +1,71 @@
1
+ import { ExtensionSlot, useLeftNav } from '@openmrs/esm-framework';
2
+ import classNames from 'classnames';
3
+ import React, { useMemo, useState } from 'react';
4
+ import { useParams } from 'react-router-dom';
5
+ import { spaBasePath } from '../constants';
6
+ import Loader from '../loader/loader.component';
7
+ import ChartReview from '../patient-chart/chart-review/chart-review.component';
8
+ import SideMenuPanel from '../side-nav/side-menu.component';
9
+ import { type LayoutMode } from './chart-review/dashboard-view.component';
10
+ import styles from './patient-chart.scss';
11
+ import { usePatientChartPatientAndVisit } from './patient-chart.resources';
12
+
13
+ const PatientChart: React.FC = () => {
14
+ const { patientUuid, view: encodedView } = useParams();
15
+
16
+ // specify key to ensure that WrapPatientChart instance is re-created
17
+ // when we switch patient
18
+ return <WrappedPatientChart key={patientUuid} patientUuid={patientUuid} encodedView={encodedView} />;
19
+ };
20
+
21
+ interface WrappedPatientChartProps {
22
+ patientUuid: string;
23
+ encodedView: string;
24
+ }
25
+
26
+ const WrappedPatientChart: React.FC<WrappedPatientChartProps> = ({ patientUuid, encodedView }) => {
27
+ const view = decodeURIComponent(encodedView);
28
+ const [layoutMode, setLayoutMode] = useState<LayoutMode>();
29
+ const state = usePatientChartPatientAndVisit(patientUuid);
30
+ const { isLoadingPatient, patient } = state;
31
+
32
+ const leftNavBasePath = useMemo(() => spaBasePath.replace(':patientUuid', patientUuid), [patientUuid]);
33
+
34
+ useLeftNav({ name: 'patient-chart-dashboard-slot', basePath: leftNavBasePath });
35
+
36
+ return (
37
+ <>
38
+ <SideMenuPanel />
39
+ <main className={classNames('omrs-main-content', styles.chartContainer)}>
40
+ <>
41
+ <div className={classNames(styles.innerChartContainer)}>
42
+ {isLoadingPatient ? (
43
+ <Loader />
44
+ ) : (
45
+ <>
46
+ <aside>
47
+ <ExtensionSlot name="patient-header-slot" state={state} />
48
+ <ExtensionSlot name="patient-info-slot" state={state} className={styles.patientInfoSlot} />
49
+ </aside>
50
+ <div className={styles.grid}>
51
+ <div
52
+ className={classNames(styles.chartReview, { [styles.widthContained]: layoutMode == 'contained' })}
53
+ >
54
+ <ChartReview
55
+ patient={patient}
56
+ patientUuid={patientUuid}
57
+ view={view}
58
+ setDashboardLayoutMode={setLayoutMode}
59
+ />
60
+ </div>
61
+ </div>
62
+ </>
63
+ )}
64
+ </div>
65
+ </>
66
+ </main>
67
+ </>
68
+ );
69
+ };
70
+
71
+ export default PatientChart;
@@ -0,0 +1,238 @@
1
+ import { renderHook, waitFor } from '@testing-library/react';
2
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
3
+ import { launchWorkspaceGroup2, usePatient, useVisit, type Visit } from '@openmrs/esm-framework';
4
+ import { usePatientChartStore } from '@openmrs/esm-patient-common-lib';
5
+ import { mockFhirPatient } from '__mocks__';
6
+ import { usePatientChartPatientAndVisit } from './patient-chart.resources';
7
+
8
+ // TODO: remove once openmrs-esm-core ships launchWorkspaceGroup2 in the framework mock
9
+ vi.mock('@openmrs/esm-framework', async () => {
10
+ const actual = (await vi.importActual('@openmrs/esm-framework')) as object;
11
+ return { ...actual, launchWorkspaceGroup2: vi.fn() };
12
+ });
13
+
14
+ vi.mock('@openmrs/esm-patient-common-lib', () => ({
15
+ usePatientChartStore: vi.fn(),
16
+ }));
17
+
18
+ const mockLaunchWorkspaceGroup = vi.mocked(launchWorkspaceGroup2);
19
+ const mockUsePatient = vi.mocked(usePatient);
20
+ const mockUseVisit = vi.mocked(useVisit);
21
+ const mockUsePatientChartStore = vi.mocked(usePatientChartStore);
22
+
23
+ const mutateVisitContext = vi.fn();
24
+ const setPatient = vi.fn();
25
+ const setVisitContext = vi.fn();
26
+
27
+ const visitA = { uuid: 'visit-a', patient: { uuid: mockFhirPatient.id } } as Visit;
28
+ const visitB = { uuid: 'visit-b', patient: { uuid: mockFhirPatient.id } } as Visit;
29
+ const patientB = { ...mockFhirPatient, id: 'patient-b' };
30
+
31
+ function mockNoActiveVisitForPatientSwitch() {
32
+ mockUsePatient.mockImplementation((patientUuid) => ({
33
+ isLoading: false,
34
+ patient: patientUuid === patientB.id ? patientB : mockFhirPatient,
35
+ patientUuid,
36
+ error: null,
37
+ }));
38
+ mockUseVisit.mockImplementation(() => ({
39
+ activeVisit: null,
40
+ mutate: mutateVisitContext,
41
+ isValidating: false,
42
+ error: null,
43
+ currentVisit: null,
44
+ currentVisitIsRetrospective: false,
45
+ isLoading: false,
46
+ }));
47
+ mockUsePatientChartStore.mockImplementation((patientUuid) => ({
48
+ patientUuid,
49
+ patient: patientUuid === patientB.id ? patientB : mockFhirPatient,
50
+ visitContext: null,
51
+ mutateVisitContext: null,
52
+ setPatient,
53
+ setVisitContext,
54
+ }));
55
+ }
56
+
57
+ describe('usePatientChartPatientAndVisit', () => {
58
+ beforeEach(() => {
59
+ vi.clearAllMocks();
60
+ mockLaunchWorkspaceGroup.mockResolvedValue(true);
61
+ mockUsePatient.mockReturnValue({
62
+ isLoading: false,
63
+ patient: mockFhirPatient,
64
+ patientUuid: mockFhirPatient.id,
65
+ error: null,
66
+ });
67
+ mockUsePatientChartStore.mockReturnValue({
68
+ patientUuid: mockFhirPatient.id,
69
+ patient: mockFhirPatient,
70
+ visitContext: null,
71
+ mutateVisitContext: null,
72
+ setPatient,
73
+ setVisitContext,
74
+ });
75
+ });
76
+
77
+ it('relaunches the patient-chart workspace group when the visit context changes', async () => {
78
+ let activeVisit = visitA;
79
+ mockUseVisit.mockImplementation(() => ({
80
+ activeVisit,
81
+ mutate: mutateVisitContext,
82
+ isValidating: false,
83
+ error: null,
84
+ currentVisit: null,
85
+ currentVisitIsRetrospective: false,
86
+ isLoading: false,
87
+ }));
88
+
89
+ const { rerender } = renderHook(() => usePatientChartPatientAndVisit(mockFhirPatient.id));
90
+
91
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1));
92
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
93
+ 'patient-chart',
94
+ expect.objectContaining({
95
+ patient: mockFhirPatient,
96
+ patientUuid: mockFhirPatient.id,
97
+ visitContext: visitA,
98
+ mutateVisitContext,
99
+ }),
100
+ );
101
+
102
+ activeVisit = visitB;
103
+ rerender();
104
+
105
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(2));
106
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
107
+ 'patient-chart',
108
+ expect.objectContaining({
109
+ patient: mockFhirPatient,
110
+ patientUuid: mockFhirPatient.id,
111
+ visitContext: visitB,
112
+ mutateVisitContext,
113
+ }),
114
+ );
115
+ });
116
+
117
+ it('does not relaunch when the same visit UUID is returned with new object references', async () => {
118
+ let activeVisit: Visit = visitA;
119
+ mockUseVisit.mockImplementation(() => ({
120
+ activeVisit,
121
+ mutate: mutateVisitContext,
122
+ isValidating: false,
123
+ error: null,
124
+ currentVisit: null,
125
+ currentVisitIsRetrospective: false,
126
+ isLoading: false,
127
+ }));
128
+
129
+ const { rerender } = renderHook(() => usePatientChartPatientAndVisit(mockFhirPatient.id));
130
+
131
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1));
132
+
133
+ // Same UUID, new object reference — mimics SWR revalidation
134
+ activeVisit = { ...visitA, stopDatetime: '2026-05-26T00:00:00.000Z' } as Visit;
135
+ rerender();
136
+
137
+ // Wait for the effect to fire before asserting the launch count didn't increase
138
+ await waitFor(() => expect(setVisitContext).toHaveBeenCalledTimes(2));
139
+ expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1);
140
+ });
141
+
142
+ it('relaunches when the patient changes and neither patient has an active visit', async () => {
143
+ mockNoActiveVisitForPatientSwitch();
144
+
145
+ const { rerender } = renderHook(({ patientUuid }) => usePatientChartPatientAndVisit(patientUuid), {
146
+ initialProps: { patientUuid: mockFhirPatient.id },
147
+ });
148
+
149
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1));
150
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
151
+ 'patient-chart',
152
+ expect.objectContaining({ patientUuid: mockFhirPatient.id, visitContext: null }),
153
+ );
154
+
155
+ rerender({ patientUuid: patientB.id });
156
+
157
+ await waitFor(() => expect(setVisitContext).toHaveBeenCalledTimes(2));
158
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(2));
159
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
160
+ 'patient-chart',
161
+ expect.objectContaining({ patientUuid: patientB.id, visitContext: null }),
162
+ );
163
+ });
164
+
165
+ it('launches the latest patient after a previous launch resolves', async () => {
166
+ let patientUuid = mockFhirPatient.id;
167
+ let resolveFirstLaunch!: (value: boolean) => void;
168
+ const firstLaunch = new Promise<boolean>((resolve) => {
169
+ resolveFirstLaunch = resolve;
170
+ });
171
+
172
+ mockLaunchWorkspaceGroup.mockReturnValueOnce(firstLaunch).mockResolvedValue(true);
173
+ mockNoActiveVisitForPatientSwitch();
174
+
175
+ const { rerender } = renderHook(() => usePatientChartPatientAndVisit(patientUuid));
176
+
177
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1));
178
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
179
+ 'patient-chart',
180
+ expect.objectContaining({ patientUuid: mockFhirPatient.id, visitContext: null }),
181
+ );
182
+
183
+ patientUuid = patientB.id;
184
+ rerender();
185
+
186
+ await waitFor(() => expect(setVisitContext).toHaveBeenCalledTimes(2));
187
+ expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1);
188
+
189
+ resolveFirstLaunch(true);
190
+
191
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(2));
192
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
193
+ 'patient-chart',
194
+ expect.objectContaining({ patientUuid: patientB.id, visitContext: null }),
195
+ );
196
+ });
197
+
198
+ it('launches the latest visit context after a previous launch resolves', async () => {
199
+ let activeVisit = visitA;
200
+ let resolveFirstLaunch!: (value: boolean) => void;
201
+ const firstLaunch = new Promise<boolean>((resolve) => {
202
+ resolveFirstLaunch = resolve;
203
+ });
204
+
205
+ mockLaunchWorkspaceGroup.mockReturnValueOnce(firstLaunch).mockResolvedValue(true);
206
+ mockUseVisit.mockImplementation(() => ({
207
+ activeVisit,
208
+ mutate: mutateVisitContext,
209
+ isValidating: false,
210
+ error: null,
211
+ currentVisit: null,
212
+ currentVisitIsRetrospective: false,
213
+ isLoading: false,
214
+ }));
215
+
216
+ const { rerender } = renderHook(() => usePatientChartPatientAndVisit(mockFhirPatient.id));
217
+
218
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1));
219
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
220
+ 'patient-chart',
221
+ expect.objectContaining({ visitContext: visitA }),
222
+ );
223
+
224
+ activeVisit = visitB;
225
+ rerender();
226
+
227
+ await waitFor(() => expect(setVisitContext).toHaveBeenCalledTimes(2));
228
+ expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(1);
229
+
230
+ resolveFirstLaunch(true);
231
+
232
+ await waitFor(() => expect(mockLaunchWorkspaceGroup).toHaveBeenCalledTimes(2));
233
+ expect(mockLaunchWorkspaceGroup).toHaveBeenLastCalledWith(
234
+ 'patient-chart',
235
+ expect.objectContaining({ visitContext: visitB }),
236
+ );
237
+ });
238
+ });