vintasend 0.4.0 → 0.4.7

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 (284) hide show
  1. package/README.md +5 -0
  2. package/dist/examples/vintasend-medplum-example/.env.example +11 -0
  3. package/dist/examples/vintasend-medplum-example/IMPLEMENTATION_PLAN_FILE_ATTACHMENTS.md +597 -0
  4. package/dist/examples/vintasend-medplum-example/README.md +190 -0
  5. package/dist/examples/vintasend-medplum-example/TUTORIAL_EMAIL_NOTIFICATIONS.md +2596 -0
  6. package/dist/examples/vintasend-medplum-example/bots/handlers/send-pending-notifications-bot.ts +39 -0
  7. package/dist/examples/vintasend-medplum-example/bots/handlers/send-task-assignment-email.ts +41 -0
  8. package/dist/examples/vintasend-medplum-example/bots/handlers/task-due-soon-notification-bot.ts +86 -0
  9. package/dist/examples/vintasend-medplum-example/bots/index.ts +53 -0
  10. package/dist/examples/vintasend-medplum-example/bots/services/emails/schedule-task-due-soon-email.ts +84 -0
  11. package/dist/examples/vintasend-medplum-example/bots/services/emails/send-task-assignment-email.test.ts +388 -0
  12. package/dist/examples/vintasend-medplum-example/bots/services/emails/send-task-assignment-email.ts +113 -0
  13. package/dist/examples/vintasend-medplum-example/bots/shared/task-due-soon-helpers.ts +115 -0
  14. package/dist/examples/vintasend-medplum-example/bots/task-assignment-bot.ts +41 -0
  15. package/dist/examples/vintasend-medplum-example/compiled-notification-templates.json +6 -0
  16. package/dist/examples/vintasend-medplum-example/esbuild-script.mjs +71 -0
  17. package/dist/examples/vintasend-medplum-example/index.html +14 -0
  18. package/dist/examples/vintasend-medplum-example/lib/constants.ts +32 -0
  19. package/dist/examples/vintasend-medplum-example/lib/extensions.ts +1 -0
  20. package/dist/examples/vintasend-medplum-example/lib/file-upload.test.ts +389 -0
  21. package/dist/examples/vintasend-medplum-example/lib/file-upload.ts +222 -0
  22. package/dist/examples/vintasend-medplum-example/lib/medplum-singleton.ts +18 -0
  23. package/dist/examples/vintasend-medplum-example/lib/notification-service.test.ts +293 -0
  24. package/dist/examples/vintasend-medplum-example/lib/notification-service.ts +284 -0
  25. package/dist/examples/vintasend-medplum-example/lib/patients.ts +20 -0
  26. package/dist/examples/vintasend-medplum-example/notification-templates/emails/task-assignment/body.html.pug +37 -0
  27. package/dist/examples/vintasend-medplum-example/notification-templates/emails/task-assignment/subject.txt.pug +4 -0
  28. package/dist/examples/vintasend-medplum-example/notification-templates/emails/task-due-soon/body.html.pug +34 -0
  29. package/dist/examples/vintasend-medplum-example/notification-templates/emails/task-due-soon/subject.txt.pug +4 -0
  30. package/dist/examples/vintasend-medplum-example/package.json +75 -0
  31. package/dist/examples/vintasend-medplum-example/plugins/gql-plugin.mjs +31 -0
  32. package/dist/examples/vintasend-medplum-example/postcss.config.mjs +21 -0
  33. package/dist/examples/vintasend-medplum-example/public/favicon.ico +0 -0
  34. package/dist/examples/vintasend-medplum-example/public/img/integrations/acuity.png +0 -0
  35. package/dist/examples/vintasend-medplum-example/public/img/integrations/auth0.png +0 -0
  36. package/dist/examples/vintasend-medplum-example/public/img/integrations/azure.png +0 -0
  37. package/dist/examples/vintasend-medplum-example/public/img/integrations/calcom.png +0 -0
  38. package/dist/examples/vintasend-medplum-example/public/img/integrations/candid.png +0 -0
  39. package/dist/examples/vintasend-medplum-example/public/img/integrations/claude.png +0 -0
  40. package/dist/examples/vintasend-medplum-example/public/img/integrations/datadog.png +0 -0
  41. package/dist/examples/vintasend-medplum-example/public/img/integrations/deepseek.png +0 -0
  42. package/dist/examples/vintasend-medplum-example/public/img/integrations/entra.png +0 -0
  43. package/dist/examples/vintasend-medplum-example/public/img/integrations/epic.png +0 -0
  44. package/dist/examples/vintasend-medplum-example/public/img/integrations/google.png +0 -0
  45. package/dist/examples/vintasend-medplum-example/public/img/integrations/healthgorilla.png +0 -0
  46. package/dist/examples/vintasend-medplum-example/public/img/integrations/healthie.png +0 -0
  47. package/dist/examples/vintasend-medplum-example/public/img/integrations/labcorp.png +0 -0
  48. package/dist/examples/vintasend-medplum-example/public/img/integrations/okta.png +0 -0
  49. package/dist/examples/vintasend-medplum-example/public/img/integrations/openai.png +0 -0
  50. package/dist/examples/vintasend-medplum-example/public/img/integrations/particle.png +0 -0
  51. package/dist/examples/vintasend-medplum-example/public/img/integrations/quest.png +0 -0
  52. package/dist/examples/vintasend-medplum-example/public/img/integrations/recaptcha.png +0 -0
  53. package/dist/examples/vintasend-medplum-example/public/img/integrations/snowflake.png +0 -0
  54. package/dist/examples/vintasend-medplum-example/public/img/integrations/stedi.png +0 -0
  55. package/dist/examples/vintasend-medplum-example/public/img/integrations/stripe.png +0 -0
  56. package/dist/examples/vintasend-medplum-example/public/img/integrations/sumo.png +0 -0
  57. package/dist/examples/vintasend-medplum-example/scripts/README.md +162 -0
  58. package/dist/examples/vintasend-medplum-example/scripts/client.ts +18 -0
  59. package/dist/examples/vintasend-medplum-example/scripts/deploy-bots.ts +171 -0
  60. package/dist/examples/vintasend-medplum-example/src/App.tsx +185 -0
  61. package/dist/examples/vintasend-medplum-example/src/components/ChargeItem/ChargeItemList.test.tsx +350 -0
  62. package/dist/examples/vintasend-medplum-example/src/components/ChargeItem/ChargeItemList.tsx +241 -0
  63. package/dist/examples/vintasend-medplum-example/src/components/ChargeItem/ChargeItemPanel.test.tsx +616 -0
  64. package/dist/examples/vintasend-medplum-example/src/components/ChargeItem/ChargeItemPanel.tsx +138 -0
  65. package/dist/examples/vintasend-medplum-example/src/components/Conditions/ConditionItem.test.tsx +92 -0
  66. package/dist/examples/vintasend-medplum-example/src/components/Conditions/ConditionItem.tsx +47 -0
  67. package/dist/examples/vintasend-medplum-example/src/components/Conditions/ConditionList.test.tsx +464 -0
  68. package/dist/examples/vintasend-medplum-example/src/components/Conditions/ConditionList.tsx +186 -0
  69. package/dist/examples/vintasend-medplum-example/src/components/Conditions/ConditionModal.test.tsx +80 -0
  70. package/dist/examples/vintasend-medplum-example/src/components/Conditions/ConditionModal.tsx +82 -0
  71. package/dist/examples/vintasend-medplum-example/src/components/DoseSpotIcon.test.tsx +100 -0
  72. package/dist/examples/vintasend-medplum-example/src/components/DoseSpotIcon.tsx +20 -0
  73. package/dist/examples/vintasend-medplum-example/src/components/IntegrationCard.module.css +3 -0
  74. package/dist/examples/vintasend-medplum-example/src/components/IntegrationCard.tsx +62 -0
  75. package/dist/examples/vintasend-medplum-example/src/components/MessageWithLinks.tsx +47 -0
  76. package/dist/examples/vintasend-medplum-example/src/components/PerformingLabInput.test.tsx +299 -0
  77. package/dist/examples/vintasend-medplum-example/src/components/PerformingLabInput.tsx +52 -0
  78. package/dist/examples/vintasend-medplum-example/src/components/ResourceFormWithRequiredProfile.tsx +82 -0
  79. package/dist/examples/vintasend-medplum-example/src/components/encounter/BillingTab.test.tsx +1016 -0
  80. package/dist/examples/vintasend-medplum-example/src/components/encounter/BillingTab.tsx +298 -0
  81. package/dist/examples/vintasend-medplum-example/src/components/encounter/EncounterChart.test.tsx +732 -0
  82. package/dist/examples/vintasend-medplum-example/src/components/encounter/EncounterChart.tsx +282 -0
  83. package/dist/examples/vintasend-medplum-example/src/components/encounter/EncounterHeader.test.tsx +268 -0
  84. package/dist/examples/vintasend-medplum-example/src/components/encounter/EncounterHeader.tsx +224 -0
  85. package/dist/examples/vintasend-medplum-example/src/components/encounter/SignAddendum.test.tsx +255 -0
  86. package/dist/examples/vintasend-medplum-example/src/components/encounter/SignAddendum.tsx +212 -0
  87. package/dist/examples/vintasend-medplum-example/src/components/encounter/SignLockDialog.test.tsx +120 -0
  88. package/dist/examples/vintasend-medplum-example/src/components/encounter/SignLockDialog.tsx +57 -0
  89. package/dist/examples/vintasend-medplum-example/src/components/encounter/VisitDetailsPanel.test.tsx +224 -0
  90. package/dist/examples/vintasend-medplum-example/src/components/encounter/VisitDetailsPanel.tsx +100 -0
  91. package/dist/examples/vintasend-medplum-example/src/components/labs/CoverageInput.test.tsx +431 -0
  92. package/dist/examples/vintasend-medplum-example/src/components/labs/CoverageInput.tsx +130 -0
  93. package/dist/examples/vintasend-medplum-example/src/components/labs/LabListItem.module.css +31 -0
  94. package/dist/examples/vintasend-medplum-example/src/components/labs/LabListItem.test.tsx +234 -0
  95. package/dist/examples/vintasend-medplum-example/src/components/labs/LabListItem.tsx +143 -0
  96. package/dist/examples/vintasend-medplum-example/src/components/labs/LabOrderDetails.module.css +11 -0
  97. package/dist/examples/vintasend-medplum-example/src/components/labs/LabOrderDetails.test.tsx +875 -0
  98. package/dist/examples/vintasend-medplum-example/src/components/labs/LabOrderDetails.tsx +943 -0
  99. package/dist/examples/vintasend-medplum-example/src/components/labs/LabResultDetails.test.tsx +413 -0
  100. package/dist/examples/vintasend-medplum-example/src/components/labs/LabResultDetails.tsx +203 -0
  101. package/dist/examples/vintasend-medplum-example/src/components/labs/LabSelectEmpty.tsx +22 -0
  102. package/dist/examples/vintasend-medplum-example/src/components/labs/README.md +104 -0
  103. package/dist/examples/vintasend-medplum-example/src/components/labs/TestMetadataCardInput.test.tsx +318 -0
  104. package/dist/examples/vintasend-medplum-example/src/components/labs/TestMetadataCardInput.tsx +87 -0
  105. package/dist/examples/vintasend-medplum-example/src/components/messages/ChatList.test.tsx +126 -0
  106. package/dist/examples/vintasend-medplum-example/src/components/messages/ChatList.tsx +38 -0
  107. package/dist/examples/vintasend-medplum-example/src/components/messages/ChatListItem.module.css +23 -0
  108. package/dist/examples/vintasend-medplum-example/src/components/messages/ChatListItem.test.tsx +167 -0
  109. package/dist/examples/vintasend-medplum-example/src/components/messages/ChatListItem.tsx +53 -0
  110. package/dist/examples/vintasend-medplum-example/src/components/messages/NewTopicDialog.test.tsx +94 -0
  111. package/dist/examples/vintasend-medplum-example/src/components/messages/NewTopicDialog.tsx +165 -0
  112. package/dist/examples/vintasend-medplum-example/src/components/messages/ParticipantFilter.module.css +8 -0
  113. package/dist/examples/vintasend-medplum-example/src/components/messages/ParticipantFilter.test.tsx +523 -0
  114. package/dist/examples/vintasend-medplum-example/src/components/messages/ParticipantFilter.tsx +230 -0
  115. package/dist/examples/vintasend-medplum-example/src/components/messages/ThreadInbox.module.css +23 -0
  116. package/dist/examples/vintasend-medplum-example/src/components/messages/ThreadInbox.test.tsx +567 -0
  117. package/dist/examples/vintasend-medplum-example/src/components/messages/ThreadInbox.tsx +358 -0
  118. package/dist/examples/vintasend-medplum-example/src/components/plandefinition/AddPlanDefinition.module.css +40 -0
  119. package/dist/examples/vintasend-medplum-example/src/components/plandefinition/AddPlanDefinition.tsx +257 -0
  120. package/dist/examples/vintasend-medplum-example/src/components/schedule/CreateVisit.module.css +7 -0
  121. package/dist/examples/vintasend-medplum-example/src/components/schedule/CreateVisit.test.tsx +279 -0
  122. package/dist/examples/vintasend-medplum-example/src/components/schedule/CreateVisit.tsx +156 -0
  123. package/dist/examples/vintasend-medplum-example/src/components/spaces/HistoryList.module.css +45 -0
  124. package/dist/examples/vintasend-medplum-example/src/components/spaces/HistoryList.test.tsx +90 -0
  125. package/dist/examples/vintasend-medplum-example/src/components/spaces/HistoryList.tsx +84 -0
  126. package/dist/examples/vintasend-medplum-example/src/components/spaces/ResourceBox.module.css +26 -0
  127. package/dist/examples/vintasend-medplum-example/src/components/spaces/ResourceBox.tsx +90 -0
  128. package/dist/examples/vintasend-medplum-example/src/components/spaces/ResourcePanel.test.tsx +305 -0
  129. package/dist/examples/vintasend-medplum-example/src/components/spaces/ResourcePanel.tsx +46 -0
  130. package/dist/examples/vintasend-medplum-example/src/components/spaces/SpacesInbox.module.css +262 -0
  131. package/dist/examples/vintasend-medplum-example/src/components/spaces/SpacesInbox.test.tsx +622 -0
  132. package/dist/examples/vintasend-medplum-example/src/components/spaces/SpacesInbox.tsx +286 -0
  133. package/dist/examples/vintasend-medplum-example/src/components/tasks/NewTaskModal.tsx +275 -0
  134. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskAttachmentList.tsx +132 -0
  135. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskBoard.module.css +45 -0
  136. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskBoard.test.tsx +749 -0
  137. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskBoard.tsx +416 -0
  138. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskDetailPanel.test.tsx +278 -0
  139. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskDetailPanel.tsx +133 -0
  140. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskDetailsModal.module.css +16 -0
  141. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskDetailsModal.test.tsx +255 -0
  142. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskDetailsModal.tsx +203 -0
  143. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskFileUpload.tsx +129 -0
  144. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskFilterMenu.test.tsx +156 -0
  145. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskFilterMenu.tsx +142 -0
  146. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskFilterMenu.utils.ts +28 -0
  147. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskInputNote.test.tsx +134 -0
  148. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskInputNote.tsx +250 -0
  149. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskListItem.module.css +23 -0
  150. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskListItem.test.tsx +149 -0
  151. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskListItem.tsx +53 -0
  152. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskNoteItem.test.tsx +68 -0
  153. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskNoteItem.tsx +46 -0
  154. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskProperties.test.tsx +555 -0
  155. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskProperties.tsx +170 -0
  156. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskSelectEmpty.test.tsx +32 -0
  157. package/dist/examples/vintasend-medplum-example/src/components/tasks/TaskSelectEmpty.tsx +34 -0
  158. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/SimpleTask.test.tsx +47 -0
  159. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/SimpleTask.tsx +29 -0
  160. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskPanel.test.tsx +285 -0
  161. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskPanel.tsx +129 -0
  162. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskQuestionnaireForm.test.tsx +455 -0
  163. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskQuestionnaireForm.tsx +167 -0
  164. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskServiceRequest.test.tsx +435 -0
  165. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskServiceRequest.tsx +116 -0
  166. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskStatusPanel.module.css +38 -0
  167. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskStatusPanel.test.tsx +200 -0
  168. package/dist/examples/vintasend-medplum-example/src/components/tasks/encounter/TaskStatusPanel.tsx +84 -0
  169. package/dist/examples/vintasend-medplum-example/src/components/utils.test.ts +176 -0
  170. package/dist/examples/vintasend-medplum-example/src/components/utils.ts +17 -0
  171. package/dist/examples/vintasend-medplum-example/src/config/constants.ts +3 -0
  172. package/dist/examples/vintasend-medplum-example/src/hooks/useDebouncedUpdateResource.test.tsx +166 -0
  173. package/dist/examples/vintasend-medplum-example/src/hooks/useDebouncedUpdateResource.ts +28 -0
  174. package/dist/examples/vintasend-medplum-example/src/hooks/useEncounter.test.tsx +94 -0
  175. package/dist/examples/vintasend-medplum-example/src/hooks/useEncounter.ts +11 -0
  176. package/dist/examples/vintasend-medplum-example/src/hooks/useEncounterChart.test.tsx +477 -0
  177. package/dist/examples/vintasend-medplum-example/src/hooks/useEncounterChart.ts +191 -0
  178. package/dist/examples/vintasend-medplum-example/src/hooks/usePatient.test.tsx +100 -0
  179. package/dist/examples/vintasend-medplum-example/src/hooks/usePatient.ts +18 -0
  180. package/dist/examples/vintasend-medplum-example/src/hooks/useThreadInbox.test.tsx +379 -0
  181. package/dist/examples/vintasend-medplum-example/src/hooks/useThreadInbox.ts +194 -0
  182. package/dist/examples/vintasend-medplum-example/src/index.css +8 -0
  183. package/dist/examples/vintasend-medplum-example/src/main.tsx +57 -0
  184. package/dist/examples/vintasend-medplum-example/src/pages/SearchPage.module.css +6 -0
  185. package/dist/examples/vintasend-medplum-example/src/pages/SearchPage.test.tsx +295 -0
  186. package/dist/examples/vintasend-medplum-example/src/pages/SearchPage.tsx +124 -0
  187. package/dist/examples/vintasend-medplum-example/src/pages/SignInPage.test.tsx +77 -0
  188. package/dist/examples/vintasend-medplum-example/src/pages/SignInPage.tsx +22 -0
  189. package/dist/examples/vintasend-medplum-example/src/pages/encounter/EncounterChartPage.test.tsx +87 -0
  190. package/dist/examples/vintasend-medplum-example/src/pages/encounter/EncounterChartPage.tsx +27 -0
  191. package/dist/examples/vintasend-medplum-example/src/pages/encounter/EncounterModal.module.css +16 -0
  192. package/dist/examples/vintasend-medplum-example/src/pages/encounter/EncounterModal.test.tsx +287 -0
  193. package/dist/examples/vintasend-medplum-example/src/pages/encounter/EncounterModal.tsx +151 -0
  194. package/dist/examples/vintasend-medplum-example/src/pages/integrations/DoseSpotFavoritesPage.test.tsx +519 -0
  195. package/dist/examples/vintasend-medplum-example/src/pages/integrations/DoseSpotFavoritesPage.tsx +179 -0
  196. package/dist/examples/vintasend-medplum-example/src/pages/integrations/FavoriteMedicationsTable.tsx +76 -0
  197. package/dist/examples/vintasend-medplum-example/src/pages/integrations/IntegrationsPage.test.tsx +234 -0
  198. package/dist/examples/vintasend-medplum-example/src/pages/integrations/IntegrationsPage.tsx +222 -0
  199. package/dist/examples/vintasend-medplum-example/src/pages/labs/OrderLabsPage.test.tsx +356 -0
  200. package/dist/examples/vintasend-medplum-example/src/pages/labs/OrderLabsPage.tsx +275 -0
  201. package/dist/examples/vintasend-medplum-example/src/pages/messages/MessagesPage.module.css +8 -0
  202. package/dist/examples/vintasend-medplum-example/src/pages/messages/MessagesPage.test.tsx +103 -0
  203. package/dist/examples/vintasend-medplum-example/src/pages/messages/MessagesPage.tsx +78 -0
  204. package/dist/examples/vintasend-medplum-example/src/pages/patient/CommunicationTab.test.tsx +84 -0
  205. package/dist/examples/vintasend-medplum-example/src/pages/patient/CommunicationTab.tsx +82 -0
  206. package/dist/examples/vintasend-medplum-example/src/pages/patient/DoseSpotAdvancedOptions.test.tsx +364 -0
  207. package/dist/examples/vintasend-medplum-example/src/pages/patient/DoseSpotAdvancedOptions.tsx +149 -0
  208. package/dist/examples/vintasend-medplum-example/src/pages/patient/DoseSpotTab.test.tsx +159 -0
  209. package/dist/examples/vintasend-medplum-example/src/pages/patient/DoseSpotTab.tsx +37 -0
  210. package/dist/examples/vintasend-medplum-example/src/pages/patient/EditTab.test.tsx +140 -0
  211. package/dist/examples/vintasend-medplum-example/src/pages/patient/EditTab.tsx +72 -0
  212. package/dist/examples/vintasend-medplum-example/src/pages/patient/ExportTab.test.tsx +57 -0
  213. package/dist/examples/vintasend-medplum-example/src/pages/patient/ExportTab.tsx +14 -0
  214. package/dist/examples/vintasend-medplum-example/src/pages/patient/IntakeFormPage.test.tsx +241 -0
  215. package/dist/examples/vintasend-medplum-example/src/pages/patient/IntakeFormPage.tsx +710 -0
  216. package/dist/examples/vintasend-medplum-example/src/pages/patient/LabsPage.module.css +37 -0
  217. package/dist/examples/vintasend-medplum-example/src/pages/patient/LabsPage.test.tsx +428 -0
  218. package/dist/examples/vintasend-medplum-example/src/pages/patient/LabsPage.tsx +334 -0
  219. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientPage.module.css +24 -0
  220. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientPage.test.tsx +154 -0
  221. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientPage.tsx +115 -0
  222. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientPage.utils.test.ts +223 -0
  223. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientPage.utils.ts +89 -0
  224. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientSearchPage.test.tsx +147 -0
  225. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientSearchPage.tsx +79 -0
  226. package/dist/examples/vintasend-medplum-example/src/pages/patient/PatientTabsNavigation.tsx +35 -0
  227. package/dist/examples/vintasend-medplum-example/src/pages/patient/TasksTab.test.tsx +185 -0
  228. package/dist/examples/vintasend-medplum-example/src/pages/patient/TasksTab.tsx +115 -0
  229. package/dist/examples/vintasend-medplum-example/src/pages/patient/TimelineTab.tsx +14 -0
  230. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceCreatePage.test.tsx +170 -0
  231. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceCreatePage.tsx +117 -0
  232. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceDetailPage.tsx +28 -0
  233. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceEditPage.test.tsx +131 -0
  234. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceEditPage.tsx +65 -0
  235. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceHistoryPage.test.tsx +108 -0
  236. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourceHistoryPage.tsx +16 -0
  237. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourcePage.module.css +7 -0
  238. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourcePage.test.tsx +37 -0
  239. package/dist/examples/vintasend-medplum-example/src/pages/resource/ResourcePage.tsx +44 -0
  240. package/dist/examples/vintasend-medplum-example/src/pages/resource/useResourceType.ts +44 -0
  241. package/dist/examples/vintasend-medplum-example/src/pages/resource/utils.ts +9 -0
  242. package/dist/examples/vintasend-medplum-example/src/pages/schedule/SchedulePage.test.tsx +302 -0
  243. package/dist/examples/vintasend-medplum-example/src/pages/schedule/SchedulePage.tsx +416 -0
  244. package/dist/examples/vintasend-medplum-example/src/pages/spaces/ChatInput.tsx +91 -0
  245. package/dist/examples/vintasend-medplum-example/src/pages/spaces/SpacesPage.module.css +6 -0
  246. package/dist/examples/vintasend-medplum-example/src/pages/spaces/SpacesPage.test.tsx +102 -0
  247. package/dist/examples/vintasend-medplum-example/src/pages/spaces/SpacesPage.tsx +44 -0
  248. package/dist/examples/vintasend-medplum-example/src/pages/tasks/TasksPage.module.css +7 -0
  249. package/dist/examples/vintasend-medplum-example/src/pages/tasks/TasksPage.test.tsx +133 -0
  250. package/dist/examples/vintasend-medplum-example/src/pages/tasks/TasksPage.tsx +91 -0
  251. package/dist/examples/vintasend-medplum-example/src/test-utils/render.tsx +20 -0
  252. package/dist/examples/vintasend-medplum-example/src/test.setup.ts +49 -0
  253. package/dist/examples/vintasend-medplum-example/src/types/encounter.ts +8 -0
  254. package/dist/examples/vintasend-medplum-example/src/types/spaces.ts +10 -0
  255. package/dist/examples/vintasend-medplum-example/src/utils/chargeitems.test.ts +141 -0
  256. package/dist/examples/vintasend-medplum-example/src/utils/chargeitems.ts +59 -0
  257. package/dist/examples/vintasend-medplum-example/src/utils/claims.test.ts +153 -0
  258. package/dist/examples/vintasend-medplum-example/src/utils/claims.ts +65 -0
  259. package/dist/examples/vintasend-medplum-example/src/utils/communication-search.ts +47 -0
  260. package/dist/examples/vintasend-medplum-example/src/utils/coverage.test.ts +48 -0
  261. package/dist/examples/vintasend-medplum-example/src/utils/coverage.ts +33 -0
  262. package/dist/examples/vintasend-medplum-example/src/utils/documentReference.test.ts +102 -0
  263. package/dist/examples/vintasend-medplum-example/src/utils/documentReference.ts +55 -0
  264. package/dist/examples/vintasend-medplum-example/src/utils/encounter.test.ts +169 -0
  265. package/dist/examples/vintasend-medplum-example/src/utils/encounter.ts +261 -0
  266. package/dist/examples/vintasend-medplum-example/src/utils/intake-form.test.ts +154 -0
  267. package/dist/examples/vintasend-medplum-example/src/utils/intake-form.ts +272 -0
  268. package/dist/examples/vintasend-medplum-example/src/utils/intake-utils.test.ts +1137 -0
  269. package/dist/examples/vintasend-medplum-example/src/utils/intake-utils.ts +827 -0
  270. package/dist/examples/vintasend-medplum-example/src/utils/notifications.test.ts +27 -0
  271. package/dist/examples/vintasend-medplum-example/src/utils/notifications.ts +15 -0
  272. package/dist/examples/vintasend-medplum-example/src/utils/spaceMessaging.ts +249 -0
  273. package/dist/examples/vintasend-medplum-example/src/utils/spacePersistence.test.ts +450 -0
  274. package/dist/examples/vintasend-medplum-example/src/utils/spacePersistence.ts +147 -0
  275. package/dist/examples/vintasend-medplum-example/src/utils/task-search.ts +63 -0
  276. package/dist/examples/vintasend-medplum-example/src/vite-env.d.ts +3 -0
  277. package/dist/examples/vintasend-medplum-example/tsconfig.bots.json +4 -0
  278. package/dist/examples/vintasend-medplum-example/tsconfig.json +19 -0
  279. package/dist/examples/vintasend-medplum-example/vercel.json +3 -0
  280. package/dist/examples/vintasend-medplum-example/vite.config.ts +44 -0
  281. package/dist/services/notification-backends/base-notification-backend.d.ts +5 -0
  282. package/dist/services/notification-service.js +11 -1
  283. package/dist/services/notification-template-renderers/base-email-template-renderer.d.ts +5 -0
  284. package/package.json +1 -1
