@cccsaurora/howler-ui 2.18.0 → 2.19.0-cases.862

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 (342) hide show
  1. package/api/index.d.ts +4 -0
  2. package/api/index.js +10 -2
  3. package/api/search/case.d.ts +4 -0
  4. package/api/search/case.js +8 -0
  5. package/api/search/facet/hit.d.ts +1 -3
  6. package/api/search/facet/index.d.ts +3 -1
  7. package/api/search/index.d.ts +2 -1
  8. package/api/search/index.js +2 -1
  9. package/api/socket/index.d.ts +3 -0
  10. package/api/socket/index.js +6 -0
  11. package/api/socket/viewers.d.ts +2 -0
  12. package/api/socket/viewers.js +8 -0
  13. package/api/socket/viewers.test.js +44 -0
  14. package/api/v2/case/index.d.ts +9 -0
  15. package/api/v2/case/index.js +21 -0
  16. package/api/v2/case/items.d.ts +6 -0
  17. package/api/v2/case/items.js +18 -0
  18. package/api/v2/case/rules.d.ts +6 -0
  19. package/api/v2/case/rules.js +18 -0
  20. package/api/v2/index.d.ts +4 -0
  21. package/api/v2/index.js +6 -0
  22. package/api/v2/search/facet.d.ts +3 -0
  23. package/api/v2/search/facet.js +12 -0
  24. package/api/v2/search/index.d.ts +5 -0
  25. package/api/v2/search/index.js +24 -0
  26. package/commons/components/leftnav/LeftNavDrawer.js +1 -1
  27. package/components/app/App.js +52 -12
  28. package/components/app/hooks/useMatchers.d.ts +1 -1
  29. package/components/app/hooks/useMatchers.js +23 -11
  30. package/components/app/hooks/useMatchers.test.js +22 -22
  31. package/components/app/hooks/useTitle.js +5 -5
  32. package/components/app/providers/FavouritesProvider.js +2 -2
  33. package/components/app/providers/ModalProvider.d.ts +1 -0
  34. package/components/app/providers/ParameterProvider.d.ts +9 -2
  35. package/components/app/providers/ParameterProvider.js +165 -240
  36. package/components/app/providers/ParameterProvider.test.js +346 -94
  37. package/components/app/providers/RecordProvider.d.ts +23 -0
  38. package/components/app/providers/{HitProvider.js → RecordProvider.js} +42 -42
  39. package/components/app/providers/{HitSearchProvider.d.ts → RecordSearchProvider.d.ts} +6 -6
  40. package/components/app/providers/{HitSearchProvider.js → RecordSearchProvider.js} +12 -17
  41. package/components/app/providers/{HitSearchProvider.test.js → RecordSearchProvider.test.js} +52 -71
  42. package/components/app/providers/SocketProvider.d.ts +11 -2
  43. package/components/app/providers/SocketProvider.js +18 -5
  44. package/components/app/providers/UserListProvider.js +28 -8
  45. package/components/elements/ContextMenu.d.ts +56 -0
  46. package/components/elements/ContextMenu.js +109 -0
  47. package/components/elements/ContextMenu.test.d.ts +1 -0
  48. package/components/elements/ContextMenu.test.js +215 -0
  49. package/components/{routes/overviews/OverviewEditor.js → elements/MarkdownEditor.js} +3 -3
  50. package/components/elements/ObjectDetails.d.ts +6 -0
  51. package/components/elements/{hit/HitDetails.js → ObjectDetails.js} +17 -17
  52. package/components/elements/PluginTypography.d.ts +2 -1
  53. package/components/elements/PluginTypography.js +3 -2
  54. package/components/elements/UserList.d.ts +5 -2
  55. package/components/elements/UserList.js +18 -8
  56. package/components/elements/addons/search/phrase/Phrase.js +1 -1
  57. package/components/elements/case/CaseCard.d.ts +12 -0
  58. package/components/elements/case/CaseCard.js +42 -0
  59. package/components/elements/case/CasePreview.d.ts +6 -0
  60. package/components/elements/case/CasePreview.js +17 -0
  61. package/components/elements/case/StatusIcon.d.ts +5 -0
  62. package/components/elements/case/StatusIcon.js +13 -0
  63. package/components/elements/display/ChipPopper.d.ts +1 -1
  64. package/components/elements/display/ChipPopper.js +5 -5
  65. package/components/elements/display/HowlerCard.js +1 -1
  66. package/components/elements/display/Modal.js +2 -0
  67. package/components/elements/hit/HitActions.js +4 -4
  68. package/components/elements/hit/HitBanner.d.ts +1 -0
  69. package/components/elements/hit/HitBanner.js +34 -51
  70. package/components/elements/hit/HitCard.d.ts +2 -0
  71. package/components/elements/hit/HitCard.js +7 -7
  72. package/components/elements/hit/HitLabels.js +2 -2
  73. package/components/elements/hit/HitOutline.d.ts +1 -0
  74. package/components/elements/hit/HitOutline.js +3 -3
  75. package/components/elements/hit/{HitQuickSearch.d.ts → HitPreview.d.ts} +3 -3
  76. package/components/elements/hit/{HitQuickSearch.js → HitPreview.js} +10 -4
  77. package/components/elements/hit/HitSummary.d.ts +2 -1
  78. package/components/elements/hit/HitSummary.js +6 -5
  79. package/components/elements/hit/aggregate/HitGraph.js +8 -8
  80. package/components/elements/hit/elements/AnalyticLink.d.ts +9 -0
  81. package/components/elements/hit/elements/AnalyticLink.js +22 -0
  82. package/components/elements/hit/elements/Assigned.js +6 -3
  83. package/components/elements/hit/elements/Assigned.test.d.ts +1 -0
  84. package/components/elements/hit/elements/Assigned.test.js +65 -0
  85. package/components/elements/hit/outlines/DefaultOutline.js +1 -1
  86. package/components/elements/hit/related/RelatedRecords.js +63 -0
  87. package/components/elements/observable/ObservableCard.d.ts +6 -0
  88. package/components/elements/observable/ObservableCard.js +22 -0
  89. package/components/elements/observable/ObservablePreview.d.ts +6 -0
  90. package/components/elements/observable/ObservablePreview.js +12 -0
  91. package/components/elements/{hit/HitComments.d.ts → record/RecordComments.d.ts} +5 -4
  92. package/components/elements/{hit/HitComments.js → record/RecordComments.js} +29 -28
  93. package/components/{routes/hits/search/HitContextMenu.d.ts → elements/record/RecordContextMenu.d.ts} +3 -3
  94. package/components/elements/record/RecordContextMenu.js +268 -0
  95. package/components/elements/record/RecordContextMenu.test.d.ts +1 -0
  96. package/components/{routes/hits/search/HitContextMenu.test.js → elements/record/RecordContextMenu.test.js} +190 -39
  97. package/components/elements/record/RecordRelated.d.ts +7 -0
  98. package/components/elements/record/RecordRelated.js +34 -0
  99. package/components/elements/{hit/HitWorklog.d.ts → record/RecordWorklog.d.ts} +4 -3
  100. package/components/elements/{hit/HitWorklog.js → record/RecordWorklog.js} +15 -13
  101. package/components/elements/view/ViewTitle.d.ts +1 -0
  102. package/components/elements/view/ViewTitle.js +9 -2
  103. package/components/hooks/useHitActions.d.ts +1 -1
  104. package/components/hooks/useHitActions.js +4 -4
  105. package/components/hooks/useMyPreferences.js +10 -1
  106. package/components/hooks/useMySearch.js +2 -2
  107. package/components/hooks/useMySitemap.js +4 -1
  108. package/components/hooks/useMyTheme.js +9 -2
  109. package/components/hooks/{useHitSelection.d.ts → useRecordSelection.d.ts} +2 -2
  110. package/components/hooks/{useHitSelection.js → useRecordSelection.js} +12 -33
  111. package/components/hooks/useRelatedRecords.d.ts +13 -0
  112. package/components/hooks/useRelatedRecords.js +32 -0
  113. package/components/routes/403.d.ts +3 -0
  114. package/components/routes/403.js +10 -0
  115. package/components/routes/action/edit/ActionEditor.js +3 -3
  116. package/components/routes/action/useMyActionFunctions.js +4 -1
  117. package/components/routes/action/view/ActionDetails.js +6 -1
  118. package/components/routes/action/view/ActionSearch.js +5 -4
  119. package/components/routes/action/view/markdown/integrations.en.md.js +1 -1
  120. package/components/routes/action/view/markdown/integrations.fr.md.js +1 -1
  121. package/components/routes/advanced/QueryBuilder.js +1 -1
  122. package/components/routes/advanced/QueryEditor.js +3 -3
  123. package/components/routes/advanced/historyCompletionProvider.js +3 -3
  124. package/components/routes/analytics/AnalyticDetails.js +2 -2
  125. package/components/routes/analytics/AnalyticSearch.js +1 -1
  126. package/components/routes/cases/CaseViewer.d.ts +2 -0
  127. package/components/routes/cases/CaseViewer.js +44 -0
  128. package/components/routes/cases/CaseViewer.test.d.ts +1 -0
  129. package/components/routes/cases/CaseViewer.test.js +133 -0
  130. package/components/routes/cases/Cases.d.ts +2 -0
  131. package/components/routes/cases/Cases.js +148 -0
  132. package/components/routes/cases/constants.d.ts +6 -0
  133. package/components/routes/cases/constants.js +6 -0
  134. package/components/routes/cases/detail/AlertPanel.d.ts +6 -0
  135. package/components/routes/cases/detail/AlertPanel.js +33 -0
  136. package/components/routes/cases/detail/CaseAssets.d.ts +11 -0
  137. package/components/routes/cases/detail/CaseAssets.js +104 -0
  138. package/components/routes/cases/detail/CaseAssets.test.d.ts +1 -0
  139. package/components/routes/cases/detail/CaseAssets.test.js +167 -0
  140. package/components/routes/cases/detail/CaseDashboard.d.ts +7 -0
  141. package/components/routes/cases/detail/CaseDashboard.js +66 -0
  142. package/components/routes/cases/detail/CaseDetails.d.ts +6 -0
  143. package/components/routes/cases/detail/CaseDetails.js +70 -0
  144. package/components/routes/cases/detail/CaseOverview.d.ts +7 -0
  145. package/components/routes/cases/detail/CaseOverview.js +43 -0
  146. package/components/routes/cases/detail/CaseRules.d.ts +7 -0
  147. package/components/routes/cases/detail/CaseRules.js +57 -0
  148. package/components/routes/cases/detail/CaseRules.test.d.ts +1 -0
  149. package/components/routes/cases/detail/CaseRules.test.js +221 -0
  150. package/components/routes/cases/detail/CaseSidebar.d.ts +8 -0
  151. package/components/routes/cases/detail/CaseSidebar.js +107 -0
  152. package/components/routes/cases/detail/CaseSidebar.test.d.ts +1 -0
  153. package/components/routes/cases/detail/CaseSidebar.test.js +266 -0
  154. package/components/routes/cases/detail/CaseTask.d.ts +11 -0
  155. package/components/routes/cases/detail/CaseTask.js +66 -0
  156. package/components/routes/cases/detail/CaseTimeline.d.ts +12 -0
  157. package/components/routes/cases/detail/CaseTimeline.js +106 -0
  158. package/components/routes/cases/detail/CaseTimeline.test.d.ts +1 -0
  159. package/components/routes/cases/detail/CaseTimeline.test.js +320 -0
  160. package/components/routes/cases/detail/CreateRuleDialog.d.ts +9 -0
  161. package/components/routes/cases/detail/CreateRuleDialog.js +163 -0
  162. package/components/routes/cases/detail/CreateRuleDialog.test.d.ts +1 -0
  163. package/components/routes/cases/detail/CreateRuleDialog.test.js +259 -0
  164. package/components/routes/cases/detail/ItemPage.d.ts +6 -0
  165. package/components/routes/cases/detail/ItemPage.js +95 -0
  166. package/components/routes/cases/detail/RelatedCasePanel.d.ts +6 -0
  167. package/components/routes/cases/detail/RelatedCasePanel.js +34 -0
  168. package/components/routes/cases/detail/TaskPanel.d.ts +7 -0
  169. package/components/routes/cases/detail/TaskPanel.js +52 -0
  170. package/components/routes/cases/detail/aggregates/CaseAggregate.d.ts +11 -0
  171. package/components/routes/cases/detail/aggregates/CaseAggregate.js +24 -0
  172. package/components/routes/cases/detail/aggregates/SourceAggregate.d.ts +6 -0
  173. package/components/routes/cases/detail/aggregates/SourceAggregate.js +26 -0
  174. package/components/routes/cases/detail/assets/Asset.d.ts +14 -0
  175. package/components/routes/cases/detail/assets/Asset.js +12 -0
  176. package/components/routes/cases/detail/assets/Asset.test.d.ts +1 -0
  177. package/components/routes/cases/detail/assets/Asset.test.js +72 -0
  178. package/components/routes/cases/detail/sidebar/CaseFolder.d.ts +20 -0
  179. package/components/routes/cases/detail/sidebar/CaseFolder.js +83 -0
  180. package/components/routes/cases/detail/sidebar/CaseFolder.test.d.ts +1 -0
  181. package/components/routes/cases/detail/sidebar/CaseFolder.test.js +295 -0
  182. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.d.ts +34 -0
  183. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.js +103 -0
  184. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.test.d.ts +1 -0
  185. package/components/routes/cases/detail/sidebar/CaseFolderContextMenu.test.js +363 -0
  186. package/components/routes/cases/detail/sidebar/FolderEntry.d.ts +25 -0
  187. package/components/routes/cases/detail/sidebar/FolderEntry.js +88 -0
  188. package/components/routes/cases/detail/sidebar/FolderEntry.test.d.ts +1 -0
  189. package/components/routes/cases/detail/sidebar/FolderEntry.test.js +206 -0
  190. package/components/routes/cases/detail/sidebar/RootDropZone.d.ts +5 -0
  191. package/components/routes/cases/detail/sidebar/RootDropZone.js +33 -0
  192. package/components/routes/cases/detail/sidebar/types.d.ts +9 -0
  193. package/components/routes/cases/detail/sidebar/utils.d.ts +3 -0
  194. package/components/routes/cases/detail/sidebar/utils.js +29 -0
  195. package/components/routes/cases/detail/sidebar/utils.test.d.ts +1 -0
  196. package/components/routes/cases/detail/sidebar/utils.test.js +82 -0
  197. package/components/routes/cases/hooks/useCase.d.ts +13 -0
  198. package/components/routes/cases/hooks/useCase.js +69 -0
  199. package/components/routes/cases/hooks/useCase.test.d.ts +1 -0
  200. package/components/routes/cases/hooks/useCase.test.js +141 -0
  201. package/components/routes/cases/modals/AddToCaseModal.d.ts +7 -0
  202. package/components/routes/cases/modals/AddToCaseModal.js +59 -0
  203. package/components/routes/cases/modals/AddToCaseModal.test.d.ts +1 -0
  204. package/components/routes/cases/modals/AddToCaseModal.test.js +313 -0
  205. package/components/routes/cases/modals/CaseRecordRow.d.ts +9 -0
  206. package/components/routes/cases/modals/CaseRecordRow.js +15 -0
  207. package/components/routes/cases/modals/CreateCaseModal.d.ts +7 -0
  208. package/components/routes/cases/modals/CreateCaseModal.js +55 -0
  209. package/components/routes/cases/modals/CreateCaseModal.test.d.ts +1 -0
  210. package/components/routes/cases/modals/CreateCaseModal.test.js +358 -0
  211. package/components/routes/cases/modals/RenameItemModal.d.ts +9 -0
  212. package/components/routes/cases/modals/RenameItemModal.js +48 -0
  213. package/components/routes/cases/modals/ResolveModal.d.ts +7 -0
  214. package/components/routes/cases/modals/ResolveModal.js +115 -0
  215. package/components/routes/cases/modals/ResolveModal.test.d.ts +1 -0
  216. package/components/routes/cases/modals/ResolveModal.test.js +394 -0
  217. package/components/routes/cases/modals/hooks.d.ts +7 -0
  218. package/components/routes/cases/modals/hooks.js +44 -0
  219. package/components/routes/cases/modals/types.d.ts +5 -0
  220. package/components/routes/cases/search/CaseAssigneeFilter.d.ts +6 -0
  221. package/components/routes/cases/search/CaseAssigneeFilter.js +33 -0
  222. package/components/routes/cases/search/CaseAssigneeFilter.test.d.ts +1 -0
  223. package/components/routes/cases/search/CaseAssigneeFilter.test.js +127 -0
  224. package/components/routes/cases/search/CaseDateFilter.d.ts +13 -0
  225. package/components/routes/cases/search/CaseDateFilter.js +26 -0
  226. package/components/routes/cases/search/CaseDateFilter.test.d.ts +1 -0
  227. package/components/routes/cases/search/CaseDateFilter.test.js +115 -0
  228. package/components/routes/cases/search/CaseStatusFilter.d.ts +6 -0
  229. package/components/routes/cases/search/CaseStatusFilter.js +13 -0
  230. package/components/routes/cases/search/CaseStatusFilter.test.d.ts +1 -0
  231. package/components/routes/cases/search/CaseStatusFilter.test.js +86 -0
  232. package/components/routes/dossiers/DossierEditor.js +2 -2
  233. package/components/routes/dossiers/DossierEditor.test.js +1 -1
  234. package/components/routes/help/ActionIntroductionDocumentation.js +1 -1
  235. package/components/routes/help/ApiDocumentation.js +1 -1
  236. package/components/routes/help/HitBannerDocumentation.js +1 -0
  237. package/components/routes/help/HitDocumentation.js +1 -3
  238. package/components/routes/help/markdown/en/retention.md.js +1 -1
  239. package/components/routes/help/markdown/fr/retention.md.js +1 -1
  240. package/components/routes/hits/search/InformationPane.d.ts +1 -0
  241. package/components/routes/hits/search/InformationPane.js +50 -63
  242. package/components/routes/hits/search/LayoutSettings.js +3 -3
  243. package/components/routes/hits/search/QuerySettings.js +2 -1
  244. package/components/routes/hits/search/QuerySettings.test.js +14 -9
  245. package/components/routes/hits/search/{HitBrowser.js → RecordBrowser.js} +9 -9
  246. package/components/routes/hits/search/{HitQuery.d.ts → RecordQuery.d.ts} +2 -2
  247. package/components/routes/hits/search/{HitQuery.js → RecordQuery.js} +6 -6
  248. package/components/routes/hits/search/SearchPane.js +26 -49
  249. package/components/routes/hits/search/ViewLink.js +3 -3
  250. package/components/routes/hits/search/ViewLink.test.js +8 -8
  251. package/components/routes/hits/search/grid/AddColumnModal.js +5 -4
  252. package/components/routes/hits/search/grid/EnhancedCell.d.ts +2 -1
  253. package/components/routes/hits/search/grid/EnhancedCell.js +2 -2
  254. package/components/routes/hits/search/grid/HitGrid.js +20 -18
  255. package/components/routes/hits/search/grid/{HitRow.d.ts → RecordRow.d.ts} +3 -2
  256. package/components/routes/hits/search/grid/{HitRow.js → RecordRow.js} +10 -8
  257. package/components/routes/hits/search/shared/IndexPicker.d.ts +2 -0
  258. package/components/routes/hits/search/shared/IndexPicker.js +20 -0
  259. package/components/routes/hits/view/HitViewer.js +12 -13
  260. package/components/routes/home/AddNewCard.js +1 -1
  261. package/components/routes/home/ViewCard.js +47 -41
  262. package/components/routes/observables/ObservableViewer.d.ts +7 -0
  263. package/components/routes/observables/ObservableViewer.js +27 -0
  264. package/components/routes/overviews/OverviewViewer.js +2 -2
  265. package/components/routes/overviews/template/en.md.js +1 -1
  266. package/components/routes/overviews/template/fr.md.js +1 -1
  267. package/components/routes/views/ViewComposer.js +46 -19
  268. package/locales/en/translation.json +125 -3
  269. package/locales/fr/translation.json +123 -3
  270. package/models/WithMetadata.d.ts +2 -1
  271. package/models/entities/generated/ApiType.d.ts +1 -1
  272. package/models/entities/generated/AttachmentsFile.d.ts +12 -0
  273. package/models/entities/generated/Case.d.ts +28 -0
  274. package/models/entities/generated/DestinationOriginal.d.ts +19 -0
  275. package/models/entities/generated/EmailAttachment.d.ts +8 -0
  276. package/models/entities/generated/EmailParent.d.ts +19 -0
  277. package/models/entities/generated/Enrichments.d.ts +7 -0
  278. package/models/entities/generated/EnrichmentsIndicator.d.ts +21 -0
  279. package/models/entities/generated/Hit.d.ts +1 -0
  280. package/models/entities/generated/Howler.d.ts +0 -5
  281. package/models/entities/generated/HttpResponse.d.ts +11 -0
  282. package/models/entities/generated/Item.d.ts +9 -0
  283. package/models/entities/generated/Observable.d.ts +85 -0
  284. package/models/entities/generated/ObservableCloud.d.ts +20 -0
  285. package/models/entities/generated/ObservableDestination.d.ts +23 -0
  286. package/models/entities/generated/ObservableEmail.d.ts +30 -0
  287. package/models/entities/generated/ObservableFile.d.ts +36 -0
  288. package/models/entities/generated/ObservableHowler.d.ts +42 -0
  289. package/models/entities/generated/ObservableHttp.d.ts +11 -0
  290. package/models/entities/generated/ObservableObserver.d.ts +21 -0
  291. package/models/entities/generated/ObservableOrganization.d.ts +7 -0
  292. package/models/entities/generated/ObservableProcess.d.ts +34 -0
  293. package/models/entities/generated/ObservableSource.d.ts +23 -0
  294. package/models/entities/generated/ObservableThreat.d.ts +21 -0
  295. package/models/entities/generated/ObservableTls.d.ts +12 -0
  296. package/models/entities/generated/ObserverIngress.d.ts +9 -0
  297. package/models/entities/generated/Rule.d.ts +6 -9
  298. package/models/entities/generated/Task.d.ts +10 -0
  299. package/models/entities/generated/Threat.d.ts +2 -2
  300. package/models/entities/generated/{Enrichment.d.ts → ThreatEnrichment.d.ts} +1 -1
  301. package/models/entities/generated/View.d.ts +1 -0
  302. package/models/socket/CaseUpdate.d.ts +5 -0
  303. package/models/socket/ViewersUpdate.d.ts +4 -0
  304. package/package.json +23 -8
  305. package/plugins/clue/components/ClueTypography.js +2 -2
  306. package/plugins/clue/utils.d.ts +2 -1
  307. package/tests/mocks.d.ts +11 -1
  308. package/tests/mocks.js +12 -7
  309. package/tests/server-handlers.js +6 -1
  310. package/tests/utils.d.ts +4 -0
  311. package/tests/utils.js +20 -0
  312. package/utils/constants.d.ts +4 -3
  313. package/utils/constants.js +6 -0
  314. package/utils/hitFunctions.d.ts +2 -1
  315. package/utils/hitFunctions.js +4 -4
  316. package/utils/menuUtils.js +1 -1
  317. package/utils/socketUtils.d.ts +14 -0
  318. package/utils/socketUtils.js +17 -1
  319. package/utils/socketUtils.test.d.ts +1 -0
  320. package/utils/socketUtils.test.js +59 -0
  321. package/utils/typeUtils.d.ts +7 -0
  322. package/utils/typeUtils.js +27 -0
  323. package/utils/viewUtils.js +3 -0
  324. package/components/app/providers/HitProvider.d.ts +0 -22
  325. package/components/elements/display/icons/BundleButton.d.ts +0 -6
  326. package/components/elements/display/icons/BundleButton.js +0 -32
  327. package/components/elements/hit/HitRelated.d.ts +0 -6
  328. package/components/elements/hit/HitRelated.js +0 -7
  329. package/components/routes/help/BundleDocumentation.d.ts +0 -3
  330. package/components/routes/help/BundleDocumentation.js +0 -12
  331. package/components/routes/help/markdown/en/bundles.md.js +0 -1
  332. package/components/routes/help/markdown/fr/bundles.md.js +0 -1
  333. package/components/routes/hits/search/BundleParentMenu.d.ts +0 -6
  334. package/components/routes/hits/search/BundleParentMenu.js +0 -32
  335. package/components/routes/hits/search/BundleScroller.d.ts +0 -2
  336. package/components/routes/hits/search/BundleScroller.js +0 -6
  337. package/components/routes/hits/search/HitContextMenu.js +0 -229
  338. /package/{components/app/providers/HitSearchProvider.test.d.ts → api/socket/viewers.test.d.ts} +0 -0
  339. /package/components/{routes/hits/search/HitContextMenu.test.d.ts → app/providers/RecordSearchProvider.test.d.ts} +0 -0
  340. /package/components/{routes/overviews/OverviewEditor.d.ts → elements/MarkdownEditor.d.ts} +0 -0
  341. /package/components/elements/hit/{HitDetails.d.ts → related/RelatedRecords.d.ts} +0 -0
  342. /package/components/routes/hits/search/{HitBrowser.d.ts → RecordBrowser.d.ts} +0 -0
