@gen-epix/ui 1.14.4 → 1.15.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 (739) hide show
  1. package/README.md +3 -3
  2. package/dist/index.js +27287 -27948
  3. package/package.json +94 -84
  4. package/src/@types/environment.d.ts +54 -0
  5. package/src/api/api.ts +43576 -0
  6. package/src/api/base.ts +111 -0
  7. package/src/api/common.ts +168 -0
  8. package/src/api/configuration.ts +118 -0
  9. package/src/api/index.ts +20 -0
  10. package/src/assets/icons/CollectionIcon.svg +1 -0
  11. package/src/assets/icons/DnaIcon.svg +1 -0
  12. package/src/assets/icons/VirusIcon.svg +1 -0
  13. package/src/assets/logo/logo-rijksoverheid.svg +42 -0
  14. package/src/assets/logo/logo-rivm.svg +6 -0
  15. package/src/classes/Subject/Subject.ts +26 -0
  16. package/src/classes/Subject/index.ts +1 -0
  17. package/src/classes/TableEventBus/TableEventBus.ts +14 -0
  18. package/src/classes/TableEventBus/index.ts +1 -0
  19. package/src/classes/abstracts/EventBusAbstract/EventBusAbstract.ts +37 -0
  20. package/src/classes/abstracts/EventBusAbstract/index.ts +1 -0
  21. package/src/classes/abstracts/FilterAbstract/FilterAbstract.ts +54 -0
  22. package/src/classes/abstracts/FilterAbstract/index.ts +1 -0
  23. package/src/classes/abstracts/SubscribableAbstract/SubscribableAbstract.ts +20 -0
  24. package/src/classes/abstracts/SubscribableAbstract/index.ts +1 -0
  25. package/src/classes/errors/UploadError.ts +11 -0
  26. package/src/classes/errors/index.ts +1 -0
  27. package/src/classes/filters/BooleanFilter.ts +31 -0
  28. package/src/classes/filters/DateFilter.ts +133 -0
  29. package/src/classes/filters/GeoFilter.ts +60 -0
  30. package/src/classes/filters/MultiSelectFilter.ts +70 -0
  31. package/src/classes/filters/NumberRangeFilter.ts +82 -0
  32. package/src/classes/filters/SelectionFilter.ts +25 -0
  33. package/src/classes/filters/TextFilter.ts +36 -0
  34. package/src/classes/filters/TreeFilter.ts +22 -0
  35. package/src/classes/managers/AuthenticationManager/AuthenticationManager.ts +94 -0
  36. package/src/classes/managers/AuthenticationManager/index.ts +1 -0
  37. package/src/classes/managers/AuthorizationManager/AuthorizationManager.ts +70 -0
  38. package/src/classes/managers/AuthorizationManager/index.ts +1 -0
  39. package/src/classes/managers/BackendVersionManager/BackendVersionManager.ts +23 -0
  40. package/src/classes/managers/BackendVersionManager/index.ts +1 -0
  41. package/src/classes/managers/BreadcrumbManager/BreadcrumbManager.ts +30 -0
  42. package/src/classes/managers/BreadcrumbManager/index.ts +1 -0
  43. package/src/classes/managers/ConfigManager/ConfigManager.ts +31 -0
  44. package/src/classes/managers/ConfigManager/index.ts +1 -0
  45. package/src/classes/managers/DevicePixelRatioManager/DevicePixelRatioManager.ts +27 -0
  46. package/src/classes/managers/DevicePixelRatioManager/index.ts +1 -0
  47. package/src/classes/managers/EmotionCacheManager/EmotionCacheManager.ts +27 -0
  48. package/src/classes/managers/EmotionCacheManager/index.ts +1 -0
  49. package/src/classes/managers/EpiDataManager/EpiDataManager.ts +196 -0
  50. package/src/classes/managers/EpiDataManager/index.ts +1 -0
  51. package/src/classes/managers/EpiEventBusManager/EpiEventBusManager.ts +61 -0
  52. package/src/classes/managers/EpiEventBusManager/index.ts +1 -0
  53. package/src/classes/managers/EpiHighlightingManager/EpiHighlightingManager.ts +32 -0
  54. package/src/classes/managers/EpiHighlightingManager/index.ts +1 -0
  55. package/src/classes/managers/EpiLineListCaseSetMembersManager/EpiListsCaseSetMembersManager.ts +118 -0
  56. package/src/classes/managers/EpiLineListCaseSetMembersManager/index.ts +1 -0
  57. package/src/classes/managers/FeatureFlagsManager/FeatureFlagsManager.ts +30 -0
  58. package/src/classes/managers/FeatureFlagsManager/index.ts +1 -0
  59. package/src/classes/managers/I18nManager/I18nManager.ts +105 -0
  60. package/src/classes/managers/I18nManager/index.ts +1 -0
  61. package/src/classes/managers/InactivityManager/InactivityManager.ts +119 -0
  62. package/src/classes/managers/InactivityManager/index.ts +1 -0
  63. package/src/classes/managers/KeyboardShortcutManager/KeyboardShortcutManager.ts +78 -0
  64. package/src/classes/managers/KeyboardShortcutManager/index.ts +1 -0
  65. package/src/classes/managers/LogManager/LogManager.ts +151 -0
  66. package/src/classes/managers/LogManager/index.ts +1 -0
  67. package/src/classes/managers/NavigationHistoryManager/NavigationHistoryManager.ts +16 -0
  68. package/src/classes/managers/NavigationHistoryManager/index.ts +1 -0
  69. package/src/classes/managers/NotificationManager/NotificationManager.ts +104 -0
  70. package/src/classes/managers/NotificationManager/index.ts +1 -0
  71. package/src/classes/managers/PageEventBusManager/PageEventBusManager.ts +116 -0
  72. package/src/classes/managers/PageEventBusManager/index.ts +1 -0
  73. package/src/classes/managers/QueryClientManager/QueryClientManager.ts +51 -0
  74. package/src/classes/managers/QueryClientManager/index.ts +1 -0
  75. package/src/classes/managers/RouterManager/RouterManager.ts +25 -0
  76. package/src/classes/managers/RouterManager/index.ts +1 -0
  77. package/src/classes/managers/UserSettingsManager/UserSettingsManager.ts +16 -0
  78. package/src/classes/managers/UserSettingsManager/index.ts +1 -0
  79. package/src/classes/managers/WindowManager/WindowManager.test.ts +23 -0
  80. package/src/classes/managers/WindowManager/WindowManager.ts +26 -0
  81. package/src/classes/managers/WindowManager/index.ts +1 -0
  82. package/src/components/app/App/App.tsx +58 -0
  83. package/src/components/app/App/index.ts +1 -0
  84. package/src/components/app/ApplicationBootstrap/ApplicationBootstrap.tsx +223 -0
  85. package/src/components/app/ApplicationBootstrap/index.ts +1 -0
  86. package/src/components/app/RouterRoot/RouterRoot.tsx +187 -0
  87. package/src/components/app/RouterRoot/index.ts +1 -0
  88. package/src/components/epi/EpiAddCasesToEventDialog/EpiAddCasesToEventDialog.tsx +347 -0
  89. package/src/components/epi/EpiAddCasesToEventDialog/EpiAddCasesToEventDialogSuccessNotificationMessage.tsx +38 -0
  90. package/src/components/epi/EpiAddCasesToEventDialog/index.ts +1 -0
  91. package/src/components/epi/EpiBulkEditCaseDialog/EpiBulkEditCaseDialog.tsx +61 -0
  92. package/src/components/epi/EpiBulkEditCaseDialog/index.ts +1 -0
  93. package/src/components/epi/EpiCaseInfoDialog/EpiCaseCaseSetInfo.tsx +137 -0
  94. package/src/components/epi/EpiCaseInfoDialog/EpiCaseContent.tsx +127 -0
  95. package/src/components/epi/EpiCaseInfoDialog/EpiCaseForm.tsx +111 -0
  96. package/src/components/epi/EpiCaseInfoDialog/EpiCaseInfoDialog.tsx +377 -0
  97. package/src/components/epi/EpiCaseInfoDialog/EpiCaseSharingForm.tsx +140 -0
  98. package/src/components/epi/EpiCaseInfoDialog/EpiCaseSharingInfo.tsx +23 -0
  99. package/src/components/epi/EpiCaseInfoDialog/EpiReadOnlyCaseContent.tsx +89 -0
  100. package/src/components/epi/EpiCaseInfoDialog/index.ts +1 -0
  101. package/src/components/epi/EpiCaseSetInfoDialog/EpiCaseSetContent.tsx +108 -0
  102. package/src/components/epi/EpiCaseSetInfoDialog/EpiCaseSetDescription.tsx +34 -0
  103. package/src/components/epi/EpiCaseSetInfoDialog/EpiCaseSetForm.tsx +151 -0
  104. package/src/components/epi/EpiCaseSetInfoDialog/EpiCaseSetInfoDialog.tsx +389 -0
  105. package/src/components/epi/EpiCaseSetInfoDialog/EpiCaseSetSharingForm.tsx +155 -0
  106. package/src/components/epi/EpiCaseSetInfoDialog/EpiCaseSetSharingInfo.tsx +23 -0
  107. package/src/components/epi/EpiCaseSetInfoDialog/index.ts +1 -0
  108. package/src/components/epi/EpiCaseSummary/EpiCaseSummary.tsx +141 -0
  109. package/src/components/epi/EpiCaseSummary/index.ts +1 -0
  110. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoAccessRights.tsx +91 -0
  111. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoColAccessRights.tsx +118 -0
  112. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoData.tsx +38 -0
  113. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoDialog.tsx +39 -0
  114. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoDialogContent.tsx +169 -0
  115. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoDialogWithLoader.tsx +42 -0
  116. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoRegions.tsx +87 -0
  117. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoTrees.tsx +73 -0
  118. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoValues.tsx +61 -0
  119. package/src/components/epi/EpiCaseTypeInfoDialog/EpiCaseTypeInfoVariableDetails.tsx +151 -0
  120. package/src/components/epi/EpiCaseTypeInfoDialog/index.ts +10 -0
  121. package/src/components/epi/EpiCasesAlreadyInCaseSetWarning/EpiCasesAlreadyInCaseSetWarning.tsx +190 -0
  122. package/src/components/epi/EpiCasesAlreadyInCaseSetWarning/EpiCasesAlreadyInCaseSetWarningCaseSetLink.tsx +62 -0
  123. package/src/components/epi/EpiCasesAlreadyInCaseSetWarning/index.ts +1 -0
  124. package/src/components/epi/EpiCompletCaseTypeLoader/EpiCompletCaseTypeLoader.tsx +87 -0
  125. package/src/components/epi/EpiCompletCaseTypeLoader/index.ts +1 -0
  126. package/src/components/epi/EpiContactDetailsDialog/EpiContactDetailsDialog.tsx +151 -0
  127. package/src/components/epi/EpiContactDetailsDialog/index.ts +1 -0
  128. package/src/components/epi/EpiContextMenu/EpiContextMenu.tsx +155 -0
  129. package/src/components/epi/EpiContextMenu/index.ts +1 -0
  130. package/src/components/epi/EpiCreateEventDialog/EpiCreateEventDialog.tsx +386 -0
  131. package/src/components/epi/EpiCreateEventDialog/EpiCreateEventDialogSuccessNotificationMessage.tsx +30 -0
  132. package/src/components/epi/EpiCreateEventDialog/index.ts +1 -0
  133. package/src/components/epi/EpiCurve/EpiCurve.tsx +525 -0
  134. package/src/components/epi/EpiCurve/index.ts +1 -0
  135. package/src/components/epi/EpiCustomTabPanel/EpiCustomTabPanel.tsx +27 -0
  136. package/src/components/epi/EpiCustomTabPanel/index.ts +1 -0
  137. package/src/components/epi/EpiDashboard/EpiDashboard.tsx +390 -0
  138. package/src/components/epi/EpiDashboard/EpiDashboardDownloadSidebarItem.tsx +154 -0
  139. package/src/components/epi/EpiDashboard/EpiDashboardGeneralSettingsForm.tsx +89 -0
  140. package/src/components/epi/EpiDashboard/EpiDashboardLayoutRenderer.tsx +248 -0
  141. package/src/components/epi/EpiDashboard/EpiDashboardLayoutSettingsForm.tsx +160 -0
  142. package/src/components/epi/EpiDashboard/EpiDashboardSettingsSidebarItem.tsx +93 -0
  143. package/src/components/epi/EpiDashboard/EpiDashboardTreeSettingsForm.tsx +90 -0
  144. package/src/components/epi/EpiDashboard/index.ts +1 -0
  145. package/src/components/epi/EpiDashboardStoreLoader/EpiDashboardStoreLoader.tsx +34 -0
  146. package/src/components/epi/EpiDashboardStoreLoader/EpiDashboardStoreLoaderContent.tsx +73 -0
  147. package/src/components/epi/EpiDashboardStoreLoader/index.ts +2 -0
  148. package/src/components/epi/EpiDashboardStoreLoader/withEpiDashboardStore.tsx +19 -0
  149. package/src/components/epi/EpiDataCollectionAccessInfo/EpiDataCollectionAccessInfo.tsx +88 -0
  150. package/src/components/epi/EpiDataCollectionAccessInfo/index.ts +1 -0
  151. package/src/components/epi/EpiFindSimilarCasesDialog/EpiFindSimilarCasesDialog.tsx +302 -0
  152. package/src/components/epi/EpiFindSimilarCasesDialog/index.ts +1 -0
  153. package/src/components/epi/EpiLegendaItem/EpiLegendaItem.tsx +106 -0
  154. package/src/components/epi/EpiLegendaItem/index.ts +1 -0
  155. package/src/components/epi/EpiLineList/EpiLineList.tsx +492 -0
  156. package/src/components/epi/EpiLineList/EpiLineListPrimaryMenu.tsx +189 -0
  157. package/src/components/epi/EpiLineList/EpiLineListSecondaryMenu.tsx +56 -0
  158. package/src/components/epi/EpiLineList/EpiLineListTitle.tsx +28 -0
  159. package/src/components/epi/EpiLineList/index.ts +1 -0
  160. package/src/components/epi/EpiLineList/useEpiLineListEmitDownloadOptions.tsx +102 -0
  161. package/src/components/epi/EpiMap/EpiMap.tsx +547 -0
  162. package/src/components/epi/EpiMap/index.ts +1 -0
  163. package/src/components/epi/EpiRemoveCasesFromEventDialog/EpiRemoveCasesFromEventDialog.tsx +169 -0
  164. package/src/components/epi/EpiRemoveCasesFromEventDialog/index.ts +1 -0
  165. package/src/components/epi/EpiRemoveFindSimilarCasesResultDialog/EpiRemoveFindSimilarCasesResultDialog.tsx +216 -0
  166. package/src/components/epi/EpiSequenceDownloadDialog/EpiSequenceDownloadDialog.tsx +151 -0
  167. package/src/components/epi/EpiSequenceDownloadDialog/index.ts +1 -0
  168. package/src/components/epi/EpiStratification/EpiStratification.tsx +241 -0
  169. package/src/components/epi/EpiStratification/index.ts +1 -0
  170. package/src/components/epi/EpiTree/EpiTree.tsx +906 -0
  171. package/src/components/epi/EpiTree/index.ts +1 -0
  172. package/src/components/epi/EpiTreeDescription/EpiTreeDescription.tsx +71 -0
  173. package/src/components/epi/EpiTreeDescription/index.ts +1 -0
  174. package/src/components/epi/EpiUpload/EpiUpload.tsx +107 -0
  175. package/src/components/epi/EpiUpload/EpiUploadCaseResultTable.tsx +383 -0
  176. package/src/components/epi/EpiUpload/EpiUploadCreateCases.tsx +266 -0
  177. package/src/components/epi/EpiUpload/EpiUploadMapColumns.tsx +261 -0
  178. package/src/components/epi/EpiUpload/EpiUploadMapSequences.tsx +332 -0
  179. package/src/components/epi/EpiUpload/EpiUploadNavigation.tsx +52 -0
  180. package/src/components/epi/EpiUpload/EpiUploadSelectFile.tsx +285 -0
  181. package/src/components/epi/EpiUpload/EpiUploadSelectSequenceFiles.tsx +375 -0
  182. package/src/components/epi/EpiUpload/EpiUploadValidate.tsx +199 -0
  183. package/src/components/epi/EpiUpload/EpiUploadValidateNavigation.tsx +34 -0
  184. package/src/components/epi/EpiUpload/index.ts +1 -0
  185. package/src/components/epi/EpiUserRightsDialog/EpiUserRightsDialog.tsx +101 -0
  186. package/src/components/epi/EpiUserRightsDialog/EpiUserRightsDialogCaseAccessPolicy.tsx +160 -0
  187. package/src/components/epi/EpiUserRightsDialog/index.ts +1 -0
  188. package/src/components/epi/EpiWarning/EpiWarning.tsx +39 -0
  189. package/src/components/epi/EpiWarning/index.ts +1 -0
  190. package/src/components/epi/EpiWidget/EpiWidget.tsx +255 -0
  191. package/src/components/epi/EpiWidget/index.ts +1 -0
  192. package/src/components/epi/EpiWidgetHeaderIconButton/EpiWidgetHeaderIconButton.tsx +40 -0
  193. package/src/components/epi/EpiWidgetHeaderIconButton/index.ts +1 -0
  194. package/src/components/epi/EpiWidgetMenu/EpiWidgetMenu.tsx +56 -0
  195. package/src/components/epi/EpiWidgetMenu/index.ts +1 -0
  196. package/src/components/epi/EpiWidgetUnavailable/EpiWidgetUnavailable.tsx +51 -0
  197. package/src/components/epi/EpiWidgetUnavailable/index.ts +1 -0
  198. package/src/components/filters/BooleanFilterField/BooleanFilterField.tsx +26 -0
  199. package/src/components/filters/BooleanFilterField/index.ts +1 -0
  200. package/src/components/filters/DateFilterField/DateFilterField.tsx +23 -0
  201. package/src/components/filters/DateFilterField/index.ts +1 -0
  202. package/src/components/filters/GeoFilterField/GeoFilterField.tsx +32 -0
  203. package/src/components/filters/GeoFilterField/index.ts +1 -0
  204. package/src/components/filters/MultiSelectFilterField/MultiSelectFilterField.tsx +31 -0
  205. package/src/components/filters/MultiSelectFilterField/index.ts +1 -0
  206. package/src/components/filters/NumberRangeFilterField/NumberRangeFilterField.tsx +20 -0
  207. package/src/components/filters/NumberRangeFilterField/index.ts +1 -0
  208. package/src/components/filters/TextFilterField/TextFilterField.tsx +18 -0
  209. package/src/components/filters/TextFilterField/index.ts +1 -0
  210. package/src/components/form/fields/Autocomplete/Autocomplete.tsx +270 -0
  211. package/src/components/form/fields/Autocomplete/index.ts +1 -0
  212. package/src/components/form/fields/CheckboxGroup/CheckboxGroup.tsx +192 -0
  213. package/src/components/form/fields/CheckboxGroup/index.ts +1 -0
  214. package/src/components/form/fields/DatePicker/DatePicker.tsx +167 -0
  215. package/src/components/form/fields/DatePicker/index.ts +1 -0
  216. package/src/components/form/fields/DateRangePicker/DateRangePicker.tsx +359 -0
  217. package/src/components/form/fields/DateRangePicker/index.ts +1 -0
  218. package/src/components/form/fields/NumberField/NumberField.tsx +277 -0
  219. package/src/components/form/fields/NumberField/index.ts +1 -0
  220. package/src/components/form/fields/NumberRangeInput/NumberRangeInput.tsx +297 -0
  221. package/src/components/form/fields/NumberRangeInput/index.ts +1 -0
  222. package/src/components/form/fields/RadioGroup/RadioGroup.tsx +130 -0
  223. package/src/components/form/fields/RadioGroup/index.ts +1 -0
  224. package/src/components/form/fields/RichTextEditor/RichTextEditor.tsx +321 -0
  225. package/src/components/form/fields/RichTextEditor/RichTextEditorContent.tsx +34 -0
  226. package/src/components/form/fields/RichTextEditor/index.ts +2 -0
  227. package/src/components/form/fields/RichTextEditor/useRichTextEditorExtensions.ts +174 -0
  228. package/src/components/form/fields/Select/Select.tsx +219 -0
  229. package/src/components/form/fields/Select/index.ts +1 -0
  230. package/src/components/form/fields/Switch/Switch.tsx +116 -0
  231. package/src/components/form/fields/Switch/index.ts +1 -0
  232. package/src/components/form/fields/TextField/TextField.tsx +170 -0
  233. package/src/components/form/fields/TextField/index.ts +1 -0
  234. package/src/components/form/fields/ToggleButtonGroup/ToggleButtonGroup.tsx +118 -0
  235. package/src/components/form/fields/ToggleButtonGroup/index.ts +1 -0
  236. package/src/components/form/fields/TransferList/TransferList.tsx +360 -0
  237. package/src/components/form/fields/TransferList/index.ts +1 -0
  238. package/src/components/form/fields/UploadButton/UploadButton.tsx +157 -0
  239. package/src/components/form/helpers/FormFieldHelperText/FormFieldHelperText.tsx +68 -0
  240. package/src/components/form/helpers/FormFieldHelperText/index.ts +1 -0
  241. package/src/components/form/helpers/FormFieldLoadingIndicator/FormFieldLoadingIndicator.tsx +22 -0
  242. package/src/components/form/helpers/FormFieldLoadingIndicator/index.ts +1 -0
  243. package/src/components/form/helpers/GenericForm/GenericForm.tsx +187 -0
  244. package/src/components/form/helpers/GenericForm/index.ts +1 -0
  245. package/src/components/ui/ApplicationBar/ApplicationBar.tsx +74 -0
  246. package/src/components/ui/ApplicationBar/ApplicationBarActions.tsx +124 -0
  247. package/src/components/ui/ApplicationBar/ApplicationBarActionsFeedbackItem.tsx +102 -0
  248. package/src/components/ui/ApplicationBar/ApplicationBarActionsInfotem.tsx +60 -0
  249. package/src/components/ui/ApplicationBar/ApplicationBarActionsNotificationsItem.tsx +82 -0
  250. package/src/components/ui/ApplicationBar/ApplicationBarActionsOrganizationSwitcherItem.tsx +58 -0
  251. package/src/components/ui/ApplicationBar/ApplicationBarActionsOutagesItem.tsx +75 -0
  252. package/src/components/ui/ApplicationBar/ApplicationBarActionsUserItem.tsx +60 -0
  253. package/src/components/ui/ApplicationBar/ApplicationBarNavigationMenu.tsx +159 -0
  254. package/src/components/ui/ApplicationBar/InfoMenu.tsx +120 -0
  255. package/src/components/ui/ApplicationBar/UserMenu.tsx +208 -0
  256. package/src/components/ui/ApplicationBar/UserOrganizationAdminMenuItem.tsx +69 -0
  257. package/src/components/ui/ApplicationBar/UserOwnOrganizationMenuItem.tsx +54 -0
  258. package/src/components/ui/ApplicationBar/index.ts +1 -0
  259. package/src/components/ui/ApplicationFooter/ApplicationFooter.tsx +121 -0
  260. package/src/components/ui/ApplicationFooter/ApplicationFooterLink.tsx +48 -0
  261. package/src/components/ui/ApplicationFooter/ApplicationFooterLinkSection.tsx +56 -0
  262. package/src/components/ui/ApplicationFooter/index.ts +1 -0
  263. package/src/components/ui/AuthenticationWrapper/AuthenticationWrapper.tsx +208 -0
  264. package/src/components/ui/AuthenticationWrapper/index.ts +1 -0
  265. package/src/components/ui/AuthorizationWrapper/AuthorizationWrapper.tsx +75 -0
  266. package/src/components/ui/AuthorizationWrapper/index.ts +1 -0
  267. package/src/components/ui/Breadcrumbs/Breadcrumbs.tsx +92 -0
  268. package/src/components/ui/Breadcrumbs/index.ts +1 -0
  269. package/src/components/ui/Confirmation/Confirmation.tsx +10 -0
  270. package/src/components/ui/Confirmation/ConfirmationRender.tsx +81 -0
  271. package/src/components/ui/Confirmation/index.ts +2 -0
  272. package/src/components/ui/ConsentDialog/ConsentDialog.tsx +57 -0
  273. package/src/components/ui/ConsentDialog/index.ts +1 -0
  274. package/src/components/ui/CopyToClipboardButton/CopyToClipboardButton.tsx +94 -0
  275. package/src/components/ui/CopyToClipboardButton/index.ts +1 -0
  276. package/src/components/ui/Dialog/Dialog.tsx +196 -0
  277. package/src/components/ui/Dialog/index.ts +1 -0
  278. package/src/components/ui/FileSelector/FileSelector.tsx +358 -0
  279. package/src/components/ui/FileSelector/index.ts +1 -0
  280. package/src/components/ui/GenericErrorMessage/GenericErrorMessage.tsx +168 -0
  281. package/src/components/ui/GenericErrorMessage/index.ts +1 -0
  282. package/src/components/ui/HomePageTrends/HomePageTrendCard.tsx +132 -0
  283. package/src/components/ui/HomePageTrends/HomePageTrends.tsx +312 -0
  284. package/src/components/ui/HomePageTrends/index.ts +1 -0
  285. package/src/components/ui/LicensesDialog/LicensesDialog.tsx +231 -0
  286. package/src/components/ui/LicensesDialog/index.ts +1 -0
  287. package/src/components/ui/LinearProgressWithLabel/LinearProgressWithLabel.tsx +31 -0
  288. package/src/components/ui/LinearProgressWithLabel/index.ts +1 -0
  289. package/src/components/ui/LoadingIndicator/LoadingIndicator.tsx +11 -0
  290. package/src/components/ui/LoadingIndicator/index.ts +1 -0
  291. package/src/components/ui/MyPermissionsDialog/MyPermissionsDialog.tsx +95 -0
  292. package/src/components/ui/MyPermissionsDialog/index.ts +1 -0
  293. package/src/components/ui/NavLink/NavLink.tsx +47 -0
  294. package/src/components/ui/NavLink/index.ts +1 -0
  295. package/src/components/ui/NestedMenu/IconMenuItem.tsx +64 -0
  296. package/src/components/ui/NestedMenu/NestedDropdown.tsx +133 -0
  297. package/src/components/ui/NestedMenu/NestedMenuItem.tsx +250 -0
  298. package/src/components/ui/NestedMenu/index.ts +4 -0
  299. package/src/components/ui/NestedMenu/nestedMenuItemsFromObject.tsx +98 -0
  300. package/src/components/ui/Notifications/NotificationItem.tsx +70 -0
  301. package/src/components/ui/Notifications/NotificationsDrawer.tsx +161 -0
  302. package/src/components/ui/Notifications/NotificationsStack.tsx +45 -0
  303. package/src/components/ui/Notifications/index.ts +2 -0
  304. package/src/components/ui/OrganizationSwitcherDialog/OrganizationSwitcherDialog.tsx +204 -0
  305. package/src/components/ui/OrganizationSwitcherDialog/index.ts +1 -0
  306. package/src/components/ui/OutageList/OutageItem.tsx +48 -0
  307. package/src/components/ui/OutageList/OutageList.tsx +54 -0
  308. package/src/components/ui/OutageList/OutageSection.tsx +39 -0
  309. package/src/components/ui/OutageList/index.ts +1 -0
  310. package/src/components/ui/OutagesDialog/OutagesDialog.tsx +71 -0
  311. package/src/components/ui/OutagesDialog/index.ts +1 -0
  312. package/src/components/ui/PageContainer/PageContainer.tsx +207 -0
  313. package/src/components/ui/PageContainer/index.ts +1 -0
  314. package/src/components/ui/PanelSeparator/PanelResizeHandle.tsx +67 -0
  315. package/src/components/ui/PanelSeparator/index.ts +1 -0
  316. package/src/components/ui/ResponseHandler/ResponseHandler.tsx +79 -0
  317. package/src/components/ui/ResponseHandler/index.ts +1 -0
  318. package/src/components/ui/Sidebar/SidebarItem.tsx +138 -0
  319. package/src/components/ui/Sidebar/SidebarMenu.tsx +33 -0
  320. package/src/components/ui/Sidebar/SidebarMenuItem.tsx +61 -0
  321. package/src/components/ui/Sidebar/index.ts +3 -0
  322. package/src/components/ui/SortableList/SortableList.tsx +172 -0
  323. package/src/components/ui/SortableList/SortableListItem.tsx +73 -0
  324. package/src/components/ui/SortableList/SortableListItemDragHandle.tsx +53 -0
  325. package/src/components/ui/SortableList/SortableOverlay.tsx +35 -0
  326. package/src/components/ui/SortableList/context/SortableListItemContext.tsx +17 -0
  327. package/src/components/ui/SortableList/context/SortableListItemContextProvider.tsx +19 -0
  328. package/src/components/ui/SortableList/context/useSortableListItemContext.tsx +5 -0
  329. package/src/components/ui/Spinner/Spinner.tsx +84 -0
  330. package/src/components/ui/Spinner/index.ts +1 -0
  331. package/src/components/ui/Stepper/Stepper.tsx +194 -0
  332. package/src/components/ui/Stepper/index.ts +2 -0
  333. package/src/components/ui/Stepper/stepperModel.ts +4 -0
  334. package/src/components/ui/Table/Table.tsx +812 -0
  335. package/src/components/ui/Table/TableActionsCell.tsx +81 -0
  336. package/src/components/ui/Table/TableCaption.tsx +33 -0
  337. package/src/components/ui/Table/TableCell.tsx +162 -0
  338. package/src/components/ui/Table/TableCellAsyncContent.tsx +28 -0
  339. package/src/components/ui/Table/TableCheckboxCell.tsx +65 -0
  340. package/src/components/ui/Table/TableCheckboxHeader.tsx +72 -0
  341. package/src/components/ui/Table/TableColumnsEditorDialog.tsx +241 -0
  342. package/src/components/ui/Table/TableFilter.tsx +31 -0
  343. package/src/components/ui/Table/TableFiltersSidebarItem.tsx +398 -0
  344. package/src/components/ui/Table/TableHeader.tsx +40 -0
  345. package/src/components/ui/Table/TableHeaderCell.tsx +434 -0
  346. package/src/components/ui/Table/TableHeaderFilter.tsx +104 -0
  347. package/src/components/ui/Table/TableMenu.tsx +26 -0
  348. package/src/components/ui/Table/TableReadableIndexCell.tsx +65 -0
  349. package/src/components/ui/Table/TableSidebarMenu.tsx +59 -0
  350. package/src/components/ui/Table/classNames.ts +7 -0
  351. package/src/components/ui/Table/index.ts +12 -0
  352. package/src/components/ui/UserFeedbackDialog/UserFeedbackDialog.tsx +168 -0
  353. package/src/components/ui/UserFeedbackDialog/index.ts +1 -0
  354. package/src/components/ui/UserInactivityConfirmation/UserInactivityConfirmation.tsx +59 -0
  355. package/src/components/ui/UserInactivityConfirmation/index.ts +1 -0
  356. package/src/components/ui/UsersEffectiveRightsDetailsDialog/UsersEffectiveRightsDetailsDialog.tsx +584 -0
  357. package/src/components/ui/UsersEffectiveRightsDetailsDialog/index.ts +1 -0
  358. package/src/context/caseAbac/CaseAbacContext.tsx +24 -0
  359. package/src/context/caseAbac/CaseAbacContextProvider.tsx +69 -0
  360. package/src/context/caseAbac/index.ts +3 -0
  361. package/src/context/caseAbac/useCaseAbacContext.tsx +6 -0
  362. package/src/context/caseTypeAbac/CaseTypeAbacContext.tsx +16 -0
  363. package/src/context/caseTypeAbac/CaseTypeAbacContextProvider.tsx +40 -0
  364. package/src/context/caseTypeAbac/index.ts +3 -0
  365. package/src/context/caseTypeAbac/useCaseTypeAbacContext.tsx +6 -0
  366. package/src/data/date.ts +8 -0
  367. package/src/data/queryDependencies.ts +166 -0
  368. package/src/dataHooks/useAssemblyProtocolsQuery/index.ts +1 -0
  369. package/src/dataHooks/useAssemblyProtocolsQuery/useAssemblyProtocolsQuery.ts +39 -0
  370. package/src/dataHooks/useCaseRightsQuery/index.ts +1 -0
  371. package/src/dataHooks/useCaseRightsQuery/useCaseRightsQuery.ts +21 -0
  372. package/src/dataHooks/useCaseSetCategoriesQuery/index.ts +1 -0
  373. package/src/dataHooks/useCaseSetCategoriesQuery/useCaseSetCategoriesQuery.ts +39 -0
  374. package/src/dataHooks/useCaseSetRightsQuery/index.ts +1 -0
  375. package/src/dataHooks/useCaseSetRightsQuery/useCaseSetRightsQuery.ts +17 -0
  376. package/src/dataHooks/useCaseSetStatsQuery/index.ts +1 -0
  377. package/src/dataHooks/useCaseSetStatsQuery/useCaseSetStatsQuery.ts +31 -0
  378. package/src/dataHooks/useCaseSetStatusesQuery/index.ts +1 -0
  379. package/src/dataHooks/useCaseSetStatusesQuery/useCaseSetStatusesQuery.ts +39 -0
  380. package/src/dataHooks/useCaseSetsQuery/index.ts +1 -0
  381. package/src/dataHooks/useCaseSetsQuery/useCaseSetsQuery.ts +41 -0
  382. package/src/dataHooks/useCaseTypeSetCategoriesQuery/index.ts +1 -0
  383. package/src/dataHooks/useCaseTypeSetCategoriesQuery/useCaseTypeSetCategoriesQuery.ts +41 -0
  384. package/src/dataHooks/useCaseTypeSetMembersQuery/index.ts +1 -0
  385. package/src/dataHooks/useCaseTypeSetMembersQuery/useCaseTypeSetMembersQuery.ts +17 -0
  386. package/src/dataHooks/useCaseTypeSetsQuery/index.ts +1 -0
  387. package/src/dataHooks/useCaseTypeSetsQuery/useCaseTypeSetsQuery.ts +58 -0
  388. package/src/dataHooks/useCaseTypeStatsQuery/index.ts +1 -0
  389. package/src/dataHooks/useCaseTypeStatsQuery/useCaseTypeStatsQuery.ts +20 -0
  390. package/src/dataHooks/useCaseTypesQuery/index.ts +1 -0
  391. package/src/dataHooks/useCaseTypesQuery/useCaseTypesQuery.ts +43 -0
  392. package/src/dataHooks/useColSetMembersQuery/index.ts +1 -0
  393. package/src/dataHooks/useColSetMembersQuery/useColSetMembersQuery.ts +17 -0
  394. package/src/dataHooks/useColSetsQuery/index.ts +1 -0
  395. package/src/dataHooks/useColSetsQuery/useColSetsQuery.ts +39 -0
  396. package/src/dataHooks/useColTypesQuery/index.ts +1 -0
  397. package/src/dataHooks/useColTypesQuery/useColTypesQuery.ts +51 -0
  398. package/src/dataHooks/useColsQuery/index.ts +1 -0
  399. package/src/dataHooks/useColsQuery/useColsQuery.ts +60 -0
  400. package/src/dataHooks/useConceptQuery/index.ts +1 -0
  401. package/src/dataHooks/useConceptQuery/useConceptQuery.ts +52 -0
  402. package/src/dataHooks/useConceptRelationTypeQuery/index.ts +1 -0
  403. package/src/dataHooks/useConceptRelationTypeQuery/useConceptRelationTypeQuery.ts +21 -0
  404. package/src/dataHooks/useConceptSetTypeQuery/index.ts +1 -0
  405. package/src/dataHooks/useConceptSetTypeQuery/useConceptSetTypeQuery.ts +27 -0
  406. package/src/dataHooks/useConceptSetsQuery/index.ts +1 -0
  407. package/src/dataHooks/useConceptSetsQuery/useConceptSetsQuery.ts +38 -0
  408. package/src/dataHooks/useDataCollectionSetMembersQuery/index.ts +1 -0
  409. package/src/dataHooks/useDataCollectionSetMembersQuery/useDataCollectionSetMembersQuery.ts +17 -0
  410. package/src/dataHooks/useDataCollectionsQuery/index.ts +1 -0
  411. package/src/dataHooks/useDataCollectionsQuery/useDataCollectionsQuery.ts +45 -0
  412. package/src/dataHooks/useDimTypesQuery/index.ts +1 -0
  413. package/src/dataHooks/useDimTypesQuery/useDimTypesQuery.ts +26 -0
  414. package/src/dataHooks/useDimsQuery/index.ts +1 -0
  415. package/src/dataHooks/useDimsQuery/useDimsQuery.ts +55 -0
  416. package/src/dataHooks/useDiseasesQuery/index.ts +1 -0
  417. package/src/dataHooks/useDiseasesQuery/useDiseasesQuery.ts +39 -0
  418. package/src/dataHooks/useEtiologicalAgentsQuery/index.ts +1 -0
  419. package/src/dataHooks/useEtiologicalAgentsQuery/useEtiologicalAgentsQuery.ts +39 -0
  420. package/src/dataHooks/useGeneticDistanceProtocolsQuery/index.ts +1 -0
  421. package/src/dataHooks/useGeneticDistanceProtocolsQuery/useGeneticDistanceProtocolsQuery.ts +29 -0
  422. package/src/dataHooks/useIdentifierIssuerOwnOrganizationQuery/index.ts +1 -0
  423. package/src/dataHooks/useIdentifierIssuerOwnOrganizationQuery/useIdentifierIssuerOwnOrganizationQuery.ts +34 -0
  424. package/src/dataHooks/useIdentifierIssuerQuery/index.ts +1 -0
  425. package/src/dataHooks/useIdentifierIssuerQuery/useIdentifierIssuerQuery.ts +42 -0
  426. package/src/dataHooks/useInviteUserConstraintsQuery/index.ts +1 -0
  427. package/src/dataHooks/useInviteUserConstraintsQuery/useInviteUserConstraintsQuery.ts +32 -0
  428. package/src/dataHooks/useOrganizationAccessCasePoliciesQuery/index.ts +1 -0
  429. package/src/dataHooks/useOrganizationAccessCasePoliciesQuery/useOrganizationAccessCasePoliciesQuery.ts +20 -0
  430. package/src/dataHooks/useOrganizationAdminPoliciesQuery/index.ts +1 -0
  431. package/src/dataHooks/useOrganizationAdminPoliciesQuery/useOrganizationAdminPoliciesQuery.ts +60 -0
  432. package/src/dataHooks/useOrganizationIdentifierIssuerLinksQuery/index.ts +1 -0
  433. package/src/dataHooks/useOrganizationIdentifierIssuerLinksQuery/useOrganizationIdentifierIssuerLinksQuery.ts +18 -0
  434. package/src/dataHooks/useOrganizationShareCasePoliciesQuery/index.ts +1 -0
  435. package/src/dataHooks/useOrganizationShareCasePoliciesQuery/useOrganizationShareCasePoliciesQuery.ts +18 -0
  436. package/src/dataHooks/useOrganizationsQuery/index.ts +1 -0
  437. package/src/dataHooks/useOrganizationsQuery/useOrganizationsQuery.ts +40 -0
  438. package/src/dataHooks/useRefColsQuery/index.ts +1 -0
  439. package/src/dataHooks/useRefColsQuery/useRefColsQuery.ts +39 -0
  440. package/src/dataHooks/useRefColsValidationRulesQuery/index.ts +1 -0
  441. package/src/dataHooks/useRefColsValidationRulesQuery/useRefColsValidationRulesQuery.ts +17 -0
  442. package/src/dataHooks/useRefDimsQuery/index.ts +1 -0
  443. package/src/dataHooks/useRefDimsQuery/useRefDimsQuery.ts +39 -0
  444. package/src/dataHooks/useRegionQuery/index.ts +1 -0
  445. package/src/dataHooks/useRegionQuery/useRegionQuery.ts +52 -0
  446. package/src/dataHooks/useRegionRelationTypeQuery/index.ts +1 -0
  447. package/src/dataHooks/useRegionRelationTypeQuery/useRegionRelationTypeQuery.ts +24 -0
  448. package/src/dataHooks/useRegionSetsQuery/index.ts +1 -0
  449. package/src/dataHooks/useRegionSetsQuery/useRegionSetsQuery.ts +39 -0
  450. package/src/dataHooks/useSequencingProtocolsQuery/index.ts +1 -0
  451. package/src/dataHooks/useSequencingProtocolsQuery/useSequencingProtocolsQuery.ts +39 -0
  452. package/src/dataHooks/useTreeAlgorithmCodesQuery/index.ts +1 -0
  453. package/src/dataHooks/useTreeAlgorithmCodesQuery/useTreeAlgorithmCodesQuery.ts +42 -0
  454. package/src/dataHooks/useUserAccessCasePoliciesQuery/index.ts +1 -0
  455. package/src/dataHooks/useUserAccessCasePoliciesQuery/useUserAccessCasePoliciesQuery.ts +18 -0
  456. package/src/dataHooks/useUserEffectiveRightsQuery/index.ts +1 -0
  457. package/src/dataHooks/useUserEffectiveRightsQuery/useUserEffectiveRightsQuery.ts +125 -0
  458. package/src/dataHooks/useUserShareCasePoliciesQuery/index.ts +1 -0
  459. package/src/dataHooks/useUserShareCasePoliciesQuery/useUserShareCasePoliciesQuery.ts +18 -0
  460. package/src/dataHooks/useUsersQuery/index.ts +1 -0
  461. package/src/dataHooks/useUsersQuery/useUsersQuery.ts +43 -0
  462. package/src/hoc/withDialog/index.ts +1 -0
  463. package/src/hoc/withDialog/withDialog.tsx +132 -0
  464. package/src/hoc/withPermissions/index.ts +1 -0
  465. package/src/hoc/withPermissions/withPermissions.tsx +34 -0
  466. package/src/hooks/useArray/index.ts +1 -0
  467. package/src/hooks/useArray/useArray.ts +5 -0
  468. package/src/hooks/useColumnsMenu/index.ts +1 -0
  469. package/src/hooks/useColumnsMenu/useColumnsMenu.tsx +152 -0
  470. package/src/hooks/useCreateMutation/index.ts +1 -0
  471. package/src/hooks/useCreateMutation/useCreateMutation.ts +103 -0
  472. package/src/hooks/useDeleteMutation/index.ts +1 -0
  473. package/src/hooks/useDeleteMutation/useDeleteMutation.ts +92 -0
  474. package/src/hooks/useDimensions/index.ts +1 -0
  475. package/src/hooks/useDimensions/useDimensions.ts +82 -0
  476. package/src/hooks/useEditMutation/index.ts +1 -0
  477. package/src/hooks/useEditMutation/useEditMutation.ts +112 -0
  478. package/src/hooks/useInitializeTableStore/index.ts +1 -0
  479. package/src/hooks/useInitializeTableStore/useInitializeTableStore.ts +41 -0
  480. package/src/hooks/useIsFormFieldRequiredFromSchema/index.ts +1 -0
  481. package/src/hooks/useIsFormFieldRequiredFromSchema/useIsFormFieldRequiredFromSchema.ts +32 -0
  482. package/src/hooks/useItemQuery/index.ts +1 -0
  483. package/src/hooks/useItemQuery/useItemQuery.ts +82 -0
  484. package/src/hooks/useOrganizationCasePolicyNameFactory/index.ts +1 -0
  485. package/src/hooks/useOrganizationCasePolicyNameFactory/useOrganizationCasePolicyNameFactory.ts +31 -0
  486. package/src/hooks/useQueryMemo/index.ts +1 -0
  487. package/src/hooks/useQueryMemo/useQueryMemo.ts +24 -0
  488. package/src/hooks/useScrollbarSize/index.ts +1 -0
  489. package/src/hooks/useScrollbarSize/useScrollbarSize.ts +23 -0
  490. package/src/hooks/useSubscribable/index.ts +1 -0
  491. package/src/hooks/useSubscribable/useSubscribable.ts +30 -0
  492. package/src/hooks/useUpdateBreadcrumb/index.ts +1 -0
  493. package/src/hooks/useUpdateBreadcrumb/useUpdateBreadcrumb.ts +21 -0
  494. package/src/hooks/useUpdateDocumentTitle/index.ts +1 -0
  495. package/src/hooks/useUpdateDocumentTitle/useUpdateDocumentTitle.ts +10 -0
  496. package/src/hooks/useUserCasePolicyNameFactory/index.ts +1 -0
  497. package/src/hooks/useUserCasePolicyNameFactory/useUserCasePolicyNameFactory.ts +37 -0
  498. package/src/index.ts +301 -0
  499. package/src/models/admin.ts +7 -0
  500. package/src/models/auth.ts +14 -0
  501. package/src/models/caseAccess.ts +21 -0
  502. package/src/models/config.ts +153 -0
  503. package/src/models/data.ts +11 -0
  504. package/src/models/dataHooks.ts +18 -0
  505. package/src/models/environment.ts +10 -0
  506. package/src/models/epi.ts +188 -0
  507. package/src/models/filter.ts +39 -0
  508. package/src/models/form.ts +74 -0
  509. package/src/models/generic.ts +4 -0
  510. package/src/models/nestedMenu.ts +21 -0
  511. package/src/models/notification.ts +12 -0
  512. package/src/models/outage.ts +7 -0
  513. package/src/models/query.ts +75 -0
  514. package/src/models/reactRouter.ts +36 -0
  515. package/src/models/table.ts +191 -0
  516. package/src/models/testId.ts +1 -0
  517. package/src/models/tree.ts +32 -0
  518. package/src/pages/AcceptInvitationPage/AcceptInvitationPage.tsx +96 -0
  519. package/src/pages/AcceptInvitationPage/index.ts +1 -0
  520. package/src/pages/AdminPage/AdminPage.tsx +164 -0
  521. package/src/pages/AdminPage/index.ts +1 -0
  522. package/src/pages/CaseSetStatusAdminPage/CaseSetStatusAdminPage.tsx +98 -0
  523. package/src/pages/CaseSetStatusAdminPage/index.ts +1 -0
  524. package/src/pages/CaseTypeSetCategoriesAdminPage/CaseTypeSetCategoriesAdminPage.tsx +108 -0
  525. package/src/pages/CaseTypeSetCategoriesAdminPage/index.ts +1 -0
  526. package/src/pages/CaseTypeSetsAdminPage/CaseTypeSetsAdminPage.tsx +200 -0
  527. package/src/pages/CaseTypeSetsAdminPage/index.ts +1 -0
  528. package/src/pages/CaseTypesAdminPage/CaseTypesAdminPage.tsx +204 -0
  529. package/src/pages/CaseTypesAdminPage/index.ts +1 -0
  530. package/src/pages/CasesDetailPage/CasesDetailPage.tsx +57 -0
  531. package/src/pages/CasesDetailPage/index.ts +1 -0
  532. package/src/pages/CasesPage/CasesPage.tsx +338 -0
  533. package/src/pages/CasesPage/index.ts +1 -0
  534. package/src/pages/ChooseIdentityProviderPage/ChooseIdentityProviderPage.tsx +166 -0
  535. package/src/pages/ChooseIdentityProviderPage/index.ts +1 -0
  536. package/src/pages/ColSetsAdminPage/ColSetsAdminPage.tsx +175 -0
  537. package/src/pages/ColSetsAdminPage/index.ts +1 -0
  538. package/src/pages/ColsAdminPage/ColsAdminPage.tsx +400 -0
  539. package/src/pages/ColsAdminPage/index.ts +1 -0
  540. package/src/pages/ConceptRelationsAdminPage/ConceptRelationsAdminPage.tsx +134 -0
  541. package/src/pages/ConceptRelationsAdminPage/index.ts +1 -0
  542. package/src/pages/ConceptSetsAdminPage/ConceptSetsAdminPage.tsx +167 -0
  543. package/src/pages/ConceptSetsAdminPage/index.ts +1 -0
  544. package/src/pages/ConceptsAdminPage/ConceptsAdminPage.tsx +138 -0
  545. package/src/pages/ConceptsAdminPage/index.ts +1 -0
  546. package/src/pages/CrudPage/CrudPage.tsx +646 -0
  547. package/src/pages/CrudPage/CrudPageDeleteDialog.tsx +85 -0
  548. package/src/pages/CrudPage/CrudPageEditDialog.tsx +165 -0
  549. package/src/pages/CrudPage/index.ts +1 -0
  550. package/src/pages/DataCollectionSetsAdminPage/DataCollectionSetsAdminPage.tsx +156 -0
  551. package/src/pages/DataCollectionSetsAdminPage/index.ts +1 -0
  552. package/src/pages/DataCollectionVisualizationPage/DataCollectionVisualizationPage.tsx +238 -0
  553. package/src/pages/DataCollectionVisualizationPage/index.ts +1 -0
  554. package/src/pages/DataCollectionsAdminPage/DataCollectionsAdminPage.tsx +96 -0
  555. package/src/pages/DataCollectionsAdminPage/index.ts +1 -0
  556. package/src/pages/DimsAdminPage/DimsAdminPage.tsx +208 -0
  557. package/src/pages/DimsAdminPage/index.ts +1 -0
  558. package/src/pages/DiseasesAdminPage/DiseasesAdminPage.tsx +97 -0
  559. package/src/pages/DiseasesAdminPage/index.ts +1 -0
  560. package/src/pages/ErrorPage/ErrorPage.tsx +25 -0
  561. package/src/pages/ErrorPage/index.ts +1 -0
  562. package/src/pages/EtiologicalAgentsAdminPage/EtiologicalAgentsAdminPage.tsx +96 -0
  563. package/src/pages/EtiologicalAgentsAdminPage/index.ts +1 -0
  564. package/src/pages/EtiologiesAdminPage/EtiologiesAdminPage.tsx +110 -0
  565. package/src/pages/EtiologiesAdminPage/index.ts +1 -0
  566. package/src/pages/EventsDetailPage/EventsDetailPage.tsx +67 -0
  567. package/src/pages/EventsDetailPage/index.ts +1 -0
  568. package/src/pages/EventsPage/EventsPage.tsx +262 -0
  569. package/src/pages/EventsPage/index.ts +1 -0
  570. package/src/pages/HomePage/HomePage.tsx +47 -0
  571. package/src/pages/HomePage/index.ts +1 -0
  572. package/src/pages/IdentifierIssuersAdminPage/IdentifierIssuersAdminPage.tsx +106 -0
  573. package/src/pages/IdentifierIssuersAdminPage/index.ts +1 -0
  574. package/src/pages/OrganizationAccessCasePoliciesAdminPage/OrganizationAccessCasePoliciesAdminPage.tsx +217 -0
  575. package/src/pages/OrganizationAccessCasePoliciesAdminPage/index.ts +1 -0
  576. package/src/pages/OrganizationAdminPoliciesAdminPage/OrganizationAdminPoliciesAdminPage.tsx +119 -0
  577. package/src/pages/OrganizationAdminPoliciesAdminPage/index.ts +1 -0
  578. package/src/pages/OrganizationContactsAdminPage/OrganizationContactsAdminPage.tsx +127 -0
  579. package/src/pages/OrganizationContactsAdminPage/index.ts +1 -0
  580. package/src/pages/OrganizationShareCasePoliciesAdminPage/OrganizationShareCasePoliciesAdminPage.tsx +183 -0
  581. package/src/pages/OrganizationShareCasePoliciesAdminPage/index.ts +1 -0
  582. package/src/pages/OrganizationSitesAdminPage/OrganizationSitesAdminPage.tsx +132 -0
  583. package/src/pages/OrganizationSitesAdminPage/index.ts +1 -0
  584. package/src/pages/OrganizationsAdminPage/OrganizationsAdminPage.tsx +188 -0
  585. package/src/pages/OrganizationsAdminPage/index.ts +1 -0
  586. package/src/pages/OutagesAdminPage/OutagesAdminPage.tsx +158 -0
  587. package/src/pages/OutagesAdminPage/index.ts +1 -0
  588. package/src/pages/PostLoginPage/PostLoginPage.tsx +39 -0
  589. package/src/pages/PostLoginPage/index.ts +1 -0
  590. package/src/pages/PostLogoutPage/PostLogoutPage.tsx +44 -0
  591. package/src/pages/PostLogoutPage/index.ts +1 -0
  592. package/src/pages/RefColsAdminPage/RefColsAdminPage.tsx +286 -0
  593. package/src/pages/RefColsAdminPage/index.ts +1 -0
  594. package/src/pages/RefDimsAdminPage/RefDimsAdminPage.tsx +152 -0
  595. package/src/pages/RefDimsAdminPage/index.ts +1 -0
  596. package/src/pages/RegionRelationsAdminPage/RegionRelationsAdminPage.tsx +132 -0
  597. package/src/pages/RegionRelationsAdminPage/index.ts +1 -0
  598. package/src/pages/RegionSetShapesAdminPage/RegionSetShapesAdminPage.tsx +132 -0
  599. package/src/pages/RegionSetShapesAdminPage/index.ts +1 -0
  600. package/src/pages/RegionSetsAdminPage/RegionSetsAdminPage.tsx +147 -0
  601. package/src/pages/RegionSetsAdminPage/index.ts +1 -0
  602. package/src/pages/RegionsAdminPage/RegionsAdminPage.tsx +150 -0
  603. package/src/pages/RegionsAdminPage/index.ts +1 -0
  604. package/src/pages/RouterErrorPage/RouterErrorPage.tsx +23 -0
  605. package/src/pages/RouterErrorPage/index.ts +1 -0
  606. package/src/pages/TrendsPage/TrendsPage.tsx +17 -0
  607. package/src/pages/TrendsPage/index.ts +1 -0
  608. package/src/pages/UploadPage/UploadPage.tsx +40 -0
  609. package/src/pages/UploadPage/index.ts +1 -0
  610. package/src/pages/UserAccessCasePoliciesAdminPage/UserAccessCasePoliciesAdminPage.tsx +209 -0
  611. package/src/pages/UserAccessCasePoliciesAdminPage/index.ts +1 -0
  612. package/src/pages/UserEffectiveRightsAdminPage/UserEffectiveRightsAdminPage.tsx +331 -0
  613. package/src/pages/UserEffectiveRightsAdminPage/index.ts +1 -0
  614. package/src/pages/UserEffectiveRightsTesterAdminPage/UserEffectiveRightsTesterAdminPage.tsx +283 -0
  615. package/src/pages/UserEffectiveRightsTesterAdminPage/index.ts +1 -0
  616. package/src/pages/UserInvitationsAdminPage/UserInvitationConsumeDialog.tsx +143 -0
  617. package/src/pages/UserInvitationsAdminPage/UserInvitationShareDialog.tsx +100 -0
  618. package/src/pages/UserInvitationsAdminPage/UserInvitationsAdminPage.tsx +218 -0
  619. package/src/pages/UserInvitationsAdminPage/index.ts +1 -0
  620. package/src/pages/UserShareCasePoliciesAdminPage/UserShareCasePoliciesAdminPage.tsx +183 -0
  621. package/src/pages/UserShareCasePoliciesAdminPage/index.ts +1 -0
  622. package/src/pages/UsersAdminPage/UsersAdminPage.tsx +240 -0
  623. package/src/pages/UsersAdminPage/index.ts +1 -0
  624. package/src/routes/adminRoutes.tsx +801 -0
  625. package/src/routes/index.ts +2 -0
  626. package/src/routes/routes.tsx +235 -0
  627. package/src/setup/index.ts +1 -0
  628. package/src/setup/setup.ts +5 -0
  629. package/src/setup/yup.ts +203 -0
  630. package/src/stores/epiDashboardStore/epiDashboardStore.ts +750 -0
  631. package/src/stores/epiDashboardStore/epiDashboardStoreContext.tsx +6 -0
  632. package/src/stores/epiDashboardStore/index.ts +2 -0
  633. package/src/stores/epiUploadStore/epiUploadStore.ts +351 -0
  634. package/src/stores/epiUploadStore/epiUploadStoreContext.tsx +6 -0
  635. package/src/stores/epiUploadStore/index.ts +2 -0
  636. package/src/stores/oidcStore/index.ts +1 -0
  637. package/src/stores/oidcStore/oidcStore.ts +43 -0
  638. package/src/stores/outagesStore/index.ts +1 -0
  639. package/src/stores/outagesStore/outagesStore.ts +31 -0
  640. package/src/stores/tableStore/TableStoreContext.tsx +6 -0
  641. package/src/stores/tableStore/TableStoreContextProvider.tsx +20 -0
  642. package/src/stores/tableStore/index.ts +4 -0
  643. package/src/stores/tableStore/tableStore.ts +547 -0
  644. package/src/stores/tableStore/useTableStoreContext.tsx +7 -0
  645. package/src/stores/userProfileStore/index.ts +1 -0
  646. package/src/stores/userProfileStore/userProfileStore.ts +112 -0
  647. package/src/test/integration/lib/render.tsx +33 -0
  648. package/src/test/integration/tests/integration-example/HelloWorld.test.tsx +14 -0
  649. package/src/test/integration/tests/integration-example/HelloWorld.tsx +9 -0
  650. package/src/test/setup-browser.ts +3 -0
  651. package/src/test/setup-jsdom.ts +6 -0
  652. package/src/test/setup.ts +9 -0
  653. package/src/utils/AbacUtil/AbacUtil.ts +21 -0
  654. package/src/utils/AbacUtil/index.ts +1 -0
  655. package/src/utils/AxiosUtil/AxiosUtil.test.ts +66 -0
  656. package/src/utils/AxiosUtil/AxiosUtil.ts +47 -0
  657. package/src/utils/AxiosUtil/index.ts +1 -0
  658. package/src/utils/CaseSelectionUtil/CaseSelectionUtil.ts +32 -0
  659. package/src/utils/CaseSelectionUtil/index.ts +1 -0
  660. package/src/utils/CaseSetUtil/CaseSetUtil.ts +13 -0
  661. package/src/utils/CaseSetUtil/index.ts +1 -0
  662. package/src/utils/CaseTypeUtil/CaseTypeUtil.ts +178 -0
  663. package/src/utils/CaseTypeUtil/index.ts +1 -0
  664. package/src/utils/CaseUtil/CaseUtil.ts +401 -0
  665. package/src/utils/CaseUtil/index.ts +1 -0
  666. package/src/utils/DashboardUtil/DashboardUtil.ts +46 -0
  667. package/src/utils/DashboardUtil/index.ts +1 -0
  668. package/src/utils/DataHookUtil/DataHookUtil.ts +137 -0
  669. package/src/utils/DataHookUtil/index.ts +1 -0
  670. package/src/utils/DataSetUtil/DataSetUtil.ts +70 -0
  671. package/src/utils/DataSetUtil/index.ts +1 -0
  672. package/src/utils/DataUtil/DataUtil.ts +84 -0
  673. package/src/utils/DataUtil/index.ts +1 -0
  674. package/src/utils/DownloadUtil/DownloadUtil.ts +237 -0
  675. package/src/utils/DownloadUtil/index.ts +1 -0
  676. package/src/utils/EffectiveRightsUtil/EffectiveRightsUtil.ts +142 -0
  677. package/src/utils/EffectiveRightsUtil/index.ts +1 -0
  678. package/src/utils/EpiCurveUtil/EpiCurveUtil.ts +180 -0
  679. package/src/utils/EpiCurveUtil/index.ts +1 -0
  680. package/src/utils/EpiFilterUtil/EpiFilterUtil.ts +232 -0
  681. package/src/utils/EpiFilterUtil/index.ts +1 -0
  682. package/src/utils/EpiLineListUtil/EpiLineListUtil.ts +15 -0
  683. package/src/utils/EpiLineListUtil/index.ts +1 -0
  684. package/src/utils/EpiMapUtil/EpiMapUtil.test.ts +54 -0
  685. package/src/utils/EpiMapUtil/EpiMapUtil.ts +80 -0
  686. package/src/utils/EpiMapUtil/index.ts +1 -0
  687. package/src/utils/EpiTreeUtil/EpiTreeUtil.test.ts +2467 -0
  688. package/src/utils/EpiTreeUtil/EpiTreeUtil.ts +1055 -0
  689. package/src/utils/EpiTreeUtil/index.ts +1 -0
  690. package/src/utils/EpiUploadUtil/EpiUploadUtil.ts +857 -0
  691. package/src/utils/EpiUploadUtil/index.ts +1 -0
  692. package/src/utils/FileUtil/FileUtil.ts +22 -0
  693. package/src/utils/FileUtil/index.ts +1 -0
  694. package/src/utils/FormUtil/FormUtil.test.ts +36 -0
  695. package/src/utils/FormUtil/FormUtil.ts +63 -0
  696. package/src/utils/FormUtil/index.ts +1 -0
  697. package/src/utils/MenuDataUtil/MenuDataUtil.ts +54 -0
  698. package/src/utils/MenuDataUtil/index.ts +1 -0
  699. package/src/utils/NewickUtil/NewickUtil.test.ts +187 -0
  700. package/src/utils/NewickUtil/NewickUtil.ts +69 -0
  701. package/src/utils/NewickUtil/index.ts +1 -0
  702. package/src/utils/NotificationUtil/NotificationUtil.ts +28 -0
  703. package/src/utils/NotificationUtil/index.ts +1 -0
  704. package/src/utils/NumberUtil/NumberUtil.test.ts +113 -0
  705. package/src/utils/NumberUtil/NumberUtil.ts +67 -0
  706. package/src/utils/NumberUtil/index.ts +1 -0
  707. package/src/utils/ObjectUtil/ObjectUtil.test.ts +43 -0
  708. package/src/utils/ObjectUtil/ObjectUtil.ts +28 -0
  709. package/src/utils/ObjectUtil/index.ts +1 -0
  710. package/src/utils/OutageUtil/OutageUtil.test.ts +108 -0
  711. package/src/utils/OutageUtil/OutageUtil.ts +81 -0
  712. package/src/utils/OutageUtil/index.ts +1 -0
  713. package/src/utils/QueryUtil/QueryUtil.test.ts +42 -0
  714. package/src/utils/QueryUtil/QueryUtil.ts +105 -0
  715. package/src/utils/QueryUtil/index.ts +1 -0
  716. package/src/utils/StringUtil/StringUtil.test.ts +61 -0
  717. package/src/utils/StringUtil/StringUtil.ts +91 -0
  718. package/src/utils/StringUtil/index.ts +1 -0
  719. package/src/utils/TableUtil/TableUtil.ts +665 -0
  720. package/src/utils/TableUtil/index.ts +1 -0
  721. package/src/utils/TestIdUtil/TestIdUtil.test.ts +12 -0
  722. package/src/utils/TestIdUtil/TestIdUtil.ts +13 -0
  723. package/src/utils/TestIdUtil/index.ts +1 -0
  724. package/src/utils/TimeUtil/TimeUtil.ts +38 -0
  725. package/src/utils/TimeUtil/index.ts +1 -0
  726. package/src/utils/UserManagerUtil/UserManagerUtil.test.ts +41 -0
  727. package/src/utils/UserManagerUtil/UserManagerUtil.ts +39 -0
  728. package/src/utils/UserManagerUtil/index.ts +1 -0
  729. package/src/utils/ValidationUtil/ValidationUtil.test.ts +61 -0
  730. package/src/utils/ValidationUtil/ValidationUtil.ts +56 -0
  731. package/src/utils/ValidationUtil/index.ts +1 -0
  732. package/dist/environment.d.ts +0 -54
  733. package/dist/index.d.ts +0 -16074
  734. /package/{dist → src/@types}/mui.d.ts +0 -0
  735. /package/{dist → src/@types}/ui.d.ts +0 -0
  736. /package/{dist → src/@types}/vite-env.d.ts +0 -0
  737. /package/{dist → src/@types}/yup.d.ts +0 -0
  738. /package/{dist → src}/locale/en.json +0 -0
  739. /package/{dist → src}/locale/nl.json +0 -0