@@ -0,0 +1,477 @@
1
+ // SPDX-FileCopyrightText: Copyright Orangebot, Inc. and Medplum contributors
2
+ // SPDX-License-Identifier: Apache-2.0
3
+ import { act, renderHook, waitFor } from '@testing-library/react';
4
+ import { MedplumProvider } from '@medplum/react';
5
+ import type { JSX } from 'react';
6
+ import type {
7
+ Appointment,
8
+ ChargeItem,
9
+ Claim,
10
+ ClinicalImpression,
11
+ Encounter,
12
+ Patient,
13
+ Practitioner,
14
+ Reference,
15
+ Task,
16
+ } from '@medplum/fhirtypes';
17
+ import { MockClient } from '@medplum/mock';
18
+ import { describe, expect, test, vi, beforeEach } from 'vitest';
19
+ import { useEncounterChart } from './useEncounterChart';
20
+ import { getChargeItemsForEncounter } from '../utils/chargeitems';
21
+ import { createClaimFromEncounter } from '../utils/claims';
22
+ import { showErrorNotification } from '../utils/notifications';
23
+
24
+ vi.mock('../utils/chargeitems');
25
+ vi.mock('../utils/claims');
26
+ vi.mock('../utils/notifications');
27
+
28
+ const mockEncounter: Encounter = {
29
+ resourceType: 'Encounter',
30
+ id: 'encounter-123',
31
+ status: 'in-progress',
32
+ class: { code: 'AMB', system: 'http://terminology.hl7.org/CodeSystem/v3-ActCode' },
33
+ subject: { reference: 'Patient/patient-123' },
34
+ participant: [
35
+ {
36
+ individual: { reference: 'Practitioner/practitioner-123' },
37
+ },
38
+ ],
39
+ appointment: [{ reference: 'Appointment/appointment-123' }],
40
+ };
41
+
42
+ const mockPatient: Patient = {
43
+ resourceType: 'Patient',
44
+ id: 'patient-123',
45
+ name: [{ given: ['Test'], family: 'Patient' }],
46
+ };
47
+
48
+ const mockPractitioner: Practitioner = {
49
+ resourceType: 'Practitioner',
50
+ id: 'practitioner-123',
51
+ name: [{ given: ['Dr.'], family: 'Test' }],
52
+ };
53
+
54
+ const mockAppointment: Appointment = {
55
+ resourceType: 'Appointment',
56
+ id: 'appointment-123',
57
+ status: 'booked',
58
+ participant: [],
59
+ };
60
+
61
+ const mockTask: Task = {
62
+ resourceType: 'Task',
63
+ id: 'task-123',
64
+ status: 'in-progress',
65
+ intent: 'order',
66
+ encounter: { reference: 'Encounter/encounter-123' },
67
+ authoredOn: '2024-01-01T10:00:00Z',
68
+ };
69
+
70
+ const mockClinicalImpression: ClinicalImpression = {
71
+ resourceType: 'ClinicalImpression',
72
+ id: 'clinical-impression-123',
73
+ status: 'completed',
74
+ subject: { reference: 'Patient/patient-123' },
75
+ encounter: { reference: 'Encounter/encounter-123' },
76
+ };
77
+
78
+ const mockChargeItem: ChargeItem = {
79
+ resourceType: 'ChargeItem',
80
+ id: 'charge-item-123',
81
+ status: 'billable',
82
+ subject: { reference: 'Patient/patient-123' },
83
+ code: { text: 'Test Charge' },
84
+ };
85
+
86
+ const mockClaim: Claim = {
87
+ resourceType: 'Claim',
88
+ id: 'claim-123',
89
+ status: 'active',
90
+ type: { coding: [{ code: 'professional' }] },
91
+ use: 'claim',
92
+ created: new Date().toISOString(),
93
+ patient: { reference: 'Patient/patient-123' },
94
+ provider: { reference: 'Practitioner/practitioner-123' },
95
+ priority: { coding: [{ code: 'normal' }] },
96
+ insurance: [],
97
+ item: [
98
+ {
99
+ sequence: 1,
100
+ encounter: [{ reference: 'Encounter/encounter-123' }],
101
+ productOrService: {
102
+ coding: [{ code: 'CPT', system: 'http://terminology.hl7.org/CodeSystem/cpt' }],
103
+ text: 'CPT Code',
104
+ },
105
+ },
106
+ ],
107
+ };
108
+
109
+ describe('useEncounterChart', () => {
110
+ let medplum: MockClient;
111
+
112
+ beforeEach(async () => {
113
+ medplum = new MockClient();
114
+ vi.clearAllMocks();
115
+ });
116
+
117
+ const wrapper = ({ children }: { children: React.ReactNode }): JSX.Element => (
118
+ <MedplumProvider medplum={medplum}>{children}</MedplumProvider>
119
+ );
120
+
121
+ test('returns initial state when no encounter provided', () => {
122
+ const { result } = renderHook(() => useEncounterChart(undefined), { wrapper });
123
+
124
+ expect(result.current.encounter).toBeUndefined();
125
+ expect(result.current.claim).toBeUndefined();
126
+ expect(result.current.practitioner).toBeUndefined();
127
+ expect(result.current.tasks).toEqual([]);
128
+ expect(result.current.clinicalImpression).toBeUndefined();
129
+ expect(result.current.chargeItems).toEqual([]);
130
+ expect(result.current.appointment).toBeUndefined();
131
+ });
132
+
133
+ test('resolves encounter reference and loads data', async () => {
134
+ // Create resources in MockClient
135
+ const createdEncounter = await medplum.createResource(mockEncounter);
136
+ await medplum.createResource(mockPractitioner);
137
+
138
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([mockChargeItem]);
139
+
140
+ const encounterRef: Reference<Encounter> = { reference: `Encounter/${createdEncounter.id}` };
141
+
142
+ const { result } = renderHook(() => useEncounterChart(encounterRef), { wrapper });
143
+
144
+ await waitFor(() => {
145
+ expect(result.current.encounter?.id).toBe(createdEncounter.id);
146
+ });
147
+
148
+ await waitFor(() => {
149
+ expect(result.current.chargeItems).toHaveLength(1);
150
+ });
151
+ });
152
+
153
+ test('loads charge items for encounter', async () => {
154
+ await medplum.createResource(mockEncounter);
155
+ await medplum.createResource(mockPractitioner);
156
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([mockChargeItem]);
157
+
158
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
159
+
160
+ await waitFor(() => {
161
+ expect(result.current.chargeItems).toHaveLength(1);
162
+ expect(result.current.chargeItems[0].id).toBe('charge-item-123');
163
+ });
164
+ });
165
+
166
+ test('fetches existing claim for encounter', async () => {
167
+ const createdEncounter = await medplum.createResource(mockEncounter);
168
+ await medplum.createResource(mockPractitioner);
169
+ const claimWithEncounter: Claim = {
170
+ ...mockClaim,
171
+ item: [
172
+ {
173
+ sequence: 1,
174
+ encounter: [{ reference: `Encounter/${createdEncounter.id}` }],
175
+ productOrService: {
176
+ coding: [{ code: 'CPT', system: 'http://terminology.hl7.org/CodeSystem/cpt' }],
177
+ text: 'CPT Code',
178
+ },
179
+ },
180
+ ],
181
+ };
182
+ await medplum.createResource(claimWithEncounter);
183
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
184
+
185
+ const { result } = renderHook(() => useEncounterChart(createdEncounter), { wrapper });
186
+
187
+ await waitFor(() => {
188
+ expect(result.current.claim).toBeDefined();
189
+ });
190
+ });
191
+
192
+ test('fetches tasks for encounter', async () => {
193
+ await medplum.createResource(mockEncounter);
194
+ await medplum.createResource(mockPractitioner);
195
+ await medplum.createResource(mockTask);
196
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
197
+
198
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
199
+
200
+ await waitFor(() => {
201
+ expect(result.current.tasks).toHaveLength(1);
202
+ expect(result.current.tasks[0].id).toBe('task-123');
203
+ });
204
+ });
205
+
206
+ test('sorts tasks by authoredOn date', async () => {
207
+ const olderTask: Task = {
208
+ ...mockTask,
209
+ id: 'task-old',
210
+ authoredOn: '2024-01-01T08:00:00Z',
211
+ };
212
+ const newerTask: Task = {
213
+ ...mockTask,
214
+ id: 'task-new',
215
+ authoredOn: '2024-01-01T12:00:00Z',
216
+ };
217
+
218
+ await medplum.createResource(mockEncounter);
219
+ await medplum.createResource(mockPractitioner);
220
+ await medplum.createResource(newerTask);
221
+ await medplum.createResource(olderTask);
222
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
223
+
224
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
225
+
226
+ await waitFor(() => {
227
+ expect(result.current.tasks).toHaveLength(2);
228
+ expect(result.current.tasks[0].id).toBe('task-old');
229
+ expect(result.current.tasks[1].id).toBe('task-new');
230
+ });
231
+ });
232
+
233
+ test('fetches clinical impression for encounter', async () => {
234
+ await medplum.createResource(mockEncounter);
235
+ await medplum.createResource(mockPractitioner);
236
+ await medplum.createResource(mockClinicalImpression);
237
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
238
+
239
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
240
+
241
+ await waitFor(() => {
242
+ expect(result.current.clinicalImpression?.id).toBe('clinical-impression-123');
243
+ });
244
+ });
245
+
246
+ test('fetches practitioner from encounter participant', async () => {
247
+ await medplum.createResource(mockEncounter);
248
+ await medplum.createResource(mockPractitioner);
249
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
250
+
251
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
252
+
253
+ await waitFor(() => {
254
+ expect(result.current.practitioner?.id).toBe('practitioner-123');
255
+ });
256
+ });
257
+
258
+ test('fetches appointment from encounter', async () => {
259
+ await medplum.createResource(mockEncounter);
260
+ await medplum.createResource(mockPractitioner);
261
+ await medplum.createResource(mockAppointment);
262
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
263
+
264
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
265
+
266
+ await waitFor(() => {
267
+ expect(result.current.appointment?.id).toBe('appointment-123');
268
+ });
269
+ });
270
+
271
+ test('creates claim when all conditions are met', async () => {
272
+ const newClaim: Claim = {
273
+ resourceType: 'Claim',
274
+ id: 'new-claim-123',
275
+ status: 'active',
276
+ type: { coding: [{ code: 'professional' }] },
277
+ use: 'claim',
278
+ created: new Date().toISOString(),
279
+ patient: { reference: 'Patient/patient-123' },
280
+ provider: { reference: 'Practitioner/practitioner-123' },
281
+ priority: { coding: [{ code: 'normal' }] },
282
+ insurance: [],
283
+ };
284
+
285
+ await medplum.createResource(mockEncounter);
286
+ await medplum.createResource(mockPractitioner);
287
+ await medplum.createResource(mockPatient);
288
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([mockChargeItem]);
289
+ vi.mocked(createClaimFromEncounter).mockResolvedValue(newClaim);
290
+
291
+ const { result } = renderHook(() => useEncounterChart(mockEncounter, mockPatient), { wrapper });
292
+
293
+ // Wait for all prerequisites to be met
294
+ await waitFor(() => {
295
+ expect(result.current.chargeItems).toHaveLength(1);
296
+ expect(result.current.practitioner).toBeDefined();
297
+ });
298
+
299
+ // Then wait for claim to be created
300
+ await waitFor(() => {
301
+ expect(result.current.claim?.id).toBe('new-claim-123');
302
+ });
303
+
304
+ expect(createClaimFromEncounter).toHaveBeenCalledWith(medplum, 'patient-123', 'encounter-123', 'practitioner-123', [
305
+ mockChargeItem,
306
+ ]);
307
+ });
308
+
309
+ test('does not create claim if one already exists', async () => {
310
+ await medplum.createResource(mockEncounter);
311
+ await medplum.createResource(mockPractitioner);
312
+ await medplum.createResource(mockPatient);
313
+ const createdClaim = await medplum.createResource(mockClaim);
314
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([mockChargeItem]);
315
+
316
+ const { result } = renderHook(() => useEncounterChart(mockEncounter, mockPatient), { wrapper });
317
+
318
+ await waitFor(() => {
319
+ expect(result.current.claim).toBeDefined();
320
+ });
321
+
322
+ // Verify the existing claim is found (may have different ID due to MockClient)
323
+ expect(result.current.claim?.id).toBe(createdClaim.id);
324
+
325
+ // Verify no new claim is created
326
+ expect(createClaimFromEncounter).not.toHaveBeenCalled();
327
+ });
328
+
329
+ test('does not create claim if patient is missing', async () => {
330
+ await medplum.createResource(mockEncounter);
331
+ await medplum.createResource(mockPractitioner);
332
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([mockChargeItem]);
333
+
334
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
335
+
336
+ // Wait for data to load
337
+ await waitFor(() => {
338
+ expect(result.current.chargeItems).toHaveLength(1);
339
+ expect(result.current.practitioner).toBeDefined();
340
+ });
341
+
342
+ // Verify claim was not created and remains undefined
343
+ expect(result.current.claim).toBeUndefined();
344
+ expect(createClaimFromEncounter).not.toHaveBeenCalled();
345
+ });
346
+
347
+ test('does not create claim if practitioner is missing', async () => {
348
+ const encounterWithoutPractitioner: Encounter = {
349
+ ...mockEncounter,
350
+ participant: [],
351
+ };
352
+
353
+ await medplum.createResource(encounterWithoutPractitioner);
354
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([mockChargeItem]);
355
+
356
+ const { result } = renderHook(() => useEncounterChart(encounterWithoutPractitioner, mockPatient), { wrapper });
357
+
358
+ await waitFor(() => {
359
+ expect(result.current.chargeItems).toHaveLength(1);
360
+ });
361
+
362
+ // Verify no practitioner and no claim
363
+ expect(result.current.practitioner).toBeUndefined();
364
+ expect(result.current.claim).toBeUndefined();
365
+ expect(createClaimFromEncounter).not.toHaveBeenCalled();
366
+ });
367
+
368
+ test('does not create claim if charge items are empty', async () => {
369
+ await medplum.createResource(mockEncounter);
370
+ await medplum.createResource(mockPractitioner);
371
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
372
+
373
+ const { result } = renderHook(() => useEncounterChart(mockEncounter, mockPatient), { wrapper });
374
+
375
+ await waitFor(() => {
376
+ expect(result.current.practitioner).toBeDefined();
377
+ });
378
+
379
+ // Verify no charge items and no claim
380
+ expect(result.current.chargeItems).toEqual([]);
381
+ expect(result.current.claim).toBeUndefined();
382
+ expect(createClaimFromEncounter).not.toHaveBeenCalled();
383
+ });
384
+
385
+ test('handles charge item fetch errors gracefully', async () => {
386
+ const error = new Error('Failed to fetch charge items');
387
+ await medplum.createResource(mockEncounter);
388
+ await medplum.createResource(mockPractitioner);
389
+ vi.mocked(getChargeItemsForEncounter).mockRejectedValue(error);
390
+
391
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
392
+
393
+ await waitFor(() => {
394
+ expect(showErrorNotification).toHaveBeenCalled();
395
+ });
396
+
397
+ // Hook should still function with empty charge items
398
+ expect(result.current.encounter).toBeDefined();
399
+ expect(result.current.chargeItems).toEqual([]);
400
+ });
401
+
402
+ test('handles search errors gracefully', async () => {
403
+ const error = new Error('Search failed');
404
+ await medplum.createResource(mockEncounter);
405
+
406
+ // Mock searchResources to fail
407
+ medplum.searchResources = vi.fn().mockRejectedValue(error);
408
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
409
+
410
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
411
+
412
+ await waitFor(() => {
413
+ expect(showErrorNotification).toHaveBeenCalled();
414
+ });
415
+
416
+ // Hook should still have the encounter
417
+ expect(result.current.encounter?.id).toBe('encounter-123');
418
+ expect(result.current.tasks).toEqual([]);
419
+ expect(result.current.clinicalImpression).toBeUndefined();
420
+ });
421
+
422
+ test('resolves patient reference when provided', async () => {
423
+ await medplum.createResource(mockPatient);
424
+ await medplum.createResource(mockEncounter);
425
+ await medplum.createResource(mockPractitioner);
426
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
427
+
428
+ const patientRef: Reference<Patient> = { reference: 'Patient/patient-123' };
429
+
430
+ const { result } = renderHook(() => useEncounterChart(mockEncounter, patientRef), { wrapper });
431
+
432
+ await waitFor(() => {
433
+ expect(result.current.encounter?.id).toBe('encounter-123');
434
+ });
435
+ });
436
+
437
+ test('allows manual state updates via setters', async () => {
438
+ await medplum.createResource(mockEncounter);
439
+ await medplum.createResource(mockPractitioner);
440
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
441
+
442
+ const { result } = renderHook(() => useEncounterChart(undefined), { wrapper });
443
+
444
+ expect(result.current.encounter).toBeUndefined();
445
+
446
+ const updatedEncounter: Encounter = {
447
+ ...mockEncounter,
448
+ status: 'finished',
449
+ };
450
+
451
+ act(() => {
452
+ result.current.setEncounter(updatedEncounter);
453
+ });
454
+
455
+ expect(result.current.encounter?.status).toBe('finished');
456
+ });
457
+
458
+ test('updates claim via setter', async () => {
459
+ await medplum.createResource(mockEncounter);
460
+ await medplum.createResource(mockPractitioner);
461
+ vi.mocked(getChargeItemsForEncounter).mockResolvedValue([]);
462
+
463
+ const { result } = renderHook(() => useEncounterChart(mockEncounter), { wrapper });
464
+
465
+ await waitFor(() => {
466
+ expect(result.current.encounter).toBeDefined();
467
+ });
468
+
469
+ expect(result.current.claim).toBeUndefined();
470
+
471
+ act(() => {
472
+ result.current.setClaim(mockClaim);
473
+ });
474
+
475
+ expect(result.current.claim?.id).toBe('claim-123');
476
+ });
477
+ });
@@ -0,0 +1,191 @@
1
+ // SPDX-FileCopyrightText: Copyright Orangebot, Inc. and Medplum contributors
2
+ // SPDX-License-Identifier: Apache-2.0
3
+ import { getReferenceString } from '@medplum/core';
4
+ import type {
5
+ Appointment,
6
+ ChargeItem,
7
+ Claim,
8
+ ClinicalImpression,
9
+ Encounter,
10
+ Patient,
11
+ Practitioner,
12
+ Reference,
13
+ Task,
14
+ } from '@medplum/fhirtypes';
15
+ import { useMedplum, useResource } from '@medplum/react';
16
+ import { useCallback, useEffect, useState } from 'react';
17
+ import { getChargeItemsForEncounter } from '../utils/chargeitems';
18
+ import { createClaimFromEncounter } from '../utils/claims';
19
+ import { showErrorNotification } from '../utils/notifications';
20
+
21
+ export interface EncounterChartHook {
22
+ // State values
23
+ encounter: Encounter | undefined;
24
+ claim: Claim | undefined;
25
+ practitioner: Practitioner | undefined;
26
+ tasks: Task[];
27
+ clinicalImpression: ClinicalImpression | undefined;
28
+ chargeItems: ChargeItem[];
29
+ appointment: Appointment | undefined;
30
+ // State setters
31
+ setEncounter: React.Dispatch<React.SetStateAction<Encounter | undefined>>;
32
+ setClaim: React.Dispatch<React.SetStateAction<Claim | undefined>>;
33
+ setPractitioner: React.Dispatch<React.SetStateAction<Practitioner | undefined>>;
34
+ setTasks: React.Dispatch<React.SetStateAction<Task[]>>;
35
+ setClinicalImpression: React.Dispatch<React.SetStateAction<ClinicalImpression | undefined>>;
36
+ setChargeItems: React.Dispatch<React.SetStateAction<ChargeItem[]>>;
37
+ setAppointment: React.Dispatch<React.SetStateAction<Appointment | undefined>>;
38
+ }
39
+
40
+ export function useEncounterChart(
41
+ encounter: Encounter | Reference<Encounter> | undefined,
42
+ patient?: Patient | Reference<Patient>
43
+ ): EncounterChartHook {
44
+ const medplum = useMedplum();
45
+ const encounterResource = useResource(encounter);
46
+ const patientResource = useResource(patient);
47
+ const [encounterState, setEncounter] = useState<Encounter | undefined>(encounterResource);
48
+ const [claim, setClaim] = useState<Claim | undefined>();
49
+ const [practitioner, setPractitioner] = useState<Practitioner | undefined>();
50
+ const [tasks, setTasks] = useState<Task[]>([]);
51
+ const [clinicalImpression, setClinicalImpression] = useState<ClinicalImpression | undefined>();
52
+ const [chargeItems, setChargeItems] = useState<ChargeItem[]>([]);
53
+ const [appointment, setAppointment] = useState<Appointment | undefined>();
54
+
55
+ const encounterToUse = encounterResource ?? encounterState;
56
+
57
+ useEffect(() => {
58
+ async function loadChargeItems(): Promise<void> {
59
+ if (encounterResource) {
60
+ const chargeItemsResult = await getChargeItemsForEncounter(medplum, encounterResource);
61
+ setChargeItems(chargeItemsResult);
62
+ }
63
+ }
64
+
65
+ async function fetchClaim(): Promise<void> {
66
+ if (!encounterResource?.id) {
67
+ return;
68
+ }
69
+ const response = await medplum.searchResources('Claim', `encounter=${getReferenceString(encounterResource)}`);
70
+ if (response.length !== 0) {
71
+ setClaim(response[0]);
72
+ }
73
+ }
74
+
75
+ loadChargeItems().catch(showErrorNotification);
76
+ fetchClaim().catch(showErrorNotification);
77
+ }, [encounterResource, medplum]);
78
+
79
+ // Fetch tasks related to the encounter
80
+ const fetchTasks = useCallback(async (): Promise<void> => {
81
+ if (!encounterResource) {
82
+ return;
83
+ }
84
+ const taskResult = await medplum.searchResources('Task', `encounter=${getReferenceString(encounterResource)}`, {
85
+ cache: 'no-cache',
86
+ });
87
+
88
+ taskResult.sort((a: Task, b: Task) => {
89
+ const dateA = new Date(a.authoredOn || '').getTime();
90
+ const dateB = new Date(b.authoredOn || '').getTime();
91
+ return dateA - dateB;
92
+ });
93
+
94
+ setTasks(taskResult);
95
+ }, [medplum, encounterResource]);
96
+
97
+ // Fetch clinical impressions related to the encounter
98
+ const fetchClinicalImpressions = useCallback(async (): Promise<void> => {
99
+ if (!encounterResource) {
100
+ return;
101
+ }
102
+ const clinicalImpressionResult = await medplum.searchResources(
103
+ 'ClinicalImpression',
104
+ `encounter=${getReferenceString(encounterResource)}`
105
+ );
106
+
107
+ const result = clinicalImpressionResult?.[0];
108
+ setClinicalImpression(result);
109
+ }, [medplum, encounterResource]);
110
+
111
+ // Fetch data on component mount or when encounter changes
112
+ useEffect(() => {
113
+ if (encounterResource) {
114
+ fetchTasks().catch((err) => showErrorNotification(err));
115
+ fetchClinicalImpressions().catch((err) => showErrorNotification(err));
116
+ }
117
+ }, [encounterResource, fetchTasks, fetchClinicalImpressions]);
118
+
119
+ // Fetch practitioner related to the encounter
120
+ useEffect(() => {
121
+ const fetchPractitioner = async (): Promise<void> => {
122
+ if (encounterResource?.participant?.[0]?.individual) {
123
+ const practitionerResult = await medplum.readReference(encounterResource.participant[0].individual);
124
+ setPractitioner(practitionerResult as Practitioner);
125
+ }
126
+ };
127
+
128
+ if (encounterResource) {
129
+ fetchPractitioner().catch((err) => showErrorNotification(err));
130
+ }
131
+ }, [encounterResource, medplum]);
132
+
133
+ // Fetch appointment related to the encounter
134
+ useEffect(() => {
135
+ const fetchAppointment = async (): Promise<void> => {
136
+ const appointmentRef = encounterResource?.appointment?.at(-1);
137
+ if (appointmentRef) {
138
+ const appointmentResult = await medplum.readReference(appointmentRef);
139
+ setAppointment(appointmentResult as Appointment);
140
+ }
141
+ };
142
+
143
+ if (encounterResource) {
144
+ fetchAppointment().catch((err) => showErrorNotification(err));
145
+ }
146
+ }, [encounterResource, medplum]);
147
+
148
+ useEffect(() => {
149
+ const createClaim = async (): Promise<void> => {
150
+ if (claim) {
151
+ // If a claim already exists, don't create a new one
152
+ return;
153
+ }
154
+
155
+ const patientId = patientResource?.id;
156
+ if (!patientId || !encounterResource?.id || !practitioner?.id || chargeItems.length === 0) {
157
+ return;
158
+ }
159
+ const newClaim = await createClaimFromEncounter(
160
+ medplum,
161
+ patientId,
162
+ encounterResource.id,
163
+ practitioner.id,
164
+ chargeItems
165
+ );
166
+ if (newClaim) {
167
+ setClaim(newClaim);
168
+ }
169
+ };
170
+ createClaim().catch((err) => showErrorNotification(err));
171
+ }, [patientResource, encounterResource, medplum, claim, practitioner, chargeItems]);
172
+
173
+ return {
174
+ // State values - use resolved resource when available
175
+ encounter: encounterToUse,
176
+ claim,
177
+ practitioner,
178
+ tasks,
179
+ clinicalImpression,
180
+ chargeItems,
181
+ appointment,
182
+ // State setters
183
+ setEncounter,
184
+ setClaim,
185
+ setPractitioner,
186
+ setTasks,
187
+ setClinicalImpression,
188
+ setChargeItems,
189
+ setAppointment,
190
+ };
191
+ }