@@ -0,0 +1,59 @@
1
+ import { createElement as _createElement } from "react";
2
+ import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import { Autocomplete, Button, CircularProgress, Divider, Stack, TextField, Typography } from '@mui/material';
4
+ import api from '@cccsaurora/howler-ui/api';
5
+ import { ModalContext } from '@cccsaurora/howler-ui/components/app/providers/ModalProvider';
6
+ import CaseCard from '@cccsaurora/howler-ui/components/elements/case/CaseCard';
7
+ import useMyApi from '@cccsaurora/howler-ui/components/hooks/useMyApi';
8
+ import { useContext, useEffect, useMemo, useState } from 'react';
9
+ import { useTranslation } from 'react-i18next';
10
+ import CaseRecordRow from './CaseRecordRow';
11
+ import { useFolderOptions, useRecordEntries } from './hooks';
12
+ // ---------------------------------------------------------------------------
13
+ // Modal
14
+ // ---------------------------------------------------------------------------
15
+ const AddToCaseModal = ({ records }) => {
16
+ const { t } = useTranslation();
17
+ const { dispatchApi } = useMyApi();
18
+ const { close } = useContext(ModalContext);
19
+ const [cases, setCases] = useState([]);
20
+ const [selectedCase, setSelectedCase] = useState(null);
21
+ const [submitting, setSubmitting] = useState(false);
22
+ const [entries, updateEntry] = useRecordEntries(records);
23
+ useEffect(() => {
24
+ dispatchApi(api.search.case.post({ query: 'case_id:*', rows: 100 }), { throwError: false }).then(result => {
25
+ if (result) {
26
+ setCases(result.items);
27
+ }
28
+ });
29
+ }, [dispatchApi]);
30
+ const folderOptions = useFolderOptions(selectedCase);
31
+ const isValid = useMemo(() => !!selectedCase &&
32
+ entries.length > 0 &&
33
+ entries.every(e => !!e.title.trim() && !e.path.startsWith('/') && !e.path.endsWith('/')), [selectedCase, entries]);
34
+ const onSubmit = async () => {
35
+ if (!isValid || !selectedCase) {
36
+ return;
37
+ }
38
+ setSubmitting(true);
39
+ try {
40
+ for (const entry of entries) {
41
+ const fullPath = entry.path ? `${entry.path}/${entry.title}` : entry.title;
42
+ await dispatchApi(api.v2.case.items.post(selectedCase.case_id, {
43
+ path: fullPath,
44
+ value: entry.record.howler.id,
45
+ type: entry.record.__index
46
+ }));
47
+ }
48
+ close();
49
+ }
50
+ finally {
51
+ setSubmitting(false);
52
+ }
53
+ };
54
+ return (_jsxs(Stack, { spacing: 2, p: 2, sx: { minWidth: 'min(800px, 60vw)', maxHeight: '90vh', height: '100%' }, children: [_jsx(Typography, { variant: "h4", children: t('modal.cases.add_to_case') }), _jsx(Autocomplete, { options: cases, getOptionLabel: option => option.title ?? option.case_id ?? '', isOptionEqualToValue: (option, value) => option.case_id === value.case_id, value: selectedCase, disablePortal: true, onChange: (_ev, newVal) => {
55
+ setSelectedCase(newVal);
56
+ }, renderOption: (props, option) => (_createElement("li", { ...props, key: option.case_id, style: { ...props.style, display: 'flex', justifyContent: 'stretch', alignItems: 'stretch' } },
57
+ _jsx(CaseCard, { case: option, slotProps: { card: { sx: { width: '100%' } } } }))), renderInput: params => (_jsx(TextField, { ...params, size: "small", placeholder: t('modal.cases.add_to_case.select_case'), fullWidth: true })) }), selectedCase && entries.length > 0 ? (_jsxs(_Fragment, { children: [_jsx(Divider, { children: _jsx(Typography, { variant: "caption", color: "textSecondary", children: t('modal.cases.add_to_case.items_section') }) }), _jsx(Stack, { spacing: 1, overflow: "auto", flex: 1, children: entries.map((entry, i) => (_jsx(CaseRecordRow, { entry: entry, folderOptions: folderOptions, onTitleChange: val => updateEntry(i, 'title', val), onPathChange: val => updateEntry(i, 'path', val) }, entry.record.howler.id))) })] })) : (_jsx("div", { style: { flex: 1, maxHeight: '100px' } })), _jsxs(Stack, { direction: "row", spacing: 1, alignSelf: "end", children: [_jsx(Button, { variant: "outlined", color: "error", onClick: close, disabled: submitting, children: t('cancel') }), _jsx(Button, { variant: "outlined", color: "success", disabled: !isValid || submitting, startIcon: submitting ? _jsx(CircularProgress, { size: 16, color: "inherit" }) : undefined, onClick: onSubmit, children: t('confirm') })] })] }));
58
+ };
59
+ export default AddToCaseModal;
@@ -0,0 +1,313 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ /// <reference types="vitest" />
3
+ import { render, screen, waitFor } from '@testing-library/react';
4
+ import userEvent, {} from '@testing-library/user-event';
5
+ import { ModalContext } from '@cccsaurora/howler-ui/components/app/providers/ModalProvider';
6
+ import i18n from '@cccsaurora/howler-ui/i18n';
7
+ import { I18nextProvider } from 'react-i18next';
8
+ import { createMockCase, createMockHit, createMockObservable } from '@cccsaurora/howler-ui/tests/utils';
9
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
10
+ import AddToCaseModal from './AddToCaseModal';
11
+ // ---------------------------------------------------------------------------
12
+ // Hoisted mocks
13
+ // ---------------------------------------------------------------------------
14
+ const mockDispatchApi = vi.hoisted(() => vi.fn());
15
+ const mockClose = vi.hoisted(() => vi.fn());
16
+ vi.mock('components/hooks/useMyApi', () => ({
17
+ default: () => ({ dispatchApi: mockDispatchApi })
18
+ }));
19
+ vi.mock('components/elements/hit/elements/EscalationChip', () => ({
20
+ default: () => null
21
+ }));
22
+ vi.mock('components/elements/case/CaseCard', () => ({
23
+ default: ({ case: c }) => _jsx("div", { children: c.title ?? c.case_id })
24
+ }));
25
+ vi.mock('api', () => ({
26
+ default: {
27
+ search: {
28
+ case: { post: vi.fn().mockReturnValue('search-case-request') }
29
+ },
30
+ v2: {
31
+ case: {
32
+ items: { post: vi.fn().mockReturnValue('items-post-request') }
33
+ }
34
+ }
35
+ }
36
+ }));
37
+ // ---------------------------------------------------------------------------
38
+ // Fixtures
39
+ // ---------------------------------------------------------------------------
40
+ const CASE_A = createMockCase({ case_id: 'case-a', title: 'Case Alpha', items: [] });
41
+ const CASE_B = createMockCase({
42
+ case_id: 'case-b',
43
+ title: 'Case Beta',
44
+ items: [{ path: 'folder/subfolder/item', type: 'hit', value: 'x', visible: true }]
45
+ });
46
+ const MOCK_HIT_1 = createMockHit({
47
+ howler: { id: 'hit-001', analytic: 'AnalyticOne', status: 'open' }
48
+ });
49
+ const MOCK_HIT_2 = createMockHit({
50
+ howler: { id: 'hit-002', analytic: 'AnalyticTwo', status: 'open' }
51
+ });
52
+ const MOCK_OBSERVABLE = createMockObservable({
53
+ howler: { id: 'obs-001' }
54
+ });
55
+ // ---------------------------------------------------------------------------
56
+ // Wrapper
57
+ // ---------------------------------------------------------------------------
58
+ const Wrapper = ({ children }) => (_jsx(I18nextProvider, { i18n: i18n, children: _jsx(ModalContext.Provider, { value: { close: mockClose, open: vi.fn(), setContent: vi.fn() }, children: children }) }));
59
+ // ---------------------------------------------------------------------------
60
+ // Helpers
61
+ // ---------------------------------------------------------------------------
62
+ const renderModal = (records) => render(_jsx(AddToCaseModal, { records: records }), { wrapper: Wrapper });
63
+ const selectCase = async (user, caseTitle) => {
64
+ const combobox = screen.getAllByRole('combobox')[0];
65
+ await user.click(combobox);
66
+ const option = await screen.findByRole('option', { name: caseTitle });
67
+ await user.click(option);
68
+ };
69
+ // ---------------------------------------------------------------------------
70
+ // Tests
71
+ // ---------------------------------------------------------------------------
72
+ describe('AddToCaseModal', () => {
73
+ let user;
74
+ beforeEach(() => {
75
+ user = userEvent.setup();
76
+ vi.clearAllMocks();
77
+ mockDispatchApi.mockResolvedValue({ items: [CASE_A, CASE_B] });
78
+ });
79
+ // -------------------------------------------------------------------------
80
+ // Initial render
81
+ // -------------------------------------------------------------------------
82
+ describe('initial render', () => {
83
+ it('shows the modal title', async () => {
84
+ renderModal([MOCK_HIT_1]);
85
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
86
+ expect(screen.getByText(i18n.t('modal.cases.add_to_case'))).toBeInTheDocument();
87
+ });
88
+ it('renders cancel and confirm buttons', async () => {
89
+ renderModal([MOCK_HIT_1]);
90
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
91
+ expect(screen.getByRole('button', { name: i18n.t('cancel') })).toBeInTheDocument();
92
+ expect(screen.getByRole('button', { name: i18n.t('confirm') })).toBeInTheDocument();
93
+ });
94
+ it('confirm is disabled before a case is selected', async () => {
95
+ renderModal([MOCK_HIT_1]);
96
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
97
+ expect(screen.getByRole('button', { name: i18n.t('confirm') })).toBeDisabled();
98
+ });
99
+ it('fetches cases on mount', async () => {
100
+ const api = (await import('api')).default;
101
+ renderModal([MOCK_HIT_1]);
102
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
103
+ expect(api.search.case.post).toHaveBeenCalledWith({ query: 'case_id:*', rows: 100 });
104
+ });
105
+ });
106
+ // -------------------------------------------------------------------------
107
+ // Default item titles
108
+ // -------------------------------------------------------------------------
109
+ describe('default item titles', () => {
110
+ it('pre-populates title for a hit with analytic and id', async () => {
111
+ renderModal([MOCK_HIT_1]);
112
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
113
+ await selectCase(user, 'Case Alpha');
114
+ const titleInput = screen.getByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
115
+ expect(titleInput).toHaveValue(`${MOCK_HIT_1.howler.analytic} (${MOCK_HIT_1.howler.id})`);
116
+ });
117
+ it('pre-populates title for an observable with Observable and id', async () => {
118
+ renderModal([MOCK_OBSERVABLE]);
119
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
120
+ await selectCase(user, 'Case Alpha');
121
+ const titleInput = screen.getByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
122
+ expect(titleInput).toHaveValue(`Observable (${MOCK_OBSERVABLE.howler.id})`);
123
+ });
124
+ it('title input is editable', async () => {
125
+ renderModal([MOCK_HIT_1]);
126
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
127
+ await selectCase(user, 'Case Alpha');
128
+ const titleInput = screen.getByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
129
+ await user.clear(titleInput);
130
+ await user.type(titleInput, 'Custom Title');
131
+ expect(titleInput).toHaveValue('Custom Title');
132
+ });
133
+ });
134
+ // -------------------------------------------------------------------------
135
+ // Multiple records
136
+ // -------------------------------------------------------------------------
137
+ describe('multiple records', () => {
138
+ it('renders a row for each hit record', async () => {
139
+ renderModal([MOCK_HIT_1, MOCK_HIT_2]);
140
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
141
+ await selectCase(user, 'Case Alpha');
142
+ expect(screen.getByText(MOCK_HIT_1.howler.analytic)).toBeInTheDocument();
143
+ expect(screen.getByText(MOCK_HIT_2.howler.analytic)).toBeInTheDocument();
144
+ });
145
+ it('renders independent title inputs for each record', async () => {
146
+ renderModal([MOCK_HIT_1, MOCK_HIT_2]);
147
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
148
+ await selectCase(user, 'Case Alpha');
149
+ const titleInputs = screen.getAllByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
150
+ expect(titleInputs).toHaveLength(2);
151
+ expect(titleInputs[0]).toHaveValue(`${MOCK_HIT_1.howler.analytic} (${MOCK_HIT_1.howler.id})`);
152
+ expect(titleInputs[1]).toHaveValue(`${MOCK_HIT_2.howler.analytic} (${MOCK_HIT_2.howler.id})`);
153
+ });
154
+ it('editing one record title does not affect the other', async () => {
155
+ renderModal([MOCK_HIT_1, MOCK_HIT_2]);
156
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
157
+ await selectCase(user, 'Case Alpha');
158
+ const titleInputs = screen.getAllByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
159
+ await user.clear(titleInputs[0]);
160
+ await user.type(titleInputs[0], 'Edited');
161
+ expect(titleInputs[0]).toHaveValue('Edited');
162
+ expect(titleInputs[1]).toHaveValue(`${MOCK_HIT_2.howler.analytic} (${MOCK_HIT_2.howler.id})`);
163
+ });
164
+ it('mixed hit and observable records each get correct default titles', async () => {
165
+ renderModal([MOCK_HIT_1, MOCK_OBSERVABLE]);
166
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
167
+ await selectCase(user, 'Case Alpha');
168
+ const titleInputs = screen.getAllByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
169
+ expect(titleInputs[0]).toHaveValue(`${MOCK_HIT_1.howler.analytic} (${MOCK_HIT_1.howler.id})`);
170
+ expect(titleInputs[1]).toHaveValue(`Observable (${MOCK_OBSERVABLE.howler.id})`);
171
+ });
172
+ });
173
+ // -------------------------------------------------------------------------
174
+ // Folder path options
175
+ // -------------------------------------------------------------------------
176
+ describe('folder path options', () => {
177
+ it('shows folder options derived from the selected case items', async () => {
178
+ renderModal([MOCK_HIT_1]);
179
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
180
+ await selectCase(user, 'Case Beta');
181
+ const pathInputs = screen.getAllByPlaceholderText(i18n.t('modal.cases.add_to_case.select_path'));
182
+ await user.click(pathInputs[0]);
183
+ expect(await screen.findByRole('option', { name: 'folder' })).toBeInTheDocument();
184
+ expect(screen.getByRole('option', { name: 'folder/subfolder' })).toBeInTheDocument();
185
+ });
186
+ it('shows the full path preview when a path and title are set', async () => {
187
+ renderModal([MOCK_HIT_1]);
188
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
189
+ await selectCase(user, 'Case Beta');
190
+ const pathInput = screen.getAllByPlaceholderText(i18n.t('modal.cases.add_to_case.select_path'))[0];
191
+ await user.type(pathInput, 'myfolder');
192
+ const expectedFull = `myfolder/${MOCK_HIT_1.howler.analytic} (${MOCK_HIT_1.howler.id})`;
193
+ await waitFor(() => {
194
+ expect(screen.getByText(i18n.t('modal.cases.add_to_case.full_path', { path: expectedFull }))).toBeInTheDocument();
195
+ });
196
+ });
197
+ });
198
+ // -------------------------------------------------------------------------
199
+ // Validation
200
+ // -------------------------------------------------------------------------
201
+ describe('validation', () => {
202
+ it('enables confirm after a case is selected and titles are filled', async () => {
203
+ renderModal([MOCK_HIT_1]);
204
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
205
+ await selectCase(user, 'Case Alpha');
206
+ expect(screen.getByRole('button', { name: i18n.t('confirm') })).toBeEnabled();
207
+ });
208
+ it('disables confirm when an item title is cleared', async () => {
209
+ renderModal([MOCK_HIT_1]);
210
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
211
+ await selectCase(user, 'Case Alpha');
212
+ const titleInput = screen.getByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
213
+ await user.clear(titleInput);
214
+ expect(screen.getByRole('button', { name: i18n.t('confirm') })).toBeDisabled();
215
+ });
216
+ });
217
+ // -------------------------------------------------------------------------
218
+ // Cancel
219
+ // -------------------------------------------------------------------------
220
+ describe('cancel button', () => {
221
+ it('calls close when cancel is clicked', async () => {
222
+ renderModal([MOCK_HIT_1]);
223
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
224
+ await user.click(screen.getByRole('button', { name: i18n.t('cancel') }));
225
+ expect(mockClose).toHaveBeenCalledTimes(1);
226
+ });
227
+ });
228
+ // -------------------------------------------------------------------------
229
+ // Submission — single record
230
+ // -------------------------------------------------------------------------
231
+ describe('form submission — single record', () => {
232
+ beforeEach(() => {
233
+ mockDispatchApi
234
+ .mockResolvedValueOnce({ items: [CASE_A, CASE_B] }) // case list fetch
235
+ .mockResolvedValue(undefined); // items.post
236
+ });
237
+ it('calls items.post with the correct arguments and closes', async () => {
238
+ const api = (await import('api')).default;
239
+ renderModal([MOCK_HIT_1]);
240
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
241
+ await selectCase(user, 'Case Alpha');
242
+ await user.click(screen.getByRole('button', { name: i18n.t('confirm') }));
243
+ await waitFor(() => expect(mockClose).toHaveBeenCalled());
244
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({
245
+ path: `${MOCK_HIT_1.howler.analytic} (${MOCK_HIT_1.howler.id})`,
246
+ value: MOCK_HIT_1.howler.id,
247
+ type: 'hit'
248
+ }));
249
+ });
250
+ it('combines folder path and title in the submitted path', async () => {
251
+ const api = (await import('api')).default;
252
+ renderModal([MOCK_HIT_1]);
253
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
254
+ await selectCase(user, 'Case Alpha');
255
+ const pathInput = screen.getByPlaceholderText(i18n.t('modal.cases.add_to_case.select_path'));
256
+ await user.type(pathInput, 'investigations');
257
+ await user.click(screen.getByRole('button', { name: i18n.t('confirm') }));
258
+ await waitFor(() => expect(mockClose).toHaveBeenCalled());
259
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({
260
+ path: `investigations/${MOCK_HIT_1.howler.analytic} (${MOCK_HIT_1.howler.id})`
261
+ }));
262
+ });
263
+ it('uses observable __index for observable records', async () => {
264
+ const api = (await import('api')).default;
265
+ renderModal([MOCK_OBSERVABLE]);
266
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
267
+ await selectCase(user, 'Case Alpha');
268
+ await user.click(screen.getByRole('button', { name: i18n.t('confirm') }));
269
+ await waitFor(() => expect(mockClose).toHaveBeenCalled());
270
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({ type: 'observable' }));
271
+ });
272
+ });
273
+ // -------------------------------------------------------------------------
274
+ // Submission — multiple records
275
+ // -------------------------------------------------------------------------
276
+ describe('form submission — multiple records', () => {
277
+ beforeEach(() => {
278
+ mockDispatchApi.mockResolvedValueOnce({ items: [CASE_A, CASE_B] }).mockResolvedValue(undefined);
279
+ });
280
+ it('calls items.post once per record', async () => {
281
+ const api = (await import('api')).default;
282
+ renderModal([MOCK_HIT_1, MOCK_HIT_2]);
283
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
284
+ await selectCase(user, 'Case Alpha');
285
+ await user.click(screen.getByRole('button', { name: i18n.t('confirm') }));
286
+ await waitFor(() => expect(mockClose).toHaveBeenCalled());
287
+ expect(api.v2.case.items.post).toHaveBeenCalledTimes(2);
288
+ });
289
+ it('submits the correct value for each record', async () => {
290
+ const api = (await import('api')).default;
291
+ renderModal([MOCK_HIT_1, MOCK_HIT_2]);
292
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
293
+ await selectCase(user, 'Case Alpha');
294
+ await user.click(screen.getByRole('button', { name: i18n.t('confirm') }));
295
+ await waitFor(() => expect(mockClose).toHaveBeenCalled());
296
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({ value: MOCK_HIT_1.howler.id }));
297
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({ value: MOCK_HIT_2.howler.id }));
298
+ });
299
+ it('uses an independently edited title for each record', async () => {
300
+ const api = (await import('api')).default;
301
+ renderModal([MOCK_HIT_1, MOCK_HIT_2]);
302
+ await waitFor(() => expect(mockDispatchApi).toHaveBeenCalled());
303
+ await selectCase(user, 'Case Alpha');
304
+ const titleInputs = screen.getAllByPlaceholderText(i18n.t('modal.cases.add_to_case.title'));
305
+ await user.clear(titleInputs[0]);
306
+ await user.type(titleInputs[0], 'First Item');
307
+ await user.click(screen.getByRole('button', { name: i18n.t('confirm') }));
308
+ await waitFor(() => expect(mockClose).toHaveBeenCalled());
309
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({ path: 'First Item', value: MOCK_HIT_1.howler.id }));
310
+ expect(api.v2.case.items.post).toHaveBeenCalledWith('case-a', expect.objectContaining({ value: MOCK_HIT_2.howler.id }));
311
+ });
312
+ });
313
+ });
@@ -0,0 +1,9 @@
1
+ import type { FC } from 'react';
2
+ import type { RecordEntry } from './types';
3
+ declare const CaseRecordRow: FC<{
4
+ entry: RecordEntry;
5
+ folderOptions?: string[];
6
+ onTitleChange: (title: string) => void;
7
+ onPathChange: (path: string) => void;
8
+ }>;
9
+ export default CaseRecordRow;
@@ -0,0 +1,15 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { KeyboardArrowDown } from '@mui/icons-material';
3
+ import { Accordion, AccordionDetails, AccordionSummary, Autocomplete, Chip, Stack, TextField, Typography } from '@mui/material';
4
+ import EscalationChip from '@cccsaurora/howler-ui/components/elements/hit/elements/EscalationChip';
5
+ import { HitLayout } from '@cccsaurora/howler-ui/components/elements/hit/HitLayout';
6
+ import { useTranslation } from 'react-i18next';
7
+ import { isHit } from '@cccsaurora/howler-ui/utils/typeUtils';
8
+ const CaseRecordRow = ({ entry, folderOptions = [], onTitleChange, onPathChange }) => {
9
+ const { t } = useTranslation();
10
+ const { record, path, title } = entry;
11
+ const fullPath = path ? `${path}/${title}` : title;
12
+ const pathError = path.startsWith('/') || path.endsWith('/');
13
+ return (_jsxs(Accordion, { variant: "outlined", defaultExpanded: true, sx: { flexShrink: 0 }, children: [_jsx(AccordionSummary, { expandIcon: _jsx(KeyboardArrowDown, {}), sx: { px: 1, minHeight: '48px !important', '& > *': { margin: '0 !important' } }, children: _jsxs(Stack, { direction: "row", alignItems: "center", spacing: 1, width: "100%", children: [_jsx(Typography, { variant: "body2", fontWeight: 500, sx: { flex: 1, overflow: 'hidden', textOverflow: 'ellipsis' }, children: isHit(record) ? record.howler.analytic : 'Observable' }), isHit(record) && _jsx(EscalationChip, { hit: record, layout: HitLayout.DENSE }), isHit(record) && _jsx(Chip, { label: record.howler.status, size: "small", color: "primary", sx: { flexShrink: 0 } }), _jsx(Typography, { variant: "caption", color: "textSecondary", sx: { flexShrink: 0 }, children: record.howler.id })] }) }), _jsx(AccordionDetails, { children: _jsxs(Stack, { spacing: 1, children: [_jsx(Autocomplete, { freeSolo: true, disablePortal: true, options: folderOptions, value: path, onInputChange: (_ev, newVal) => onPathChange(newVal), renderInput: params => (_jsx(TextField, { ...params, size: "small", placeholder: t('modal.cases.add_to_case.select_path'), fullWidth: true, error: pathError, helperText: pathError ? t('modal.cases.add_to_case.path_invalid') : undefined })) }), _jsx(TextField, { size: "small", fullWidth: true, placeholder: t('modal.cases.add_to_case.title'), value: title, onChange: ev => onTitleChange(ev.target.value) }), title && (_jsx(Typography, { variant: "caption", color: "textSecondary", children: t('modal.cases.add_to_case.full_path', { path: fullPath }) }))] }) })] }));
14
+ };
15
+ export default CaseRecordRow;
@@ -0,0 +1,7 @@
1
+ import type { Hit } from '@cccsaurora/howler-ui/models/entities/generated/Hit';
2
+ import type { Observable } from '@cccsaurora/howler-ui/models/entities/generated/Observable';
3
+ import { type FC } from 'react';
4
+ declare const CreateCaseModal: FC<{
5
+ records: (Hit | Observable)[];
6
+ }>;
7
+ export default CreateCaseModal;
@@ -0,0 +1,55 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import { Autocomplete, Button, CircularProgress, Divider, Stack, TextField, Typography } from '@mui/material';
3
+ import api from '@cccsaurora/howler-ui/api';
4
+ import { ModalContext } from '@cccsaurora/howler-ui/components/app/providers/ModalProvider';
5
+ import MarkdownEditor from '@cccsaurora/howler-ui/components/elements/MarkdownEditor';
6
+ import useMyApi from '@cccsaurora/howler-ui/components/hooks/useMyApi';
7
+ import { useContext, useMemo, useState } from 'react';
8
+ import { useTranslation } from 'react-i18next';
9
+ import CaseRecordRow from './CaseRecordRow';
10
+ import { useRecordEntries } from './hooks';
11
+ const ESCALATIONS = ['normal', 'focus', 'crisis'];
12
+ const CreateCaseModal = ({ records }) => {
13
+ const { t } = useTranslation();
14
+ const { dispatchApi } = useMyApi();
15
+ const { close } = useContext(ModalContext);
16
+ const [caseTitle, setCaseTitle] = useState('');
17
+ const [summary, setSummary] = useState('');
18
+ const [overview, setOverview] = useState('');
19
+ const [escalation, setEscalation] = useState(null);
20
+ const [submitting, setSubmitting] = useState(false);
21
+ const [entries, updateEntry] = useRecordEntries(records);
22
+ const isValid = useMemo(() => !!caseTitle.trim() &&
23
+ !!summary.trim() &&
24
+ entries.every(e => !!e.title.trim() && !e.path.startsWith('/') && !e.path.endsWith('/')), [caseTitle, summary, entries]);
25
+ const onSubmit = async () => {
26
+ if (!isValid) {
27
+ return;
28
+ }
29
+ setSubmitting(true);
30
+ try {
31
+ const newCase = await dispatchApi(api.v2.case.post({
32
+ title: caseTitle.trim(),
33
+ summary: summary.trim(),
34
+ ...(overview.trim() ? { overview: overview.trim() } : {}),
35
+ ...(escalation ? { escalation } : {})
36
+ }));
37
+ if (newCase?.case_id) {
38
+ for (const entry of entries) {
39
+ const fullPath = entry.path ? `${entry.path}/${entry.title}` : entry.title;
40
+ await dispatchApi(api.v2.case.items.post(newCase.case_id, {
41
+ path: fullPath,
42
+ value: entry.record.howler.id,
43
+ type: entry.record.__index
44
+ }));
45
+ }
46
+ close();
47
+ }
48
+ }
49
+ finally {
50
+ setSubmitting(false);
51
+ }
52
+ };
53
+ return (_jsxs(Stack, { spacing: 2, p: 2, sx: { minWidth: 'min(800px, 60vw)', maxHeight: '90vh', height: '100%' }, children: [_jsx(Typography, { variant: "h4", children: t('modal.cases.create_case') }), _jsxs(Stack, { spacing: 1, children: [_jsx(TextField, { size: "small", fullWidth: true, placeholder: t('modal.cases.create_case.title'), value: caseTitle, onChange: ev => setCaseTitle(ev.target.value) }), _jsx(TextField, { size: "small", fullWidth: true, placeholder: t('modal.cases.create_case.summary'), value: summary, onChange: ev => setSummary(ev.target.value) }), _jsx(Autocomplete, { options: ESCALATIONS, value: escalation, disablePortal: true, onChange: (_ev, val) => setEscalation(val), renderInput: params => (_jsx(TextField, { ...params, size: "small", placeholder: t('modal.cases.create_case.escalation'), fullWidth: true })) }), _jsxs(Stack, { spacing: 0.5, children: [_jsx(Typography, { variant: "caption", color: "textSecondary", children: t('modal.cases.create_case.overview') }), _jsx(MarkdownEditor, { content: overview, setContent: setOverview, height: "200px", fontSize: 14 })] })] }), entries.length > 0 ? (_jsxs(_Fragment, { children: [_jsx(Divider, { children: _jsx(Typography, { variant: "caption", color: "textSecondary", children: t('modal.cases.create_case.items_section') }) }), _jsx(Stack, { spacing: 1, overflow: "auto", flex: 1, children: entries.map((entry, i) => (_jsx(CaseRecordRow, { entry: entry, onTitleChange: val => updateEntry(i, 'title', val), onPathChange: val => updateEntry(i, 'path', val) }, entry.record.howler.id))) })] })) : (_jsx("div", { style: { flex: 1 } })), _jsxs(Stack, { direction: "row", spacing: 1, alignSelf: "end", children: [_jsx(Button, { variant: "outlined", color: "error", onClick: close, disabled: submitting, children: t('cancel') }), _jsx(Button, { variant: "outlined", color: "success", disabled: !isValid || submitting, startIcon: submitting ? _jsx(CircularProgress, { size: 16, color: "inherit" }) : undefined, onClick: onSubmit, children: t('confirm') })] })] }));
54
+ };
55
+ export default CreateCaseModal;