@@ -0,0 +1,906 @@
1
+ import {
2
+ Alert,
3
+ AlertTitle,
4
+ Box,
5
+ Button,
6
+ ListItemIcon,
7
+ ListItemText,
8
+ MenuItem,
9
+ useTheme,
10
+ } from '@mui/material';
11
+ import ClearIcon from '@mui/icons-material/Clear';
12
+ import ArrowUpwardIcon from '@mui/icons-material/ArrowUpward';
13
+ import FilterAltIcon from '@mui/icons-material/FilterAlt';
14
+ import InfoOutlinedIcon from '@mui/icons-material/InfoOutlined';
15
+ import LinkIcon from '@mui/icons-material/Link';
16
+ import type {
17
+ ReactElement,
18
+ Ref,
19
+ } from 'react';
20
+ import {
21
+ useCallback,
22
+ useContext,
23
+ useEffect,
24
+ useImperativeHandle,
25
+ useMemo,
26
+ useRef,
27
+ useState,
28
+ } from 'react';
29
+ import { useTranslation } from 'react-i18next';
30
+ import { useStore } from 'zustand';
31
+ import { useDebouncedCallback } from 'use-debounce';
32
+
33
+ import type { EpiContextMenuConfigWithPosition } from '../EpiContextMenu';
34
+ import { EpiContextMenu } from '../EpiContextMenu';
35
+ import { EpiWidgetUnavailable } from '../EpiWidgetUnavailable';
36
+ import { EpiTreeDescription } from '../EpiTreeDescription';
37
+ import type { RetrievePhylogeneticTreeRequestBody } from '../../../api';
38
+ import { CaseApi } from '../../../api';
39
+ import { TreeFilter } from '../../../classes/filters/TreeFilter';
40
+ import { ConfigManager } from '../../../classes/managers/ConfigManager';
41
+ import { DevicePixelRatioManager } from '../../../classes/managers/DevicePixelRatioManager';
42
+ import { EpiEventBusManager } from '../../../classes/managers/EpiEventBusManager';
43
+ import { EpiHighlightingManager } from '../../../classes/managers/EpiHighlightingManager';
44
+ import { Subject } from '../../../classes/Subject';
45
+ import { useDimensions } from '../../../hooks/useDimensions';
46
+ import { useScrollbarSize } from '../../../hooks/useScrollbarSize';
47
+ import { useSubscribable } from '../../../hooks/useSubscribable';
48
+ import type {
49
+ EpiLinkedScrollSubjectValue,
50
+ TreeConfiguration,
51
+ Highlighting,
52
+ } from '../../../models/epi';
53
+ import { EPI_ZONE } from '../../../models/epi';
54
+ import type { MenuItemData } from '../../../models/nestedMenu';
55
+ import type {
56
+ TreeAssembly,
57
+ TreePathProperties,
58
+ } from '../../../models/tree';
59
+ import { EpiDashboardStoreContext } from '../../../stores/epiDashboardStore';
60
+ import { userProfileStore } from '../../../stores/userProfileStore';
61
+ import { SELECTION_FILTER_GROUP } from '../../../utils/CaseTypeUtil';
62
+ import { QueryUtil } from '../../../utils/QueryUtil';
63
+ import { Spinner } from '../../ui/Spinner';
64
+ import { EpiWidget } from '../EpiWidget';
65
+ import { DownloadUtil } from '../../../utils/DownloadUtil';
66
+ import { useQueryMemo } from '../../../hooks/useQueryMemo';
67
+ import { EpiTreeUtil } from '../../../utils/EpiTreeUtil';
68
+
69
+ type ZoomInMenuItemConfig = {
70
+ caseIds?: string[];
71
+ rootId?: string;
72
+ };
73
+
74
+ type EpiTreeProps = {
75
+ readonly linkedScrollSubject: Subject<EpiLinkedScrollSubjectValue>;
76
+ readonly ref: Ref<EpiTreeRef>;
77
+ readonly itemHeight: number;
78
+ };
79
+
80
+ export interface EpiTreeRef {
81
+ link: () => void;
82
+ }
83
+
84
+ export const EpiTree = ({ linkedScrollSubject, ref, itemHeight }: EpiTreeProps) => {
85
+ const theme = useTheme();
86
+ const { t } = useTranslation();
87
+ const scrollbarSize = useScrollbarSize();
88
+ const scrollContainerRef = useRef<HTMLDivElement>(null);
89
+ const containerRef = useRef<HTMLDivElement>(null);
90
+ const { dimensions: { width, height } } = useDimensions(containerRef);
91
+ const [treeCanvas, setTreeCanvas] = useState<HTMLCanvasElement>();
92
+ const [headerCanvas, setHeaderCanvas] = useState<HTMLCanvasElement>();
93
+ const highlightingManager = useMemo(() => EpiHighlightingManager.instance, []);
94
+ const canvasScrollSubject = useMemo<Subject<{ x: number; y: number }>>(() => new Subject({ x: 0, y: 0 }), []);
95
+ const epiDashboardStore = useContext(EpiDashboardStoreContext);
96
+ const setPhylogeneticTreeResponse = useStore(epiDashboardStore, (state) => state.setPhylogeneticTreeResponse);
97
+ const baseData = useStore(epiDashboardStore, (state) => state.baseData);
98
+ const filteredCases = useStore(epiDashboardStore, (state) => state.filteredData[SELECTION_FILTER_GROUP]);
99
+ const setSorting = useStore(epiDashboardStore, (state) => state.setSorting);
100
+ const tree = useStore(epiDashboardStore, (state) => state.tree);
101
+ const sortByField = useStore(epiDashboardStore, (state) => state.sortByField);
102
+ const stratification = useStore(epiDashboardStore, (state) => state.stratification);
103
+ const completeCaseType = useStore(epiDashboardStore, (state) => state.completeCaseType);
104
+ const hasActiveTreeFilter = useStore(epiDashboardStore, (state) => !state.filters.find(filter => filter instanceof TreeFilter).isInitialFilterValue());
105
+ const addTreeFilter = useStore(epiDashboardStore, (state) => state.addTreeFilter);
106
+ const treeFilterStepOut = useStore(epiDashboardStore, (state) => state.treeFilterStepOut);
107
+ const updateEpiTreeWidgetData = useStore(epiDashboardStore, (state) => state.updateEpiTreeWidgetData);
108
+ const removeTreeFilter = useStore(epiDashboardStore, (state) => state.removeTreeFilter);
109
+ const isCaseDataLoading = useStore(epiDashboardStore, (state) => state.isDataLoading);
110
+ const newick = useStore(epiDashboardStore, (state) => state.newick);
111
+ const resetTreeAddresses = useStore(epiDashboardStore, (state) => state.resetTreeAddresses);
112
+ const isShowDistancesEnabled = useStore(userProfileStore, (state) => state.epiDashboardTreeSettings.isShowDistancesEnabled);
113
+ const [epiContextMenuConfig, setEpiContextMenuConfig] = useState<EpiContextMenuConfigWithPosition | null>(null);
114
+ const [zoomInMenuItemConfig, setZoomInMenuItemConfig] = useState<ZoomInMenuItemConfig>(null);
115
+ const [extraLeafInfoId, setExtraLeafInfoId] = useState<string>(null);
116
+ const [treeConfiguration, setTreeConfiguration] = useState<TreeConfiguration>(epiDashboardStore.getState().epiTreeWidgetData.treeConfiguration);
117
+ const [treeAssembly, setTreeAssembly] = useState<TreeAssembly>(null);
118
+ const [verticalScrollPosition, setVerticalScrollPosition] = useState<number>(!isNaN(epiDashboardStore.getState().epiTreeWidgetData.verticalScrollPosition) ? epiDashboardStore.getState().epiTreeWidgetData.verticalScrollPosition : 0);
119
+ const [horizontalScrollPosition, setHorizontalScrollPosition] = useState<number>(!isNaN(epiDashboardStore.getState().epiTreeWidgetData.horizontalScrollPosition) ? epiDashboardStore.getState().epiTreeWidgetData.horizontalScrollPosition : 0);
120
+ const [zoomLevel, setZoomLevel] = useState<number>(!isNaN(epiDashboardStore.getState().epiTreeWidgetData.zoomLevel) ? epiDashboardStore.getState().epiTreeWidgetData.zoomLevel : 1);
121
+ const [devicePixelRatio, setDevicePixelRatio] = useState<number>(DevicePixelRatioManager.instance.data);
122
+ const [isLinked, setIsLinked] = useState(true);
123
+ const internalHighlightingSubject = useMemo(() => new Subject<Highlighting>({
124
+ caseIds: [],
125
+ origin: null,
126
+ }), []);
127
+
128
+ const treeConfigurations = useMemo(() => EpiTreeUtil.getTreeConfigurations(completeCaseType), [completeCaseType]);
129
+
130
+ const treeCanvasAriaLabel = useMemo(() => {
131
+ if (!treeConfiguration) {
132
+ return t('Phylogenetic tree visualization');
133
+ }
134
+
135
+ const label = treeConfiguration.col?.label || t('data');
136
+ const geneticDistanceProtocol = treeConfiguration.geneticDistanceProtocol?.name || t('unknown protocol');
137
+ const treeAlgorithm = treeConfiguration.treeAlgorithm?.name || t('unknown algorithm');
138
+
139
+ return t('Figure of a phylogenetic tree belonging to {{label}}. Generated using {{geneticDistanceProtocol}} and {{treeAlgorithm}}.', {
140
+ label,
141
+ geneticDistanceProtocol,
142
+ treeAlgorithm,
143
+ });
144
+ }, [treeConfiguration, t]);
145
+
146
+ useEffect(() => {
147
+ const unsubscribeHighlightingManager = highlightingManager.subscribe((highlighting) => {
148
+ if (highlighting.origin === EPI_ZONE.TREE) {
149
+ return;
150
+ }
151
+ internalHighlightingSubject.next(highlighting);
152
+ });
153
+ const unsubscribeInternalHighlightingSubject = internalHighlightingSubject.subscribe((highlighting) => {
154
+ if (highlighting.origin !== EPI_ZONE.TREE) {
155
+ return;
156
+ }
157
+ highlightingManager.highlight(highlighting);
158
+ });
159
+
160
+ return () => {
161
+ unsubscribeHighlightingManager();
162
+ unsubscribeInternalHighlightingSubject();
163
+ };
164
+ }, [highlightingManager, internalHighlightingSubject]);
165
+
166
+ useEffect(() => {
167
+ if (isLinked && zoomLevel !== 1) {
168
+ setIsLinked(false);
169
+ }
170
+ }, [isLinked, zoomLevel]);
171
+
172
+ const caseIds = useMemo(() => filteredCases.map(c => c.id).sort(), [filteredCases]);
173
+
174
+ useEffect(() => {
175
+ resetTreeAddresses();
176
+ }, [caseIds, resetTreeAddresses]);
177
+
178
+ const hasEnoughSequencesToShowTree = useMemo(() => caseIds.length >= 2 && caseIds.every(x => !!x), [caseIds]);
179
+ const hasToManyResultsToShowTree = useMemo(() => caseIds.length > 0 && completeCaseType.read_max_tree_size > 0 && caseIds.length > completeCaseType.read_max_tree_size, [caseIds, completeCaseType.read_max_tree_size]);
180
+
181
+ const retrievePhylogeneticTreeRequestBody = useMemo<RetrievePhylogeneticTreeRequestBody>(() => ({
182
+ case_ids: caseIds,
183
+ genetic_distance_col_id: treeConfiguration?.col.id,
184
+ tree_algorithm_code: treeConfiguration?.treeAlgorithm.code,
185
+ case_type_id: completeCaseType.id,
186
+ }), [caseIds, completeCaseType.id, treeConfiguration?.col.id, treeConfiguration?.treeAlgorithm.code]);
187
+
188
+ const { isLoading: isTreeLoading, error: treeError, data: treeData } = useQueryMemo({
189
+ queryKey: QueryUtil.getRetrievePhylogeneticTreeKey(retrievePhylogeneticTreeRequestBody),
190
+ queryFn: async ({ signal }) => {
191
+ const response = await CaseApi.instance.retrievePhylogeneticTree(retrievePhylogeneticTreeRequestBody, { signal });
192
+ return response.data;
193
+ },
194
+ enabled: hasEnoughSequencesToShowTree && !!treeConfiguration && !hasToManyResultsToShowTree,
195
+ retry: false,
196
+ staleTime: Infinity,
197
+ });
198
+
199
+ const isLoading = !!treeConfiguration && (isCaseDataLoading || (hasEnoughSequencesToShowTree && isTreeLoading));
200
+ const isTreeUnavailable = hasToManyResultsToShowTree || !isCaseDataLoading && ((!isLoading && !!treeError) || !hasEnoughSequencesToShowTree || tree?.maxBranchLength?.toNumber() === 0 || tree?.size === 0 || !treeConfiguration || (!isLoading && tree === null));
201
+ const shouldShowTree = !hasToManyResultsToShowTree && !!treeConfiguration && !isCaseDataLoading && !treeError && !isTreeLoading && width > 0 && tree?.size > 0 && hasEnoughSequencesToShowTree;
202
+
203
+ const treeCanvasWidth = width;
204
+ const treeCanvasHeight = height - ConfigManager.instance.config.epiTree.HEADER_HEIGHT;
205
+
206
+ const treeWidthMinusPadding = treeCanvasWidth - (2 * ConfigManager.instance.config.epiTree.TREE_PADDING);
207
+ const pixelToGeneticDistanceRatio = tree?.maxBranchLength ? treeWidthMinusPadding / tree.maxBranchLength.toNumber() : null;
208
+ // Note: There is some magic here, because of the position: sticky for the table header
209
+ const treeHeight = tree?.size ? (tree.size * itemHeight) + scrollbarSize : itemHeight;
210
+
211
+ useEffect(() => {
212
+ if (treeData) {
213
+ setPhylogeneticTreeResponse(treeData);
214
+ }
215
+ }, [treeData, setPhylogeneticTreeResponse]);
216
+
217
+ useEffect(() => {
218
+ const unsubscribe = canvasScrollSubject.subscribe((data) => {
219
+ if (!isNaN(data.x)) {
220
+ setHorizontalScrollPosition(data.x);
221
+ }
222
+ if (!isNaN(data.y)) {
223
+ setVerticalScrollPosition(data.y);
224
+ }
225
+ });
226
+
227
+ return () => {
228
+ unsubscribe();
229
+ };
230
+ }, [canvasScrollSubject]);
231
+
232
+ useEffect(() => {
233
+ if (!treeConfigurations?.length) {
234
+ setTreeConfiguration(null);
235
+ }
236
+ const newConfig = epiDashboardStore.getState().epiTreeWidgetData.treeConfiguration || treeConfigurations[0];
237
+
238
+ setTreeConfiguration(newConfig);
239
+ updateEpiTreeWidgetData({
240
+ treeConfiguration: newConfig,
241
+ });
242
+ }, [epiDashboardStore, treeConfigurations, updateEpiTreeWidgetData]);
243
+
244
+ const updateLinkedScrollSubjectDebounced = useDebouncedCallback((position: number) => {
245
+ linkedScrollSubject.next({
246
+ position: position / devicePixelRatio,
247
+ origin: scrollContainerRef.current,
248
+ });
249
+ }, ConfigManager.instance.config.epiTree.LINKED_SCROLL_DEBOUNCE_DELAY_MS, { leading: true, trailing: true });
250
+
251
+ const updateScrollPosition = useCallback((positionX: number, positionY: number, internalZoomLevel: number) => {
252
+ const { newPositionX, newPositionY } = EpiTreeUtil.getSanitizedScrollPosition({
253
+ devicePixelRatio,
254
+ internalZoomLevel,
255
+ isLinked,
256
+ positionX,
257
+ positionY,
258
+ treeCanvasHeight,
259
+ treeCanvasWidth,
260
+ treeHeight,
261
+ });
262
+
263
+ canvasScrollSubject.next({
264
+ x: newPositionX,
265
+ y: newPositionY,
266
+ });
267
+
268
+ if (isLinked && internalZoomLevel === 1) {
269
+ updateLinkedScrollSubjectDebounced(newPositionY);
270
+ }
271
+ }, [devicePixelRatio, isLinked, treeCanvasWidth, canvasScrollSubject, treeHeight, treeCanvasHeight, updateLinkedScrollSubjectDebounced]);
272
+
273
+ useEffect(() => {
274
+ if (sortByField && isLinked) {
275
+ setIsLinked(false);
276
+ setZoomLevel(ConfigManager.instance.config.epiTree.INITIAL_UNLINKED_ZOOM_LEVEL);
277
+ updateScrollPosition(0, 0, 0);
278
+ }
279
+ }, [isLinked, sortByField, updateScrollPosition]);
280
+
281
+ const devicePixelRatioManagerCallback = useCallback((newDevicePixelRation: number, previousDevicePixelRatio: number) => {
282
+ canvasScrollSubject.next({
283
+ x: (canvasScrollSubject.data.x / previousDevicePixelRatio) * newDevicePixelRation,
284
+ y: (canvasScrollSubject.data.y / previousDevicePixelRatio) * newDevicePixelRation,
285
+ });
286
+ setDevicePixelRatio(newDevicePixelRation);
287
+ }, [canvasScrollSubject]);
288
+
289
+ useSubscribable(DevicePixelRatioManager.instance, {
290
+ callback: devicePixelRatioManagerCallback,
291
+ });
292
+
293
+ const updateEpiTreeWidgetDataDebounced = useDebouncedCallback(() => {
294
+ updateEpiTreeWidgetData({
295
+ zoomLevel,
296
+ verticalScrollPosition,
297
+ horizontalScrollPosition,
298
+ });
299
+ }, 500);
300
+
301
+ useEffect(() => {
302
+ updateEpiTreeWidgetDataDebounced();
303
+ }, [updateEpiTreeWidgetData, zoomLevel, verticalScrollPosition, horizontalScrollPosition, updateEpiTreeWidgetDataDebounced]);
304
+
305
+ const tickerMarkScale = useMemo(() => {
306
+ return EpiTreeUtil.getTickMarkScale({
307
+ treeWidthMinusPadding,
308
+ geneticTreeWidth: tree?.maxBranchLength,
309
+ minGeneticScaleUnit: Math.min(EpiTreeUtil.getMinGeneticScaleUnit(tree), treeConfiguration?.geneticDistanceProtocol?.min_scale_unit ?? Infinity),
310
+ // minGeneticScaleUnit: treeConfiguration?.geneticDistanceProtocol?.min_scale_unit,
311
+ zoomLevel,
312
+ });
313
+ }, [treeWidthMinusPadding, tree, treeConfiguration?.geneticDistanceProtocol?.min_scale_unit, zoomLevel]);
314
+
315
+ const onEpiContextMenuClose = useCallback(() => {
316
+ setEpiContextMenuConfig(null);
317
+ setZoomInMenuItemConfig(null);
318
+ }, []);
319
+
320
+ const linkLineListToTree = useCallback(() => {
321
+ const perform = async () => {
322
+ const newScrollPosition = EpiTreeUtil.getScrollPositionFromTreeVisibility({
323
+ treeHeight,
324
+ treeSize: tree?.size,
325
+ treeCanvasHeight,
326
+ verticalScrollPosition,
327
+ zoomLevel,
328
+ itemHeight,
329
+ });
330
+
331
+ await setSorting(null, null);
332
+ setZoomLevel(1);
333
+ setIsLinked(true);
334
+ updateScrollPosition(0, newScrollPosition, 1);
335
+ linkedScrollSubject.next({
336
+ position: newScrollPosition,
337
+ origin: scrollContainerRef.current,
338
+ });
339
+ };
340
+
341
+ // eslint-disable-next-line @typescript-eslint/no-floating-promises
342
+ perform();
343
+
344
+ }, [linkedScrollSubject, setSorting, tree?.size, treeCanvasHeight, treeHeight, verticalScrollPosition, updateScrollPosition, zoomLevel, itemHeight]);
345
+
346
+ const onLinkButtonClick = useCallback(() => {
347
+ linkLineListToTree();
348
+
349
+ }, [linkLineListToTree]);
350
+
351
+ const onOpenFiltersButtonClick = useCallback(() => {
352
+ EpiEventBusManager.instance.emit('openFiltersMenu');
353
+ }, []);
354
+
355
+ const resetZoomLevelAndScrollPosition = useCallback(() => {
356
+ setZoomLevel(1);
357
+ setIsLinked(true);
358
+ updateScrollPosition(0, 0, 1);
359
+ }, [updateScrollPosition]);
360
+
361
+ const onAddTreeFilterMenuItemClick = useCallback(async (onMenuClose: () => void) => {
362
+ await addTreeFilter(zoomInMenuItemConfig.rootId);
363
+ resetZoomLevelAndScrollPosition();
364
+ onMenuClose();
365
+ }, [addTreeFilter, zoomInMenuItemConfig?.rootId, resetZoomLevelAndScrollPosition]);
366
+
367
+ const onRemoveTreeFilterButtonClick = useCallback(async () => {
368
+ await removeTreeFilter();
369
+ resetZoomLevelAndScrollPosition();
370
+ }, [removeTreeFilter, resetZoomLevelAndScrollPosition]);
371
+
372
+ const onTreeFilterStepOutButtonClick = useCallback(async () => {
373
+ await treeFilterStepOut();
374
+ resetZoomLevelAndScrollPosition();
375
+ }, [resetZoomLevelAndScrollPosition, treeFilterStepOut]);
376
+
377
+ const getPathPropertiesFromCanvas = useCallback((canvas: HTMLCanvasElement, event: MouseEvent): TreePathProperties => {
378
+ return EpiTreeUtil.getPathPropertiesFromCanvas({
379
+ canvas,
380
+ event,
381
+ treeAssembly,
382
+ devicePixelRatio,
383
+ });
384
+ }, [treeAssembly, devicePixelRatio]);
385
+
386
+ useEffect(() => {
387
+ const unsubscribe = linkedScrollSubject.subscribe((data) => {
388
+ if (data.origin === scrollContainerRef.current) {
389
+ return;
390
+ }
391
+ if (isLinked && zoomLevel === 1) {
392
+ canvasScrollSubject.next({
393
+ x: canvasScrollSubject.data.x * devicePixelRatio,
394
+ y: data.position * devicePixelRatio,
395
+ });
396
+ }
397
+ });
398
+
399
+ return () => {
400
+ unsubscribe();
401
+ };
402
+ }, [linkedScrollSubject, canvasScrollSubject, updateScrollPosition, devicePixelRatio, isLinked, zoomLevel]);
403
+
404
+ // Setup canvas
405
+ useEffect(() => {
406
+ setTreeAssembly(EpiTreeUtil.assembleTree({ rootNode: tree, treeCanvasWidth, pixelToGeneticDistanceRatio, itemHeight }));
407
+ }, [pixelToGeneticDistanceRatio, tree, treeCanvasWidth, itemHeight]);
408
+
409
+ useEffect(() => {
410
+ if (!treeCanvas || !treeAssembly || !tree) {
411
+ return;
412
+ }
413
+
414
+ EpiTreeUtil.drawTreeCanvas({ canvas: treeCanvas, theme, geneticTreeWidth: tree?.maxBranchLength, treeAssembly, stratification, zoomLevel, isLinked, horizontalScrollPosition, verticalScrollPosition, treeCanvasWidth, treeCanvasHeight, pixelToGeneticDistanceRatio, tickerMarkScale, shouldShowDistances: isShowDistancesEnabled, devicePixelRatio });
415
+ let animationFrameId: number;
416
+ const unsubscribe = internalHighlightingSubject.subscribe((highlighting) => {
417
+ cancelAnimationFrame(animationFrameId);
418
+ animationFrameId = requestAnimationFrame(() => {
419
+ EpiTreeUtil.drawTreeCanvas({ canvas: treeCanvas, theme, geneticTreeWidth: tree?.maxBranchLength, treeAssembly, stratification, zoomLevel, isLinked, horizontalScrollPosition, verticalScrollPosition, treeCanvasWidth, treeCanvasHeight, pixelToGeneticDistanceRatio, tickerMarkScale, highlightedNodeNames: highlighting.caseIds, shouldShowDistances: isShowDistancesEnabled, devicePixelRatio });
420
+ });
421
+ });
422
+ return () => {
423
+ unsubscribe();
424
+ cancelAnimationFrame(animationFrameId);
425
+ };
426
+ }, [treeCanvasHeight, treeCanvas, internalHighlightingSubject, pixelToGeneticDistanceRatio, stratification, theme, tickerMarkScale, treeAssembly, treeCanvasWidth, horizontalScrollPosition, verticalScrollPosition, width, zoomLevel, isLinked, isShowDistancesEnabled, devicePixelRatio, tree?.maxBranchLength, tree]);
427
+
428
+ // Setup canvas event listeners (note: must be in a separate useEffect to prevent render loop)
429
+ useEffect(() => {
430
+ if (!treeCanvas) {
431
+ return;
432
+ }
433
+
434
+ let pos = {
435
+ x: 0,
436
+ y: 0,
437
+ currentX: 0,
438
+ currentY: 0,
439
+ };
440
+ let followMouse = false;
441
+
442
+ const onMouseDown = (event: MouseEvent) => {
443
+ // store the current mouse position (to be used on mouse move for scrolling)
444
+ pos = {
445
+ x: event.clientX,
446
+ y: event.clientY,
447
+ currentX: canvasScrollSubject.data.x,
448
+ currentY: canvasScrollSubject.data.y,
449
+ };
450
+ followMouse = true;
451
+ };
452
+
453
+ const onMouseMove = (event: MouseEvent) => {
454
+ if (followMouse) {
455
+ treeCanvas.style.cursor = 'move';
456
+
457
+ // update the scroll position based on how far the mouse has moved
458
+ // how far the mouse has moved:
459
+ const deltaX = event.clientX - pos.x;
460
+ const deltaY = event.clientY - pos.y;
461
+ // the new scroll position:
462
+ const scrollPositionX = pos.currentX - deltaX;
463
+ const scrollPositionY = pos.currentY - deltaY;
464
+
465
+ let sanitizedScrollPositionX = scrollPositionX;
466
+ if (zoomLevel === 1 && Math.abs(deltaX) < ConfigManager.instance.config.epiTree.PANNING_THRESHOLD && pos.currentX === 0) {
467
+ sanitizedScrollPositionX = 0;
468
+ }
469
+
470
+ updateScrollPosition(sanitizedScrollPositionX, scrollPositionY, zoomLevel);
471
+ return;
472
+ }
473
+
474
+ const pathProperties = getPathPropertiesFromCanvas(treeCanvas, event);
475
+ if (pathProperties) {
476
+ treeCanvas.style.cursor = 'pointer';
477
+ internalHighlightingSubject.next({
478
+ caseIds: pathProperties.subTreeLeaveNames,
479
+ origin: EPI_ZONE.TREE,
480
+ });
481
+ } else {
482
+ treeCanvas.style.cursor = 'default';
483
+ // remove highlighting (only there is highlighting to prevent a render loop)
484
+ if (internalHighlightingSubject.data?.caseIds?.length) {
485
+ internalHighlightingSubject.next({
486
+ caseIds: [],
487
+ origin: EPI_ZONE.TREE,
488
+ });
489
+ }
490
+ }
491
+ };
492
+
493
+ const onMouseUp = (event: MouseEvent) => {
494
+ // stop following the mouse for scrolling
495
+ followMouse = false;
496
+
497
+ // Handle the click when the user clicked a path in the canvas
498
+ const pathProperties = getPathPropertiesFromCanvas(treeCanvas, event);
499
+ if (pathProperties?.treeNode) {
500
+ const { treeNode } = pathProperties;
501
+ setEpiContextMenuConfig({
502
+ position: {
503
+ left: event.clientX,
504
+ top: event.clientY,
505
+ },
506
+ caseIds: treeNode.subTreeLeaveNames ? treeNode.subTreeLeaveNames : [treeNode.name],
507
+ mouseEvent: event,
508
+ });
509
+ if (treeNode.subTreeNames.length && treeNode.name && treeNode.maxBranchLength.toNumber()) {
510
+ setZoomInMenuItemConfig({
511
+ caseIds: treeNode.subTreeLeaveNames,
512
+ rootId: treeNode.name,
513
+ });
514
+ }
515
+ if (treeNode.name && !treeNode.subTreeNames.length) {
516
+ setExtraLeafInfoId(treeNode.name);
517
+ }
518
+ } else if (pathProperties?.subTreeLeaveNames?.length) {
519
+ setEpiContextMenuConfig({
520
+ position: {
521
+ left: event.clientX,
522
+ top: event.clientY,
523
+ },
524
+ caseIds: pathProperties.subTreeLeaveNames,
525
+ mouseEvent: event,
526
+ });
527
+ }
528
+ };
529
+
530
+ const onMouseWheel = (event: WheelEvent) => {
531
+ event.preventDefault();
532
+
533
+ if (event.shiftKey) {
534
+ updateScrollPosition(canvasScrollSubject.data.x + (event.deltaX || event.deltaY), canvasScrollSubject.data.y, zoomLevel);
535
+ return;
536
+ }
537
+ if (event.metaKey || event.ctrlKey) {
538
+ updateScrollPosition(canvasScrollSubject.data.x, canvasScrollSubject.data.y + (event.deltaX || event.deltaY), zoomLevel);
539
+ return;
540
+ }
541
+
542
+ const { MAX_ZOOM_SPEED, MIN_ZOOM_SPEED, MAX_ZOOM_LEVEL, MIN_ZOOM_LEVEL } = ConfigManager.instance.config.epiTree;
543
+
544
+ const zoomSpeed = Math.min(MAX_ZOOM_SPEED, Math.max(MIN_ZOOM_SPEED, treeHeight / treeCanvasHeight * 0.2));
545
+ const newZoomLevel = Math.min(MAX_ZOOM_LEVEL, Math.max(MIN_ZOOM_LEVEL, zoomLevel + (event.deltaY > 0 ? zoomSpeed : -zoomSpeed)));
546
+
547
+ const newScrollPositionY = EpiTreeUtil.getNewScrollPositionForZoomLevel({
548
+ eventOffset: event.offsetY,
549
+ scrollPosition: canvasScrollSubject.data.y,
550
+ dimensionSize: treeHeight,
551
+ currentZoomLevel: zoomLevel,
552
+ newZoomLevel,
553
+ });
554
+ const newScrollPositionX = EpiTreeUtil.getNewScrollPositionForZoomLevel({
555
+ eventOffset: event.offsetX,
556
+ scrollPosition: canvasScrollSubject.data.x,
557
+ dimensionSize: treeCanvasWidth,
558
+ currentZoomLevel: zoomLevel,
559
+ newZoomLevel,
560
+ });
561
+
562
+ setZoomLevel(newZoomLevel);
563
+ if (newZoomLevel !== 1) {
564
+ updateScrollPosition(newScrollPositionX, newScrollPositionY, newZoomLevel);
565
+ } else {
566
+ updateScrollPosition(newScrollPositionX, linkedScrollSubject.data?.position ?? 0, 1);
567
+ }
568
+
569
+ if (newZoomLevel > MIN_ZOOM_LEVEL && newZoomLevel < MAX_ZOOM_LEVEL) {
570
+ // Disconnect the event listener and allow a render cycle to complete
571
+ // But only when zoomLevel is not at it's boundaries, otherwise the event listener will not be reconnected
572
+ treeCanvas.removeEventListener('wheel', onMouseWheel);
573
+ }
574
+ };
575
+
576
+ const onMouseOut = () => {
577
+ followMouse = false;
578
+
579
+ // reset highlighting (because mouse move event may not have triggered it correctly)
580
+ internalHighlightingSubject.next({
581
+ caseIds: [],
582
+ origin: EPI_ZONE.TREE,
583
+ });
584
+ };
585
+
586
+ treeCanvas.addEventListener('mousemove', onMouseMove);
587
+ treeCanvas.addEventListener('mousedown', onMouseDown);
588
+ treeCanvas.addEventListener('mouseup', onMouseUp);
589
+ treeCanvas.addEventListener('mouseout', onMouseOut);
590
+ treeCanvas.addEventListener('wheel', onMouseWheel);
591
+
592
+ return () => {
593
+ treeCanvas.removeEventListener('mousemove', onMouseMove);
594
+ treeCanvas.removeEventListener('mousedown', onMouseDown);
595
+ treeCanvas.removeEventListener('mouseup', onMouseUp);
596
+ treeCanvas.removeEventListener('mouseout', onMouseOut);
597
+ treeCanvas.removeEventListener('wheel', onMouseWheel);
598
+ };
599
+ // note: we don't want to re-run this effect when the treeSubject.data changes; it would disconnect the event handles and prevent the user from interacting with the tree
600
+ // eslint-disable-next-line react-hooks/exhaustive-deps
601
+ }, [treeCanvasHeight, treeCanvas, getPathPropertiesFromCanvas, internalHighlightingSubject, updateScrollPosition, scrollContainerRef, treeHeight, zoomLevel]);
602
+
603
+ // Setup header canvas
604
+ useEffect(() => {
605
+ if (!headerCanvas || !tree) {
606
+ return;
607
+ }
608
+ const canvas = headerCanvas;
609
+ const ctx = canvas.getContext('2d');
610
+ canvas.width = canvas.clientWidth * devicePixelRatio;
611
+ canvas.height = canvas.clientHeight * devicePixelRatio;
612
+
613
+ EpiTreeUtil.drawGuides({ canvas, geneticTreeWidth: tree?.maxBranchLength, tickerMarkScale, pixelToGeneticDistanceRatio, horizontalScrollPosition, paddingTop: ConfigManager.instance.config.epiTree.HEADER_HEIGHT * 0.7, paddingBottom: 0, zoomLevel, devicePixelRatio });
614
+ EpiTreeUtil.drawGuides({ canvas, geneticTreeWidth: tree?.maxBranchLength, tickerMarkScale, pixelToGeneticDistanceRatio, horizontalScrollPosition, paddingTop: 0, paddingBottom: ConfigManager.instance.config.epiTree.HEADER_HEIGHT * 0.7, zoomLevel, devicePixelRatio });
615
+ EpiTreeUtil.drawScale({ canvas, theme, tickerMarkScale, geneticTreeWidth: tree?.maxBranchLength, pixelToGeneticDistanceRatio, zoomLevel, devicePixelRatio, horizontalScrollPosition });
616
+
617
+ // Draw horizontal top divider
618
+ EpiTreeUtil.drawDivider({ canvas, y: 0, devicePixelRatio });
619
+ // Draw horizontal bottom divider
620
+ EpiTreeUtil.drawDivider({ canvas, y: ConfigManager.instance.config.epiTree.HEADER_HEIGHT - 1, devicePixelRatio });
621
+
622
+ ctx.translate(-0.5, -0.5);
623
+ }, [headerCanvas, pixelToGeneticDistanceRatio, theme, tickerMarkScale, zoomLevel, devicePixelRatio, horizontalScrollPosition, tree?.maxBranchLength, tree]);
624
+
625
+ const onShowDetailsSelectionMenuItemClick = useCallback((onMenuClose: () => void) => {
626
+ EpiEventBusManager.instance.emit('openCaseInfoDialog', {
627
+ caseId: baseData.find(c => c.id === extraLeafInfoId).id,
628
+ caseTypeId: completeCaseType.id,
629
+ });
630
+ onMenuClose();
631
+ }, [extraLeafInfoId, baseData, completeCaseType.id]);
632
+
633
+ const getEpiContextMenuExtraItems = useCallback((onMenuClose: () => void): ReactElement => {
634
+ if (zoomInMenuItemConfig) {
635
+ return (
636
+ <MenuItem
637
+ divider
638
+ // eslint-disable-next-line react/jsx-no-bind
639
+ onClick={async () => onAddTreeFilterMenuItemClick(onMenuClose)}
640
+ >
641
+ <ListItemIcon>
642
+ <FilterAltIcon fontSize={'small'} />
643
+ </ListItemIcon>
644
+ <ListItemText>
645
+ {t`Filter (show only this subtree)`}
646
+ </ListItemText>
647
+ </MenuItem>
648
+ );
649
+ }
650
+ if (extraLeafInfoId) {
651
+ return (
652
+ <MenuItem
653
+ divider
654
+ // eslint-disable-next-line react/jsx-no-bind
655
+ onClick={() => onShowDetailsSelectionMenuItemClick(onMenuClose)}
656
+ >
657
+ <ListItemIcon>
658
+ <InfoOutlinedIcon fontSize={'small'} />
659
+ </ListItemIcon>
660
+ <ListItemText>
661
+ {t`Show details`}
662
+ </ListItemText>
663
+ </MenuItem>
664
+ );
665
+ }
666
+ }, [extraLeafInfoId, onShowDetailsSelectionMenuItemClick, onAddTreeFilterMenuItemClick, t, zoomInMenuItemConfig]);
667
+
668
+ const titleMenu = useMemo<MenuItemData | string>(() => {
669
+ if (!treeConfigurations?.length) {
670
+ return t`Tree`;
671
+ }
672
+
673
+ const menu: MenuItemData = {
674
+ label: treeConfiguration ? t('Tree: {{algorithm}}', { algorithm: EpiTreeUtil.getTreeConfigurationLabel(treeConfiguration) }) : t`Tree`,
675
+ tooltip: treeConfiguration
676
+ ? (
677
+ <EpiTreeDescription
678
+ treeConfiguration={treeConfiguration}
679
+ />
680
+ )
681
+ : undefined,
682
+ disabled: !treeConfiguration,
683
+ items: treeConfigurations?.map<MenuItemData>(config => ({
684
+ label: EpiTreeUtil.getTreeConfigurationLabel(config),
685
+ callback: () => {
686
+ const perform = async () => {
687
+ await removeTreeFilter();
688
+ updateEpiTreeWidgetData({
689
+ treeConfiguration: config,
690
+ });
691
+ setTreeConfiguration(config);
692
+ };
693
+ // eslint-disable-next-line @typescript-eslint/no-floating-promises
694
+ perform();
695
+ },
696
+ active: !!treeConfiguration && treeConfiguration.computedId === config.computedId,
697
+ tooltip: (
698
+ <EpiTreeDescription
699
+ treeConfiguration={config}
700
+ />
701
+ ),
702
+ })) ?? [],
703
+ };
704
+
705
+ return menu;
706
+ }, [treeConfigurations, treeConfiguration, t, removeTreeFilter, updateEpiTreeWidgetData]);
707
+
708
+ const primaryMenu = useMemo<MenuItemData[]>(() => {
709
+ return [
710
+ {
711
+ disabled: !hasActiveTreeFilter,
712
+ label: t`Change tree filter to nearest ancestor`,
713
+ leftIcon: <ArrowUpwardIcon />,
714
+ callback: onTreeFilterStepOutButtonClick,
715
+ },
716
+ {
717
+ disabled: !hasActiveTreeFilter,
718
+ label: t`Remove tree filter`,
719
+ leftIcon: <ClearIcon />,
720
+ callback: onRemoveTreeFilterButtonClick,
721
+ },
722
+ {
723
+ label: t`Link and snap the Line List to the Tree (resets tree zoom level and Line List sorting)`,
724
+ leftIcon: (
725
+ <LinkIcon
726
+ sx={{
727
+ color: isLinked ? undefined : theme.palette.error.main,
728
+ }}
729
+ />
730
+ ),
731
+ callback: onLinkButtonClick,
732
+ },
733
+ ];
734
+ }, [hasActiveTreeFilter, t, onTreeFilterStepOutButtonClick, onRemoveTreeFilterButtonClick, isLinked, theme.palette.error.main, onLinkButtonClick]);
735
+
736
+ useEffect(() => {
737
+ const emitDownloadOptions = () => {
738
+ const baseName = t('Phylogenetic Tree - {{geneticDistanceProtocol}} - {{treeAlgorithm}}',
739
+ {
740
+ treeAlgorithm: treeConfiguration?.treeAlgorithm.name ?? '',
741
+ geneticDistanceProtocol: treeConfiguration?.geneticDistanceProtocol.name ?? '',
742
+ });
743
+
744
+ EpiEventBusManager.instance.emit('onDownloadOptionsChanged', {
745
+ zone: EPI_ZONE.TREE,
746
+ disabled: isTreeUnavailable,
747
+ zoneLabel: t`Phylogenetic Tree`,
748
+ items: [
749
+ {
750
+ label: t`Save as Newick`,
751
+ callback: () => DownloadUtil.downloadNewick(baseName, newick, completeCaseType, t),
752
+ },
753
+ {
754
+ label: t`Save as JPEG`,
755
+ callback: () => DownloadUtil.downloadCanvasImage(baseName, treeCanvas, 'jpeg', completeCaseType, t),
756
+ },
757
+ {
758
+ label: t`Save as PNG`,
759
+ callback: () => DownloadUtil.downloadCanvasImage(baseName, treeCanvas, 'png', completeCaseType, t),
760
+ },
761
+ ],
762
+ });
763
+ };
764
+
765
+
766
+ emitDownloadOptions();
767
+ const remove = EpiEventBusManager.instance.addEventListener('onDownloadOptionsRequested', emitDownloadOptions);
768
+
769
+ return () => {
770
+ EpiEventBusManager.instance.emit('onDownloadOptionsChanged', {
771
+ zone: EPI_ZONE.TREE,
772
+ items: null,
773
+ zoneLabel: t`Tree`,
774
+ });
775
+ remove();
776
+ };
777
+ }, [completeCaseType, isLinked, isTreeUnavailable, newick, t, treeCanvas, treeConfiguration?.geneticDistanceProtocol.name, treeConfiguration?.treeAlgorithm.name]);
778
+
779
+
780
+ const link = useCallback(() => {
781
+ // Link the tree to the current scroll position of the Line List
782
+ setIsLinked(true);
783
+ setZoomLevel(1);
784
+ updateScrollPosition(0, linkedScrollSubject.data?.position ?? 0, 1);
785
+ }, [linkedScrollSubject.data?.position, updateScrollPosition]);
786
+
787
+ useImperativeHandle(ref, () => ({
788
+ link,
789
+ }));
790
+
791
+ return (
792
+ <EpiWidget
793
+ expandDisabled={isTreeUnavailable}
794
+ primaryMenu={primaryMenu}
795
+ title={titleMenu}
796
+ zone={EPI_ZONE.TREE}
797
+ >
798
+ <Box
799
+ ref={containerRef}
800
+ sx={{
801
+ position: 'relative',
802
+ height: '100%',
803
+ width: '100%',
804
+ overflow: 'clip',
805
+ }}
806
+ >
807
+ {isTreeUnavailable && (
808
+ <>
809
+ {hasToManyResultsToShowTree && (
810
+ <Box>
811
+ <Alert severity={'warning'}>
812
+ <AlertTitle>
813
+ {t`Too many cases to display the phylogenetic tree`}
814
+ </AlertTitle>
815
+ <Box marginY={2}>
816
+ {t('The phylogenetic tree cannot be displayed because the number of cases ({{caseCount}}) exceeds the maximum allowed number of cases ({{maxSize}}) to display a phylogenetic tree. Refine your filters to reduce the number of results.', {
817
+ caseCount: caseIds.length,
818
+ maxSize: completeCaseType.read_max_tree_size,
819
+ })}
820
+ </Box>
821
+ <Button
822
+ color={'inherit'}
823
+ variant={'outlined'}
824
+ onClick={onOpenFiltersButtonClick}
825
+ >
826
+ {t`Refine filters`}
827
+ </Button>
828
+ </Alert>
829
+ </Box>
830
+ )}
831
+ {!hasToManyResultsToShowTree && (
832
+ <Box
833
+ sx={{
834
+ position: 'absolute',
835
+ zIndex: 1,
836
+ }}
837
+ >
838
+ <EpiWidgetUnavailable
839
+ epiZone={EPI_ZONE.TREE}
840
+ widgetName={t`phylogenetic tree`}
841
+ />
842
+ </Box>
843
+ )}
844
+ </>
845
+ )}
846
+ {(isLoading && !isTreeUnavailable) && (
847
+ <Spinner
848
+ label={t`Loading`}
849
+ takingLongerTimeoutMs={ConfigManager.instance.config.epiTree.TAKING_LONGER_TIMEOUT_MS}
850
+ />
851
+ )}
852
+ {!isTreeUnavailable && shouldShowTree && (
853
+ <Box
854
+ sx={{
855
+ height: ConfigManager.instance.config.epiTree.HEADER_HEIGHT,
856
+ position: 'absolute',
857
+ background: theme.palette.background.paper,
858
+ width: treeCanvasWidth,
859
+ top: 0,
860
+ zIndex: 1,
861
+ }}
862
+ >
863
+ <Box
864
+ ref={setHeaderCanvas}
865
+ aria-hidden
866
+ component={'canvas'}
867
+ role={'figure'}
868
+ sx={{
869
+ width: treeCanvasWidth,
870
+ height: ConfigManager.instance.config.epiTree.HEADER_HEIGHT,
871
+ }}
872
+ />
873
+ </Box>
874
+ )}
875
+ <Box
876
+ ref={scrollContainerRef}
877
+ sx={{
878
+ paddingTop: `${ConfigManager.instance.config.epiTree.HEADER_HEIGHT}px`,
879
+ position: 'absolute',
880
+ height,
881
+ width,
882
+ overflowY: 'hidden',
883
+ }}
884
+ >
885
+ {!isTreeUnavailable && shouldShowTree && (
886
+ <Box
887
+ ref={setTreeCanvas}
888
+ aria-label={treeCanvasAriaLabel}
889
+ component={'canvas'}
890
+ role={'figure'}
891
+ sx={{
892
+ width: treeCanvasWidth,
893
+ height: treeCanvasHeight,
894
+ }}
895
+ />
896
+ )}
897
+ </Box>
898
+ </Box>
899
+ <EpiContextMenu
900
+ config={epiContextMenuConfig}
901
+ getExtraItems={getEpiContextMenuExtraItems}
902
+ onMenuClose={onEpiContextMenuClose}
903
+ />
904
+ </EpiWidget>
905
+ );
906
+ };