@alepha/ui 0.18.2 → 0.19.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 (257) hide show
  1. package/dist/admin/{AdminApiKeys-BJhIwfD6.js → AdminApiKeys-Bt1PjO6o.js} +3 -4
  2. package/dist/admin/{AdminApiKeys-BJhIwfD6.js.map → AdminApiKeys-Bt1PjO6o.js.map} +1 -1
  3. package/dist/admin/{AdminAudits-DzD_4cDt.js → AdminAudits-C7c1CN4c.js} +3 -4
  4. package/dist/admin/{AdminAudits-DzD_4cDt.js.map → AdminAudits-C7c1CN4c.js.map} +1 -1
  5. package/dist/admin/{AdminDashboard-C92tIc6x.js → AdminDashboard-C3RXpTp6.js} +3 -4
  6. package/dist/admin/{AdminDashboard-C92tIc6x.js.map → AdminDashboard-C3RXpTp6.js.map} +1 -1
  7. package/dist/admin/{AdminFiles-DLpfhBkf.js → AdminFiles-31ivR6Wq.js} +3 -4
  8. package/dist/admin/{AdminFiles-DLpfhBkf.js.map → AdminFiles-31ivR6Wq.js.map} +1 -1
  9. package/dist/admin/{AdminJobDashboard-KIOkeMgE.js → AdminJobDashboard-BABLe7hL.js} +73 -25
  10. package/dist/admin/AdminJobDashboard-BABLe7hL.js.map +1 -0
  11. package/dist/admin/{AdminJobExecutions-D0Yo_PU0.js → AdminJobExecutions-D-G8RIlr.js} +3 -4
  12. package/dist/admin/{AdminJobExecutions-D0Yo_PU0.js.map → AdminJobExecutions-D-G8RIlr.js.map} +1 -1
  13. package/dist/admin/{AdminJobRegistry-PFajqaGK.js → AdminJobRegistry-oIS3K9NX.js} +3 -4
  14. package/dist/admin/{AdminJobRegistry-PFajqaGK.js.map → AdminJobRegistry-oIS3K9NX.js.map} +1 -1
  15. package/dist/admin/{AdminLayout-B1DXZHDn.js → AdminLayout-BmZ9mtXh.js} +8 -25
  16. package/dist/admin/AdminLayout-BmZ9mtXh.js.map +1 -0
  17. package/dist/admin/AdminNotifications-DHdzksww.js +541 -0
  18. package/dist/admin/AdminNotifications-DHdzksww.js.map +1 -0
  19. package/dist/admin/{AdminParameters-BspPeqp_.js → AdminParameters-CyZQSXnN.js} +118 -112
  20. package/dist/admin/AdminParameters-CyZQSXnN.js.map +1 -0
  21. package/dist/admin/{AdminSessions-BnH5CZQl.js → AdminSessions--xwELDSO.js} +3 -4
  22. package/dist/admin/{AdminSessions-BnH5CZQl.js.map → AdminSessions--xwELDSO.js.map} +1 -1
  23. package/dist/admin/{AdminUserLayout-DUbC6-BI.js → AdminUserLayout-DvBTG5gd.js} +82 -115
  24. package/dist/admin/AdminUserLayout-DvBTG5gd.js.map +1 -0
  25. package/dist/admin/{AdminUserProfile-DuTUnjdG.js → AdminUserProfile-CzsPBl6Z.js} +7 -6
  26. package/dist/admin/AdminUserProfile-CzsPBl6Z.js.map +1 -0
  27. package/dist/admin/{AdminUserSessions-DvZdAGpL.js → AdminUserSessions-C-aUnhVN.js} +3 -4
  28. package/dist/admin/{AdminUserSessions-DvZdAGpL.js.map → AdminUserSessions-C-aUnhVN.js.map} +1 -1
  29. package/dist/admin/{AdminUsers-CR9z0g_5.js → AdminUsers-BYwei5sj.js} +4 -4
  30. package/dist/admin/AdminUsers-BYwei5sj.js.map +1 -0
  31. package/dist/admin/{AuthLayout-DsUfp9RG.js → AuthLayout-CkPGLJku.js} +3 -4
  32. package/dist/admin/{AuthLayout-DsUfp9RG.js.map → AuthLayout-CkPGLJku.js.map} +1 -1
  33. package/dist/{demo/IconGoogle-CSQLPYwX.js → admin/IconGoogle-8Nkx6yax.js} +2 -4
  34. package/dist/admin/{IconGoogle-Ch1m3Uzl.js.map → IconGoogle-8Nkx6yax.js.map} +1 -1
  35. package/dist/admin/Login-DSBqNsZc.js +274 -0
  36. package/dist/admin/Login-DSBqNsZc.js.map +1 -0
  37. package/dist/admin/{Profile-B2EcIDB9.js → Profile-CDRjJo0P.js} +31 -29
  38. package/dist/admin/Profile-CDRjJo0P.js.map +1 -0
  39. package/dist/admin/{Register-Z3fxRbUF.js → Register-4QGFOnfh.js} +201 -146
  40. package/dist/admin/Register-4QGFOnfh.js.map +1 -0
  41. package/dist/admin/{ResetPassword-_Y1qTTKh.js → ResetPassword-Gxc9L_mY.js} +9 -10
  42. package/dist/admin/ResetPassword-Gxc9L_mY.js.map +1 -0
  43. package/dist/admin/{VerifyEmail-Bg22bwcC.js → VerifyEmail-D7G5NnaN.js} +25 -11
  44. package/dist/admin/VerifyEmail-D7G5NnaN.js.map +1 -0
  45. package/dist/admin/adminUserAtom-DCi4wf-v.js +11 -0
  46. package/dist/admin/adminUserAtom-DCi4wf-v.js.map +1 -0
  47. package/dist/admin/{core-BVO_TQxb.js → core-D1AbU50V.js} +662 -570
  48. package/dist/admin/core-D1AbU50V.js.map +1 -0
  49. package/dist/admin/index.d.ts +141 -53
  50. package/dist/admin/index.d.ts.map +1 -1
  51. package/dist/admin/index.js +67 -49
  52. package/dist/admin/index.js.map +1 -1
  53. package/dist/admin/rolldown-runtime-CiIaOW0V.js +13 -0
  54. package/dist/{demo/AuthLayout-DN-ClJQk.js → auth/AuthLayout-CfRKcTqP.js} +3 -4
  55. package/dist/auth/{AuthLayout-C161NeF6.js.map → AuthLayout-CfRKcTqP.js.map} +1 -1
  56. package/dist/{admin/IconGoogle-Ch1m3Uzl.js → auth/IconGoogle-8Nkx6yax.js} +2 -4
  57. package/dist/auth/{IconGoogle-Ch1m3Uzl.js.map → IconGoogle-8Nkx6yax.js.map} +1 -1
  58. package/dist/auth/Login-DJyweoPS.js +274 -0
  59. package/dist/auth/Login-DJyweoPS.js.map +1 -0
  60. package/dist/auth/{Profile-BMpXJ0oi.js → Profile-Cy93pNTw.js} +31 -29
  61. package/dist/auth/Profile-Cy93pNTw.js.map +1 -0
  62. package/dist/auth/{Register-2gx8qll-.js → Register-CSqzzitW.js} +201 -146
  63. package/dist/auth/Register-CSqzzitW.js.map +1 -0
  64. package/dist/{demo/ResetPassword-CAPj8MO3.js → auth/ResetPassword-B61QPlQi.js} +9 -10
  65. package/dist/auth/ResetPassword-B61QPlQi.js.map +1 -0
  66. package/dist/{demo/VerifyEmail-DFmdCdYs.js → auth/VerifyEmail-CqBJ11id.js} +25 -11
  67. package/dist/auth/VerifyEmail-CqBJ11id.js.map +1 -0
  68. package/dist/auth/{core-DyfeVr5c.js → core-C6D3pazL.js} +403 -343
  69. package/dist/auth/core-C6D3pazL.js.map +1 -0
  70. package/dist/auth/index.d.ts +93 -54
  71. package/dist/auth/index.d.ts.map +1 -1
  72. package/dist/auth/index.js +30 -31
  73. package/dist/auth/index.js.map +1 -1
  74. package/dist/auth/rolldown-runtime-CiIaOW0V.js +13 -0
  75. package/dist/core/index.d.ts +123 -62
  76. package/dist/core/index.d.ts.map +1 -1
  77. package/dist/core/index.js +878 -776
  78. package/dist/core/index.js.map +1 -1
  79. package/dist/{auth/AuthLayout-C161NeF6.js → demo/AuthLayout-Dq5tSLSc.js} +3 -4
  80. package/dist/demo/{AuthLayout-DN-ClJQk.js.map → AuthLayout-Dq5tSLSc.js.map} +1 -1
  81. package/dist/demo/DemoButton-_Ws2w-J0.js +181 -0
  82. package/dist/demo/DemoButton-_Ws2w-J0.js.map +1 -0
  83. package/dist/demo/DemoControlSelect-ChP4ZOpQ.js +304 -0
  84. package/dist/demo/DemoControlSelect-ChP4ZOpQ.js.map +1 -0
  85. package/dist/demo/DemoDataTable-Hwf_UUni.js +361 -0
  86. package/dist/demo/DemoDataTable-Hwf_UUni.js.map +1 -0
  87. package/dist/demo/{DemoDialog-DW8QEvD1.js → DemoDialog-B01OMVRd.js} +3 -4
  88. package/dist/demo/{DemoDialog-DW8QEvD1.js.map → DemoDialog-B01OMVRd.js.map} +1 -1
  89. package/dist/demo/{DemoFlex-CAhLUanT.js → DemoFlex-870PEl0V.js} +4 -5
  90. package/dist/demo/{DemoFlex-CAhLUanT.js.map → DemoFlex-870PEl0V.js.map} +1 -1
  91. package/dist/demo/{DemoHeading-yIFmNjHB.js → DemoHeading-C1YR27fz.js} +4 -5
  92. package/dist/demo/{DemoHeading-yIFmNjHB.js.map → DemoHeading-C1YR27fz.js.map} +1 -1
  93. package/dist/demo/{DemoHome-BSGuBHus.js → DemoHome-DRbL2eGf.js} +4 -5
  94. package/dist/demo/{DemoHome-BSGuBHus.js.map → DemoHome-DRbL2eGf.js.map} +1 -1
  95. package/dist/demo/{DemoJsonViewer-DsA2IpgV.js → DemoJsonViewer-DoABiqBW.js} +4 -5
  96. package/dist/demo/{DemoJsonViewer-DsA2IpgV.js.map → DemoJsonViewer-DoABiqBW.js.map} +1 -1
  97. package/dist/demo/{DemoLayout-Cy6xjn6P.js → DemoLayout-CN_PDCX2.js} +16 -8
  98. package/dist/demo/DemoLayout-CN_PDCX2.js.map +1 -0
  99. package/dist/demo/{DemoLogin-vqxgTu4P.js → DemoLogin-B5x-ug3Q.js} +51 -35
  100. package/dist/demo/DemoLogin-B5x-ug3Q.js.map +1 -0
  101. package/dist/demo/{DemoRegister-YHPvPg77.js → DemoRegister-Q6sg2xuV.js} +51 -53
  102. package/dist/demo/DemoRegister-Q6sg2xuV.js.map +1 -0
  103. package/dist/demo/{DemoResetPassword-mOW18Zlm.js → DemoResetPassword-DrqZfmEw.js} +14 -19
  104. package/dist/demo/DemoResetPassword-DrqZfmEw.js.map +1 -0
  105. package/dist/demo/{DemoSidebar-od7aLjP_.js → DemoSidebar-CfKS6w1o.js} +4 -5
  106. package/dist/demo/{DemoSidebar-od7aLjP_.js.map → DemoSidebar-CfKS6w1o.js.map} +1 -1
  107. package/dist/demo/{DemoText-DU3JeRS0.js → DemoText-pT6Gi5b5.js} +4 -5
  108. package/dist/demo/{DemoText-DU3JeRS0.js.map → DemoText-pT6Gi5b5.js.map} +1 -1
  109. package/dist/demo/{DemoToast-CUJEiPRa.js → DemoToast-I13NBzQQ.js} +3 -4
  110. package/dist/demo/{DemoToast-CUJEiPRa.js.map → DemoToast-I13NBzQQ.js.map} +1 -1
  111. package/dist/demo/{DemoTypeForm-C1dNkahD.js → DemoTypeForm-BqzcrtvN.js} +9 -6
  112. package/dist/demo/DemoTypeForm-BqzcrtvN.js.map +1 -0
  113. package/dist/demo/DemoVerifyEmail-HwD8xfQw.js +33 -0
  114. package/dist/demo/DemoVerifyEmail-HwD8xfQw.js.map +1 -0
  115. package/dist/{auth/IconGoogle-Ch1m3Uzl.js → demo/IconGoogle-CwQy4G9y.js} +2 -4
  116. package/dist/demo/{IconGoogle-CSQLPYwX.js.map → IconGoogle-CwQy4G9y.js.map} +1 -1
  117. package/dist/demo/Login-CqG1iJbn.js +274 -0
  118. package/dist/demo/Login-CqG1iJbn.js.map +1 -0
  119. package/dist/demo/{Profile-BE_Y3co2.js → Profile-C0ojJCaG.js} +31 -29
  120. package/dist/demo/Profile-C0ojJCaG.js.map +1 -0
  121. package/dist/demo/{Register-fXHmBpr3.js → Register-KKZwr_lL.js} +201 -146
  122. package/dist/demo/Register-KKZwr_lL.js.map +1 -0
  123. package/dist/{auth/ResetPassword-DBxt9hKk.js → demo/ResetPassword-DMrLFEtr.js} +9 -10
  124. package/dist/demo/ResetPassword-DMrLFEtr.js.map +1 -0
  125. package/dist/demo/{Showcase-BtEU0pY9.js → Showcase-D49Wud2v.js} +65 -68
  126. package/dist/demo/Showcase-D49Wud2v.js.map +1 -0
  127. package/dist/{auth/VerifyEmail-Z80Ubajk.js → demo/VerifyEmail-BFCAFz6T.js} +25 -11
  128. package/dist/demo/VerifyEmail-BFCAFz6T.js.map +1 -0
  129. package/dist/demo/{auth-Djd7SKiw.js → auth-D9qTZzCa.js} +18 -35
  130. package/dist/demo/{auth-Djd7SKiw.js.map → auth-D9qTZzCa.js.map} +1 -1
  131. package/dist/demo/{core-B7LNjM78.js → core-DRtQklr3.js} +752 -647
  132. package/dist/demo/core-DRtQklr3.js.map +1 -0
  133. package/dist/demo/index.d.ts +1 -0
  134. package/dist/demo/index.d.ts.map +1 -1
  135. package/dist/demo/index.js +25 -22
  136. package/dist/demo/index.js.map +1 -1
  137. package/dist/demo/rolldown-runtime-CiIaOW0V.js +13 -0
  138. package/package.json +19 -19
  139. package/src/admin/AdminRouter.tsx +42 -2
  140. package/src/admin/atoms/adminUserAtom.ts +7 -0
  141. package/src/admin/components/AdminLayout.tsx +2 -14
  142. package/src/admin/components/jobs/AdminJobDashboard.tsx +51 -20
  143. package/src/admin/components/notifications/AdminNotifications.tsx +519 -0
  144. package/src/admin/components/parameters/ParameterDetails.tsx +12 -270
  145. package/src/admin/components/parameters/ParameterDetailsConfigForm.tsx +238 -0
  146. package/src/admin/components/parameters/ParameterDetailsLoading.tsx +24 -0
  147. package/src/admin/components/parameters/ParameterHistory.tsx +10 -11
  148. package/src/admin/components/parameters/ParameterTree.tsx +28 -184
  149. package/src/admin/components/parameters/ParameterTreeNode.tsx +151 -0
  150. package/src/admin/components/shared/AdminResourceHeader.tsx +2 -25
  151. package/src/admin/components/shared/AdminResourceHeaderMenuItem.tsx +37 -0
  152. package/src/admin/components/shared/AdminResourceTabs.tsx +2 -26
  153. package/src/admin/components/shared/AdminResourceTabsItem.tsx +36 -0
  154. package/src/admin/components/users/AdminUserLayout.tsx +84 -127
  155. package/src/admin/components/users/AdminUserProfile.tsx +5 -2
  156. package/src/admin/components/users/AdminUsers.tsx +1 -1
  157. package/src/auth/components/Login.tsx +188 -121
  158. package/src/auth/components/Profile.tsx +1 -22
  159. package/src/auth/components/ProfileField.tsx +39 -0
  160. package/src/auth/components/Register.tsx +215 -158
  161. package/src/auth/components/ResetPassword.tsx +7 -11
  162. package/src/auth/components/VerifyEmail.tsx +35 -10
  163. package/src/auth/components/buttons/UserButton.tsx +19 -21
  164. package/src/auth/index.ts +1 -0
  165. package/src/core/components/Flex.tsx +34 -0
  166. package/src/core/components/buttons/ActionButton.tsx +105 -78
  167. package/src/core/components/data/DetailDrawer.tsx +102 -96
  168. package/src/core/components/data/DetailList.tsx +2 -1
  169. package/src/core/components/dialogs/PromptDialog.tsx +1 -1
  170. package/src/core/components/layout/Breadcrumb.tsx +4 -7
  171. package/src/core/components/layout/DashboardShell.tsx +18 -4
  172. package/src/core/components/layout/Sidebar.tsx +16 -241
  173. package/src/core/components/layout/SidebarCollapsedItem.tsx +91 -0
  174. package/src/core/components/layout/SidebarItem.tsx +146 -0
  175. package/src/core/components/layout/index.ts +3 -1
  176. package/src/core/form/components/Control.tsx +31 -29
  177. package/src/core/form/components/ControlArray.tsx +13 -39
  178. package/src/core/form/components/ControlDate.tsx +10 -21
  179. package/src/core/form/components/ControlNumber.tsx +4 -33
  180. package/src/core/form/components/ControlQueryBuilder.tsx +12 -175
  181. package/src/core/form/components/ControlQueryBuilderHelp.tsx +165 -0
  182. package/src/core/form/components/ControlSelect.browser.spec.tsx +343 -0
  183. package/src/core/form/components/ControlSelect.tsx +294 -92
  184. package/src/core/form/components/TypeForm.browser.spec.tsx +3 -3
  185. package/src/core/form/components/TypeForm.tsx +5 -2
  186. package/src/core/form/index.ts +8 -1
  187. package/src/core/form/utils/parseInput.ts +7 -3
  188. package/src/core/index.ts +3 -1
  189. package/src/core/json/components/JsonViewer.tsx +103 -319
  190. package/src/core/json/components/JsonViewerCopyButton.tsx +46 -0
  191. package/src/core/json/components/JsonViewerRowNode.tsx +120 -0
  192. package/src/core/json/components/JsonViewerShared.ts +76 -0
  193. package/src/core/services/DialogService.tsx +2 -2
  194. package/src/core/styles.css +13 -2
  195. package/src/core/table/components/ColumnPicker.tsx +3 -3
  196. package/src/core/table/components/DataTable.tsx +88 -29
  197. package/src/core/table/components/DataTableFilters.tsx +6 -11
  198. package/src/core/table/components/DataTablePagination.tsx +9 -3
  199. package/src/core/table/components/DataTableToolbar.tsx +7 -3
  200. package/src/core/table/components/FilterPicker.tsx +3 -3
  201. package/src/core/table/interfaces/types.ts +29 -0
  202. package/src/core/utils/icons.tsx +2 -2
  203. package/src/demo/DemoRouter.ts +8 -1
  204. package/src/demo/components/DemoLayout.tsx +12 -2
  205. package/src/demo/components/auth/DemoLogin.tsx +35 -28
  206. package/src/demo/components/auth/DemoRegister.tsx +35 -49
  207. package/src/demo/components/auth/DemoResetPassword.tsx +5 -9
  208. package/src/demo/components/auth/DemoVerifyEmail.tsx +7 -6
  209. package/src/demo/components/core/DemoButton.tsx +123 -103
  210. package/src/demo/components/core/DemoControlSelect.tsx +325 -0
  211. package/src/demo/components/core/DemoDataTable.tsx +255 -237
  212. package/src/demo/components/core/DemoTypeForm.tsx +7 -2
  213. package/src/demo/components/shared/MacWindow.tsx +5 -11
  214. package/src/demo/components/shared/Showcase.tsx +28 -42
  215. package/dist/admin/AdminJobDashboard-KIOkeMgE.js.map +0 -1
  216. package/dist/admin/AdminLayout-B1DXZHDn.js.map +0 -1
  217. package/dist/admin/AdminParameters-BspPeqp_.js.map +0 -1
  218. package/dist/admin/AdminUserLayout-DUbC6-BI.js.map +0 -1
  219. package/dist/admin/AdminUserProfile-DuTUnjdG.js.map +0 -1
  220. package/dist/admin/AdminUsers-CR9z0g_5.js.map +0 -1
  221. package/dist/admin/Login-DHbYJKwg.js +0 -219
  222. package/dist/admin/Login-DHbYJKwg.js.map +0 -1
  223. package/dist/admin/Profile-B2EcIDB9.js.map +0 -1
  224. package/dist/admin/Register-Z3fxRbUF.js.map +0 -1
  225. package/dist/admin/ResetPassword-_Y1qTTKh.js.map +0 -1
  226. package/dist/admin/VerifyEmail-Bg22bwcC.js.map +0 -1
  227. package/dist/admin/core-BVO_TQxb.js.map +0 -1
  228. package/dist/admin/rolldown-runtime-CjeV3_4I.js +0 -18
  229. package/dist/auth/Login-C7jIqf00.js +0 -219
  230. package/dist/auth/Login-C7jIqf00.js.map +0 -1
  231. package/dist/auth/Profile-BMpXJ0oi.js.map +0 -1
  232. package/dist/auth/Register-2gx8qll-.js.map +0 -1
  233. package/dist/auth/ResetPassword-DBxt9hKk.js.map +0 -1
  234. package/dist/auth/VerifyEmail-Z80Ubajk.js.map +0 -1
  235. package/dist/auth/core-DyfeVr5c.js.map +0 -1
  236. package/dist/auth/rolldown-runtime-CjeV3_4I.js +0 -18
  237. package/dist/demo/DemoButton-CGUyR9eM.js +0 -178
  238. package/dist/demo/DemoButton-CGUyR9eM.js.map +0 -1
  239. package/dist/demo/DemoDataTable-QFG-xXSx.js +0 -358
  240. package/dist/demo/DemoDataTable-QFG-xXSx.js.map +0 -1
  241. package/dist/demo/DemoLayout-Cy6xjn6P.js.map +0 -1
  242. package/dist/demo/DemoLogin-vqxgTu4P.js.map +0 -1
  243. package/dist/demo/DemoRegister-YHPvPg77.js.map +0 -1
  244. package/dist/demo/DemoResetPassword-mOW18Zlm.js.map +0 -1
  245. package/dist/demo/DemoTypeForm-C1dNkahD.js.map +0 -1
  246. package/dist/demo/DemoVerifyEmail-D9EcXZ38.js +0 -30
  247. package/dist/demo/DemoVerifyEmail-D9EcXZ38.js.map +0 -1
  248. package/dist/demo/Login-CoYf_P_F.js +0 -219
  249. package/dist/demo/Login-CoYf_P_F.js.map +0 -1
  250. package/dist/demo/Profile-BE_Y3co2.js.map +0 -1
  251. package/dist/demo/Register-fXHmBpr3.js.map +0 -1
  252. package/dist/demo/ResetPassword-CAPj8MO3.js.map +0 -1
  253. package/dist/demo/Showcase-BtEU0pY9.js.map +0 -1
  254. package/dist/demo/VerifyEmail-DFmdCdYs.js.map +0 -1
  255. package/dist/demo/core-B7LNjM78.js.map +0 -1
  256. package/dist/demo/rolldown-runtime-CjeV3_4I.js +0 -18
  257. package/src/demo/styles.css +0 -0
@@ -0,0 +1,541 @@
1
+ import { _ as ActionButton, b as useToast, d as DetailList, l as Flex$1, r as DataTable, s as Text$1 } from "./core-D1AbU50V.js";
2
+ import { t } from "alepha";
3
+ import { useI18n } from "alepha/react/i18n";
4
+ import { Badge, Code, Paper } from "@mantine/core";
5
+ import { jsx, jsxs } from "react/jsx-runtime";
6
+ import { useCallback, useEffect, useState } from "react";
7
+ import { IconRefresh } from "@tabler/icons-react";
8
+ import { useClient } from "alepha/react";
9
+ //#region ../../src/admin/components/notifications/AdminNotifications.tsx
10
+ const formatDuration = (start, end) => {
11
+ const startTime = new Date(start).getTime();
12
+ const duration = (end ? new Date(end).getTime() : Date.now()) - startTime;
13
+ if (duration < 1e3) return `${duration}ms`;
14
+ if (duration < 6e4) return `${(duration / 1e3).toFixed(1)}s`;
15
+ if (duration < 36e5) return `${Math.floor(duration / 6e4)}m ${Math.floor(duration % 6e4 / 1e3)}s`;
16
+ return `${Math.floor(duration / 36e5)}h ${Math.floor(duration % 36e5 / 6e4)}m`;
17
+ };
18
+ const notificationFilters = t.object({ status: t.optional(t.enum([
19
+ "pending",
20
+ "scheduled",
21
+ "retrying",
22
+ "running",
23
+ "completed",
24
+ "failed",
25
+ "dead",
26
+ "cancelled"
27
+ ])) });
28
+ const AdminNotifications = () => {
29
+ const client = useClient();
30
+ const { l } = useI18n();
31
+ return /* @__PURE__ */ jsx(Flex$1, {
32
+ p: "md",
33
+ flex: 1,
34
+ direction: "column",
35
+ gap: "md",
36
+ children: /* @__PURE__ */ jsx(DataTable, {
37
+ submitOnInit: true,
38
+ defaultSize: 20,
39
+ typeFormProps: {
40
+ skipSubmitButton: true,
41
+ columns: 3
42
+ },
43
+ tableProps: {
44
+ horizontalSpacing: "sm",
45
+ verticalSpacing: "sm"
46
+ },
47
+ onFilterChange: (_key, _value, form) => form.submit(),
48
+ filters: notificationFilters,
49
+ defaultFilters: ["status"],
50
+ items: async (filters) => {
51
+ return await client.findNotifications({ query: { ...filters } });
52
+ },
53
+ columns: {
54
+ status: {
55
+ label: "Status",
56
+ value: (item) => {
57
+ return /* @__PURE__ */ jsx(Badge, {
58
+ size: "sm",
59
+ variant: "light",
60
+ color: item.status === "completed" ? "green" : item.status === "running" ? "blue" : item.status === "failed" || item.status === "dead" ? "red" : item.status === "cancelled" ? "yellow" : "gray",
61
+ children: item.status
62
+ });
63
+ }
64
+ },
65
+ template: {
66
+ label: "Template",
67
+ value: (item) => /* @__PURE__ */ jsx(Text$1, {
68
+ size: "sm",
69
+ fw: 500,
70
+ ff: "monospace",
71
+ children: item.template ?? "—"
72
+ })
73
+ },
74
+ type: {
75
+ label: "Type",
76
+ value: (item) => /* @__PURE__ */ jsx(Badge, {
77
+ size: "sm",
78
+ variant: "light",
79
+ color: item.type === "email" ? "blue" : "teal",
80
+ children: item.type ?? "—"
81
+ })
82
+ },
83
+ contact: {
84
+ label: "Contact",
85
+ value: (item) => /* @__PURE__ */ jsx(Text$1, {
86
+ size: "xs",
87
+ c: "dimmed",
88
+ lineClamp: 1,
89
+ children: item.contact ?? "—"
90
+ })
91
+ },
92
+ category: {
93
+ label: "Category",
94
+ defaultHidden: true,
95
+ value: (item) => /* @__PURE__ */ jsx(Text$1, {
96
+ size: "xs",
97
+ c: "dimmed",
98
+ children: item.category ?? "—"
99
+ })
100
+ },
101
+ flags: {
102
+ label: "Flags",
103
+ defaultHidden: true,
104
+ value: (item) => /* @__PURE__ */ jsxs(Flex$1, {
105
+ gap: 4,
106
+ children: [
107
+ item.critical && /* @__PURE__ */ jsx(Badge, {
108
+ size: "xs",
109
+ variant: "light",
110
+ color: "red",
111
+ children: "critical"
112
+ }),
113
+ item.sensitive && /* @__PURE__ */ jsx(Badge, {
114
+ size: "xs",
115
+ variant: "light",
116
+ color: "orange",
117
+ children: "sensitive"
118
+ }),
119
+ !item.critical && !item.sensitive && /* @__PURE__ */ jsx(Text$1, {
120
+ size: "xs",
121
+ c: "dimmed",
122
+ children: "—"
123
+ })
124
+ ]
125
+ })
126
+ },
127
+ createdAt: {
128
+ label: "Created",
129
+ value: (item) => /* @__PURE__ */ jsx(Text$1, {
130
+ size: "xs",
131
+ c: "dimmed",
132
+ children: l(item.createdAt, { date: "fromNow" })
133
+ })
134
+ },
135
+ duration: {
136
+ label: "Duration",
137
+ defaultHidden: true,
138
+ value: (item) => /* @__PURE__ */ jsx(Text$1, {
139
+ size: "xs",
140
+ c: "dimmed",
141
+ ff: "monospace",
142
+ children: item.startedAt && (item.completedAt || item.status === "running") ? formatDuration(item.startedAt, item.completedAt) : "—"
143
+ })
144
+ },
145
+ error: {
146
+ label: "Error",
147
+ defaultHidden: true,
148
+ value: (item) => /* @__PURE__ */ jsx(Text$1, {
149
+ size: "xs",
150
+ c: "dimmed",
151
+ lineClamp: 1,
152
+ children: item.error ?? "—"
153
+ })
154
+ }
155
+ },
156
+ panel: {
157
+ can: (item) => Boolean(item.error),
158
+ render: (item) => /* @__PURE__ */ jsx(Flex$1, {
159
+ direction: "column",
160
+ gap: "sm",
161
+ p: "sm",
162
+ children: item.error && /* @__PURE__ */ jsxs(Flex$1, {
163
+ direction: "column",
164
+ gap: 2,
165
+ children: [/* @__PURE__ */ jsx(Text$1, {
166
+ size: "xs",
167
+ c: "dimmed",
168
+ tt: "uppercase",
169
+ fw: 600,
170
+ children: "Error"
171
+ }), /* @__PURE__ */ jsx(Paper, {
172
+ p: "xs",
173
+ radius: "sm",
174
+ withBorder: true,
175
+ children: /* @__PURE__ */ jsx(Text$1, {
176
+ size: "sm",
177
+ style: {
178
+ whiteSpace: "pre-wrap",
179
+ wordBreak: "break-word"
180
+ },
181
+ children: item.error
182
+ })
183
+ })]
184
+ })
185
+ })
186
+ },
187
+ drawer: (item) => /* @__PURE__ */ jsx(NotificationDetailContent, { item })
188
+ })
189
+ });
190
+ };
191
+ const NotificationDetailContent = ({ item }) => {
192
+ const client = useClient();
193
+ const { l } = useI18n();
194
+ const toast = useToast();
195
+ const [detail, setDetail] = useState(null);
196
+ const [loading, setLoading] = useState(false);
197
+ const loadDetail = useCallback(async (id) => {
198
+ setDetail(null);
199
+ setLoading(true);
200
+ try {
201
+ setDetail(await client.getNotification({ params: { id } }));
202
+ } catch {
203
+ toast.danger("Failed to load notification details");
204
+ } finally {
205
+ setLoading(false);
206
+ }
207
+ }, [client, toast]);
208
+ useEffect(() => {
209
+ loadDetail(item.id);
210
+ }, [item.id, loadDetail]);
211
+ if (loading) return /* @__PURE__ */ jsx(Flex$1, {
212
+ align: "center",
213
+ justify: "center",
214
+ py: "xl",
215
+ children: /* @__PURE__ */ jsx(Text$1, {
216
+ c: "dimmed",
217
+ children: "Loading..."
218
+ })
219
+ });
220
+ if (!detail) return null;
221
+ const rendered = detail.rendered;
222
+ const detailItems = [
223
+ {
224
+ label: "ID",
225
+ value: /* @__PURE__ */ jsx(Text$1, {
226
+ size: "sm",
227
+ ff: "monospace",
228
+ children: detail.id
229
+ }),
230
+ copyable: detail.id
231
+ },
232
+ {
233
+ label: "Status",
234
+ value: /* @__PURE__ */ jsx(Text$1, {
235
+ size: "sm",
236
+ tt: "capitalize",
237
+ children: detail.status
238
+ })
239
+ },
240
+ {
241
+ label: "Template",
242
+ value: /* @__PURE__ */ jsx(Text$1, {
243
+ size: "sm",
244
+ ff: "monospace",
245
+ children: detail.template
246
+ })
247
+ },
248
+ {
249
+ label: "Type",
250
+ value: /* @__PURE__ */ jsx(Badge, {
251
+ size: "sm",
252
+ variant: "light",
253
+ color: detail.type === "email" ? "blue" : "teal",
254
+ children: detail.type
255
+ })
256
+ },
257
+ {
258
+ label: "Contact",
259
+ value: detail.contact
260
+ },
261
+ {
262
+ label: "Category",
263
+ value: detail.category,
264
+ hidden: !detail.category
265
+ },
266
+ {
267
+ label: "Critical",
268
+ value: detail.critical ? "Yes" : "No",
269
+ hidden: !detail.critical
270
+ },
271
+ {
272
+ label: "Sensitive",
273
+ value: detail.sensitive ? "Yes" : "No",
274
+ hidden: !detail.sensitive
275
+ },
276
+ {
277
+ label: "Created",
278
+ value: String(l(detail.createdAt, { date: "lll" }))
279
+ },
280
+ {
281
+ label: "Started",
282
+ value: detail.startedAt ? String(l(detail.startedAt, { date: "lll" })) : void 0,
283
+ hidden: !detail.startedAt
284
+ },
285
+ {
286
+ label: "Duration",
287
+ value: detail.startedAt && (detail.completedAt || detail.status === "running") ? /* @__PURE__ */ jsx(Text$1, {
288
+ size: "sm",
289
+ ff: "monospace",
290
+ children: formatDuration(detail.startedAt, detail.completedAt)
291
+ }) : void 0,
292
+ hidden: !(detail.startedAt && (detail.completedAt || detail.status === "running"))
293
+ }
294
+ ];
295
+ return /* @__PURE__ */ jsxs(Flex$1, {
296
+ direction: "column",
297
+ gap: "md",
298
+ children: [
299
+ /* @__PURE__ */ jsxs(Flex$1, {
300
+ align: "center",
301
+ gap: "sm",
302
+ children: [
303
+ /* @__PURE__ */ jsx(Text$1, {
304
+ fw: 600,
305
+ ff: "monospace",
306
+ children: detail.template
307
+ }),
308
+ /* @__PURE__ */ jsx(Badge, {
309
+ size: "sm",
310
+ variant: "light",
311
+ color: detail.type === "email" ? "blue" : "teal",
312
+ children: detail.type
313
+ }),
314
+ /* @__PURE__ */ jsx(Text$1, {
315
+ size: "sm",
316
+ tt: "capitalize",
317
+ c: "dimmed",
318
+ children: detail.status
319
+ })
320
+ ]
321
+ }),
322
+ /* @__PURE__ */ jsx(Flex$1, {
323
+ gap: "xs",
324
+ children: /* @__PURE__ */ jsx(ActionButton, {
325
+ tooltip: "Refresh",
326
+ variant: "light",
327
+ size: "xs",
328
+ icon: IconRefresh,
329
+ onClick: () => loadDetail(item.id)
330
+ })
331
+ }),
332
+ /* @__PURE__ */ jsxs(Paper, {
333
+ p: "sm",
334
+ radius: "md",
335
+ withBorder: true,
336
+ children: [/* @__PURE__ */ jsx(Text$1, {
337
+ size: "sm",
338
+ fw: 600,
339
+ mb: "xs",
340
+ children: "Details"
341
+ }), /* @__PURE__ */ jsx(DetailList, {
342
+ items: detailItems,
343
+ columns: 2
344
+ })]
345
+ }),
346
+ rendered && /* @__PURE__ */ jsxs(Paper, {
347
+ p: "sm",
348
+ radius: "md",
349
+ withBorder: true,
350
+ children: [
351
+ /* @__PURE__ */ jsx(Text$1, {
352
+ size: "sm",
353
+ fw: 600,
354
+ mb: "xs",
355
+ children: "Content"
356
+ }),
357
+ rendered.type === "email" && /* @__PURE__ */ jsxs(Flex$1, {
358
+ direction: "column",
359
+ gap: "xs",
360
+ children: [
361
+ /* @__PURE__ */ jsxs(Flex$1, {
362
+ direction: "column",
363
+ gap: 2,
364
+ children: [/* @__PURE__ */ jsx(Text$1, {
365
+ size: "xs",
366
+ c: "dimmed",
367
+ tt: "uppercase",
368
+ fw: 600,
369
+ children: "To"
370
+ }), /* @__PURE__ */ jsx(Text$1, {
371
+ size: "sm",
372
+ children: String(rendered.to ?? "")
373
+ })]
374
+ }),
375
+ /* @__PURE__ */ jsxs(Flex$1, {
376
+ direction: "column",
377
+ gap: 2,
378
+ children: [/* @__PURE__ */ jsx(Text$1, {
379
+ size: "xs",
380
+ c: "dimmed",
381
+ tt: "uppercase",
382
+ fw: 600,
383
+ children: "Subject"
384
+ }), /* @__PURE__ */ jsx(Text$1, {
385
+ size: "sm",
386
+ children: String(rendered.subject ?? "")
387
+ })]
388
+ }),
389
+ /* @__PURE__ */ jsxs(Flex$1, {
390
+ direction: "column",
391
+ gap: 2,
392
+ children: [/* @__PURE__ */ jsx(Text$1, {
393
+ size: "xs",
394
+ c: "dimmed",
395
+ tt: "uppercase",
396
+ fw: 600,
397
+ children: "Body"
398
+ }), /* @__PURE__ */ jsx(Paper, {
399
+ p: "xs",
400
+ radius: "sm",
401
+ withBorder: true,
402
+ children: /* @__PURE__ */ jsx(Text$1, {
403
+ size: "sm",
404
+ style: {
405
+ whiteSpace: "pre-wrap",
406
+ wordBreak: "break-word"
407
+ },
408
+ children: String(rendered.body ?? "")
409
+ })
410
+ })]
411
+ })
412
+ ]
413
+ }),
414
+ rendered.type === "sms" && /* @__PURE__ */ jsxs(Flex$1, {
415
+ direction: "column",
416
+ gap: "xs",
417
+ children: [/* @__PURE__ */ jsxs(Flex$1, {
418
+ direction: "column",
419
+ gap: 2,
420
+ children: [/* @__PURE__ */ jsx(Text$1, {
421
+ size: "xs",
422
+ c: "dimmed",
423
+ tt: "uppercase",
424
+ fw: 600,
425
+ children: "To"
426
+ }), /* @__PURE__ */ jsx(Text$1, {
427
+ size: "sm",
428
+ children: String(rendered.to ?? "")
429
+ })]
430
+ }), /* @__PURE__ */ jsxs(Flex$1, {
431
+ direction: "column",
432
+ gap: 2,
433
+ children: [/* @__PURE__ */ jsx(Text$1, {
434
+ size: "xs",
435
+ c: "dimmed",
436
+ tt: "uppercase",
437
+ fw: 600,
438
+ children: "Message"
439
+ }), /* @__PURE__ */ jsx(Paper, {
440
+ p: "xs",
441
+ radius: "sm",
442
+ withBorder: true,
443
+ children: /* @__PURE__ */ jsx(Text$1, {
444
+ size: "sm",
445
+ style: {
446
+ whiteSpace: "pre-wrap",
447
+ wordBreak: "break-word"
448
+ },
449
+ children: String(rendered.message ?? "")
450
+ })
451
+ })]
452
+ })]
453
+ })
454
+ ]
455
+ }),
456
+ detail.variables && Object.keys(detail.variables).length > 0 && /* @__PURE__ */ jsxs(Paper, {
457
+ p: "sm",
458
+ radius: "md",
459
+ withBorder: true,
460
+ children: [/* @__PURE__ */ jsx(Text$1, {
461
+ size: "sm",
462
+ fw: 600,
463
+ mb: "xs",
464
+ children: "Variables"
465
+ }), /* @__PURE__ */ jsx(Code, {
466
+ block: true,
467
+ children: JSON.stringify(detail.variables, null, 2)
468
+ })]
469
+ }),
470
+ detail.error && /* @__PURE__ */ jsxs(Paper, {
471
+ p: "sm",
472
+ radius: "md",
473
+ withBorder: true,
474
+ children: [/* @__PURE__ */ jsx(Text$1, {
475
+ size: "sm",
476
+ fw: 600,
477
+ mb: "xs",
478
+ children: "Error"
479
+ }), /* @__PURE__ */ jsx(Paper, {
480
+ p: "xs",
481
+ radius: "sm",
482
+ withBorder: true,
483
+ children: /* @__PURE__ */ jsx(Text$1, {
484
+ size: "sm",
485
+ style: {
486
+ whiteSpace: "pre-wrap",
487
+ wordBreak: "break-word"
488
+ },
489
+ children: detail.error
490
+ })
491
+ })]
492
+ }),
493
+ detail.logs && detail.logs.length > 0 && /* @__PURE__ */ jsxs(Paper, {
494
+ p: "sm",
495
+ radius: "md",
496
+ withBorder: true,
497
+ children: [/* @__PURE__ */ jsxs(Text$1, {
498
+ size: "sm",
499
+ fw: 600,
500
+ mb: "xs",
501
+ children: [
502
+ "Logs (",
503
+ detail.logs.length,
504
+ ")"
505
+ ]
506
+ }), /* @__PURE__ */ jsx(Flex$1, {
507
+ direction: "column",
508
+ style: {
509
+ maxHeight: 300,
510
+ overflowY: "auto"
511
+ },
512
+ children: detail.logs.map((log, i) => /* @__PURE__ */ jsxs(Flex$1, {
513
+ gap: "sm",
514
+ py: 2,
515
+ children: [
516
+ /* @__PURE__ */ jsx(Badge, {
517
+ size: "xs",
518
+ variant: "default",
519
+ children: log.level
520
+ }),
521
+ /* @__PURE__ */ jsx(Text$1, {
522
+ size: "xs",
523
+ c: "dimmed",
524
+ ff: "monospace",
525
+ children: new Date(log.timestamp).toLocaleTimeString()
526
+ }),
527
+ /* @__PURE__ */ jsx(Text$1, {
528
+ size: "xs",
529
+ children: log.message
530
+ })
531
+ ]
532
+ }, i))
533
+ })]
534
+ })
535
+ ]
536
+ });
537
+ };
538
+ //#endregion
539
+ export { AdminNotifications as default };
540
+
541
+ //# sourceMappingURL=AdminNotifications-DHdzksww.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AdminNotifications-DHdzksww.js","names":["Flex","Text"],"sources":["../../src/admin/components/notifications/AdminNotifications.tsx"],"sourcesContent":["import type { DetailListItem } from \"@alepha/ui\";\nimport {\n ActionButton,\n DataTable,\n DetailList,\n Flex,\n Text,\n useToast,\n} from \"@alepha/ui\";\nimport { Badge, Code, Paper } from \"@mantine/core\";\nimport { IconRefresh } from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type {\n AdminNotificationController,\n NotificationDetailResource,\n NotificationResource,\n} from \"alepha/api/notifications\";\nimport type { LogEntry } from \"alepha/logger\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useCallback, useEffect, useState } from \"react\";\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst formatDuration = (\n start: Date | string,\n end?: Date | string | null,\n): string => {\n const startTime = new Date(start).getTime();\n const endTime = end ? new Date(end).getTime() : Date.now();\n const duration = endTime - startTime;\n\n if (duration < 1000) return `${duration}ms`;\n if (duration < 60000) return `${(duration / 1000).toFixed(1)}s`;\n if (duration < 3600000)\n return `${Math.floor(duration / 60000)}m ${Math.floor((duration % 60000) / 1000)}s`;\n return `${Math.floor(duration / 3600000)}h ${Math.floor((duration % 3600000) / 60000)}m`;\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst notificationFilters = t.object({\n status: t.optional(\n t.enum([\n \"pending\",\n \"scheduled\",\n \"retrying\",\n \"running\",\n \"completed\",\n \"failed\",\n \"dead\",\n \"cancelled\",\n ]),\n ),\n});\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst AdminNotifications = () => {\n const client = useClient<AdminNotificationController>();\n const { l } = useI18n();\n\n return (\n <Flex p=\"md\" flex={1} direction=\"column\" gap=\"md\">\n <DataTable<NotificationResource, typeof notificationFilters>\n submitOnInit\n defaultSize={20}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 3,\n }}\n tableProps={{\n horizontalSpacing: \"sm\",\n verticalSpacing: \"sm\",\n }}\n onFilterChange={(_key, _value, form) => form.submit()}\n filters={notificationFilters}\n defaultFilters={[\"status\"]}\n items={async (filters) => {\n const response = await client.findNotifications({\n query: { ...filters },\n });\n return response as Page<NotificationResource>;\n }}\n columns={{\n status: {\n label: \"Status\",\n value: (item) => {\n const color =\n item.status === \"completed\"\n ? \"green\"\n : item.status === \"running\"\n ? \"blue\"\n : item.status === \"failed\" || item.status === \"dead\"\n ? \"red\"\n : item.status === \"cancelled\"\n ? \"yellow\"\n : \"gray\";\n return (\n <Badge size=\"sm\" variant=\"light\" color={color}>\n {item.status}\n </Badge>\n );\n },\n },\n template: {\n label: \"Template\",\n value: (item) => (\n <Text size=\"sm\" fw={500} ff=\"monospace\">\n {item.template ?? \"\\u2014\"}\n </Text>\n ),\n },\n type: {\n label: \"Type\",\n value: (item) => (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={item.type === \"email\" ? \"blue\" : \"teal\"}\n >\n {item.type ?? \"\\u2014\"}\n </Badge>\n ),\n },\n contact: {\n label: \"Contact\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" lineClamp={1}>\n {item.contact ?? \"\\u2014\"}\n </Text>\n ),\n },\n category: {\n label: \"Category\",\n defaultHidden: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {item.category ?? \"\\u2014\"}\n </Text>\n ),\n },\n flags: {\n label: \"Flags\",\n defaultHidden: true,\n value: (item) => (\n <Flex gap={4}>\n {item.critical && (\n <Badge size=\"xs\" variant=\"light\" color=\"red\">\n critical\n </Badge>\n )}\n {item.sensitive && (\n <Badge size=\"xs\" variant=\"light\" color=\"orange\">\n sensitive\n </Badge>\n )}\n {!item.critical && !item.sensitive && (\n <Text size=\"xs\" c=\"dimmed\">\n {\"\\u2014\"}\n </Text>\n )}\n </Flex>\n ),\n },\n createdAt: {\n label: \"Created\",\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n ),\n },\n duration: {\n label: \"Duration\",\n defaultHidden: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {item.startedAt &&\n (item.completedAt || item.status === \"running\")\n ? formatDuration(item.startedAt, item.completedAt)\n : \"\\u2014\"}\n </Text>\n ),\n },\n error: {\n label: \"Error\",\n defaultHidden: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" lineClamp={1}>\n {item.error ?? \"\\u2014\"}\n </Text>\n ),\n },\n }}\n panel={{\n can: (item) => Boolean(item.error),\n render: (item) => (\n <Flex direction=\"column\" gap=\"sm\" p=\"sm\">\n {item.error && (\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" c=\"dimmed\" tt=\"uppercase\" fw={600}>\n Error\n </Text>\n <Paper p=\"xs\" radius=\"sm\" withBorder>\n <Text\n size=\"sm\"\n style={{\n whiteSpace: \"pre-wrap\",\n wordBreak: \"break-word\",\n }}\n >\n {item.error}\n </Text>\n </Paper>\n </Flex>\n )}\n </Flex>\n ),\n }}\n drawer={(item) => <NotificationDetailContent item={item} />}\n />\n </Flex>\n );\n};\n\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst NotificationDetailContent = ({\n item,\n}: {\n item: NotificationResource;\n}) => {\n const client = useClient<AdminNotificationController>();\n const { l } = useI18n();\n const toast = useToast();\n const [detail, setDetail] = useState<NotificationDetailResource | null>(null);\n const [loading, setLoading] = useState(false);\n\n const loadDetail = useCallback(\n async (id: string) => {\n setDetail(null);\n setLoading(true);\n try {\n const data = await client.getNotification({ params: { id } });\n setDetail(data);\n } catch {\n toast.danger(\"Failed to load notification details\");\n } finally {\n setLoading(false);\n }\n },\n [client, toast],\n );\n\n useEffect(() => {\n loadDetail(item.id);\n }, [item.id, loadDetail]);\n\n if (loading) {\n return (\n <Flex align=\"center\" justify=\"center\" py=\"xl\">\n <Text c=\"dimmed\">Loading...</Text>\n </Flex>\n );\n }\n\n if (!detail) return null;\n\n const rendered = detail.rendered as Record<string, unknown> | undefined;\n\n const detailItems: DetailListItem[] = [\n {\n label: \"ID\",\n value: (\n <Text size=\"sm\" ff=\"monospace\">\n {detail.id}\n </Text>\n ),\n copyable: detail.id,\n },\n {\n label: \"Status\",\n value: (\n <Text size=\"sm\" tt=\"capitalize\">\n {detail.status}\n </Text>\n ),\n },\n {\n label: \"Template\",\n value: (\n <Text size=\"sm\" ff=\"monospace\">\n {detail.template}\n </Text>\n ),\n },\n {\n label: \"Type\",\n value: (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={detail.type === \"email\" ? \"blue\" : \"teal\"}\n >\n {detail.type}\n </Badge>\n ),\n },\n {\n label: \"Contact\",\n value: detail.contact,\n },\n {\n label: \"Category\",\n value: detail.category,\n hidden: !detail.category,\n },\n {\n label: \"Critical\",\n value: detail.critical ? \"Yes\" : \"No\",\n hidden: !detail.critical,\n },\n {\n label: \"Sensitive\",\n value: detail.sensitive ? \"Yes\" : \"No\",\n hidden: !detail.sensitive,\n },\n {\n label: \"Created\",\n value: String(l(detail.createdAt, { date: \"lll\" })),\n },\n {\n label: \"Started\",\n value: detail.startedAt\n ? String(l(detail.startedAt, { date: \"lll\" }))\n : undefined,\n hidden: !detail.startedAt,\n },\n {\n label: \"Duration\",\n value:\n detail.startedAt &&\n (detail.completedAt || detail.status === \"running\") ? (\n <Text size=\"sm\" ff=\"monospace\">\n {formatDuration(detail.startedAt, detail.completedAt)}\n </Text>\n ) : undefined,\n hidden: !(\n detail.startedAt &&\n (detail.completedAt || detail.status === \"running\")\n ),\n },\n ];\n\n return (\n <Flex direction=\"column\" gap=\"md\">\n {/* Header */}\n <Flex align=\"center\" gap=\"sm\">\n <Text fw={600} ff=\"monospace\">\n {detail.template}\n </Text>\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={detail.type === \"email\" ? \"blue\" : \"teal\"}\n >\n {detail.type}\n </Badge>\n <Text size=\"sm\" tt=\"capitalize\" c=\"dimmed\">\n {detail.status}\n </Text>\n </Flex>\n\n {/* Actions */}\n <Flex gap=\"xs\">\n <ActionButton\n tooltip=\"Refresh\"\n variant=\"light\"\n size=\"xs\"\n icon={IconRefresh}\n onClick={() => loadDetail(item.id)}\n />\n </Flex>\n\n {/* Details */}\n <Paper p=\"sm\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"xs\">\n Details\n </Text>\n <DetailList items={detailItems} columns={2} />\n </Paper>\n\n {/* Rendered Content */}\n {rendered && (\n <Paper p=\"sm\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"xs\">\n Content\n </Text>\n {rendered.type === \"email\" && (\n <Flex direction=\"column\" gap=\"xs\">\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" c=\"dimmed\" tt=\"uppercase\" fw={600}>\n To\n </Text>\n <Text size=\"sm\">{String(rendered.to ?? \"\")}</Text>\n </Flex>\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" c=\"dimmed\" tt=\"uppercase\" fw={600}>\n Subject\n </Text>\n <Text size=\"sm\">{String(rendered.subject ?? \"\")}</Text>\n </Flex>\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" c=\"dimmed\" tt=\"uppercase\" fw={600}>\n Body\n </Text>\n <Paper p=\"xs\" radius=\"sm\" withBorder>\n <Text\n size=\"sm\"\n style={{\n whiteSpace: \"pre-wrap\",\n wordBreak: \"break-word\",\n }}\n >\n {String(rendered.body ?? \"\")}\n </Text>\n </Paper>\n </Flex>\n </Flex>\n )}\n {rendered.type === \"sms\" && (\n <Flex direction=\"column\" gap=\"xs\">\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" c=\"dimmed\" tt=\"uppercase\" fw={600}>\n To\n </Text>\n <Text size=\"sm\">{String(rendered.to ?? \"\")}</Text>\n </Flex>\n <Flex direction=\"column\" gap={2}>\n <Text size=\"xs\" c=\"dimmed\" tt=\"uppercase\" fw={600}>\n Message\n </Text>\n <Paper p=\"xs\" radius=\"sm\" withBorder>\n <Text\n size=\"sm\"\n style={{\n whiteSpace: \"pre-wrap\",\n wordBreak: \"break-word\",\n }}\n >\n {String(rendered.message ?? \"\")}\n </Text>\n </Paper>\n </Flex>\n </Flex>\n )}\n </Paper>\n )}\n\n {/* Variables */}\n {detail.variables && Object.keys(detail.variables).length > 0 && (\n <Paper p=\"sm\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"xs\">\n Variables\n </Text>\n <Code block>{JSON.stringify(detail.variables, null, 2)}</Code>\n </Paper>\n )}\n\n {/* Error */}\n {detail.error && (\n <Paper p=\"sm\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"xs\">\n Error\n </Text>\n <Paper p=\"xs\" radius=\"sm\" withBorder>\n <Text\n size=\"sm\"\n style={{\n whiteSpace: \"pre-wrap\",\n wordBreak: \"break-word\",\n }}\n >\n {detail.error}\n </Text>\n </Paper>\n </Paper>\n )}\n\n {/* Logs */}\n {detail.logs && detail.logs.length > 0 && (\n <Paper p=\"sm\" radius=\"md\" withBorder>\n <Text size=\"sm\" fw={600} mb=\"xs\">\n Logs ({detail.logs.length})\n </Text>\n <Flex\n direction=\"column\"\n style={{ maxHeight: 300, overflowY: \"auto\" }}\n >\n {detail.logs.map((log: LogEntry, i: number) => (\n <Flex key={i} gap=\"sm\" py={2}>\n <Badge size=\"xs\" variant=\"default\">\n {log.level}\n </Badge>\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {new Date(log.timestamp).toLocaleTimeString()}\n </Text>\n <Text size=\"xs\">{log.message}</Text>\n </Flex>\n ))}\n </Flex>\n </Paper>\n )}\n </Flex>\n );\n};\n\nexport default AdminNotifications;\n"],"mappings":";;;;;;;;;AAwBA,MAAM,kBACJ,OACA,QACW;CACX,MAAM,YAAY,IAAI,KAAK,MAAM,CAAC,SAAS;CAE3C,MAAM,YADU,MAAM,IAAI,KAAK,IAAI,CAAC,SAAS,GAAG,KAAK,KAAK,IAC/B;AAE3B,KAAI,WAAW,IAAM,QAAO,GAAG,SAAS;AACxC,KAAI,WAAW,IAAO,QAAO,IAAI,WAAW,KAAM,QAAQ,EAAE,CAAC;AAC7D,KAAI,WAAW,KACb,QAAO,GAAG,KAAK,MAAM,WAAW,IAAM,CAAC,IAAI,KAAK,MAAO,WAAW,MAAS,IAAK,CAAC;AACnF,QAAO,GAAG,KAAK,MAAM,WAAW,KAAQ,CAAC,IAAI,KAAK,MAAO,WAAW,OAAW,IAAM,CAAC;;AAKxF,MAAM,sBAAsB,EAAE,OAAO,EACnC,QAAQ,EAAE,SACR,EAAE,KAAK;CACL;CACA;CACA;CACA;CACA;CACA;CACA;CACA;CACD,CAAC,CACH,EACF,CAAC;AAIF,MAAM,2BAA2B;CAC/B,MAAM,SAAS,WAAwC;CACvD,MAAM,EAAE,MAAM,SAAS;AAEvB,QACE,oBAACA,QAAD;EAAM,GAAE;EAAK,MAAM;EAAG,WAAU;EAAS,KAAI;YAC3C,oBAAC,WAAD;GACE,cAAA;GACA,aAAa;GACb,eAAe;IACb,kBAAkB;IAClB,SAAS;IACV;GACD,YAAY;IACV,mBAAmB;IACnB,iBAAiB;IAClB;GACD,iBAAiB,MAAM,QAAQ,SAAS,KAAK,QAAQ;GACrD,SAAS;GACT,gBAAgB,CAAC,SAAS;GAC1B,OAAO,OAAO,YAAY;AAIxB,WAHiB,MAAM,OAAO,kBAAkB,EAC9C,OAAO,EAAE,GAAG,SAAS,EACtB,CAAC;;GAGJ,SAAS;IACP,QAAQ;KACN,OAAO;KACP,QAAQ,SAAS;AAWf,aACE,oBAAC,OAAD;OAAO,MAAK;OAAK,SAAQ;OAAQ,OAVjC,KAAK,WAAW,cACZ,UACA,KAAK,WAAW,YACd,SACA,KAAK,WAAW,YAAY,KAAK,WAAW,SAC1C,QACA,KAAK,WAAW,cACd,WACA;iBAGP,KAAK;OACA,CAAA;;KAGb;IACD,UAAU;KACR,OAAO;KACP,QAAQ,SACN,oBAACC,QAAD;MAAM,MAAK;MAAK,IAAI;MAAK,IAAG;gBACzB,KAAK,YAAY;MACb,CAAA;KAEV;IACD,MAAM;KACJ,OAAO;KACP,QAAQ,SACN,oBAAC,OAAD;MACE,MAAK;MACL,SAAQ;MACR,OAAO,KAAK,SAAS,UAAU,SAAS;gBAEvC,KAAK,QAAQ;MACR,CAAA;KAEX;IACD,SAAS;KACP,OAAO;KACP,QAAQ,SACN,oBAACA,QAAD;MAAM,MAAK;MAAK,GAAE;MAAS,WAAW;gBACnC,KAAK,WAAW;MACZ,CAAA;KAEV;IACD,UAAU;KACR,OAAO;KACP,eAAe;KACf,QAAQ,SACN,oBAACA,QAAD;MAAM,MAAK;MAAK,GAAE;gBACf,KAAK,YAAY;MACb,CAAA;KAEV;IACD,OAAO;KACL,OAAO;KACP,eAAe;KACf,QAAQ,SACN,qBAACD,QAAD;MAAM,KAAK;gBAAX;OACG,KAAK,YACJ,oBAAC,OAAD;QAAO,MAAK;QAAK,SAAQ;QAAQ,OAAM;kBAAM;QAErC,CAAA;OAET,KAAK,aACJ,oBAAC,OAAD;QAAO,MAAK;QAAK,SAAQ;QAAQ,OAAM;kBAAS;QAExC,CAAA;OAET,CAAC,KAAK,YAAY,CAAC,KAAK,aACvB,oBAACC,QAAD;QAAM,MAAK;QAAK,GAAE;kBACf;QACI,CAAA;OAEJ;;KAEV;IACD,WAAW;KACT,OAAO;KACP,QAAQ,SACN,oBAACA,QAAD;MAAM,MAAK;MAAK,GAAE;gBACf,EAAE,KAAK,WAAW,EAAE,MAAM,WAAW,CAAC;MAClC,CAAA;KAEV;IACD,UAAU;KACR,OAAO;KACP,eAAe;KACf,QAAQ,SACN,oBAACA,QAAD;MAAM,MAAK;MAAK,GAAE;MAAS,IAAG;gBAC3B,KAAK,cACL,KAAK,eAAe,KAAK,WAAW,aACjC,eAAe,KAAK,WAAW,KAAK,YAAY,GAChD;MACC,CAAA;KAEV;IACD,OAAO;KACL,OAAO;KACP,eAAe;KACf,QAAQ,SACN,oBAACA,QAAD;MAAM,MAAK;MAAK,GAAE;MAAS,WAAW;gBACnC,KAAK,SAAS;MACV,CAAA;KAEV;IACF;GACD,OAAO;IACL,MAAM,SAAS,QAAQ,KAAK,MAAM;IAClC,SAAS,SACP,oBAACD,QAAD;KAAM,WAAU;KAAS,KAAI;KAAK,GAAE;eACjC,KAAK,SACJ,qBAACA,QAAD;MAAM,WAAU;MAAS,KAAK;gBAA9B,CACE,oBAACC,QAAD;OAAM,MAAK;OAAK,GAAE;OAAS,IAAG;OAAY,IAAI;iBAAK;OAE5C,CAAA,EACP,oBAAC,OAAD;OAAO,GAAE;OAAK,QAAO;OAAK,YAAA;iBACxB,oBAACA,QAAD;QACE,MAAK;QACL,OAAO;SACL,YAAY;SACZ,WAAW;SACZ;kBAEA,KAAK;QACD,CAAA;OACD,CAAA,CACH;;KAEJ,CAAA;IAEV;GACD,SAAS,SAAS,oBAAC,2BAAD,EAAiC,MAAQ,CAAA;GAC3D,CAAA;EACG,CAAA;;AAMX,MAAM,6BAA6B,EACjC,WAGI;CACJ,MAAM,SAAS,WAAwC;CACvD,MAAM,EAAE,MAAM,SAAS;CACvB,MAAM,QAAQ,UAAU;CACxB,MAAM,CAAC,QAAQ,aAAa,SAA4C,KAAK;CAC7E,MAAM,CAAC,SAAS,cAAc,SAAS,MAAM;CAE7C,MAAM,aAAa,YACjB,OAAO,OAAe;AACpB,YAAU,KAAK;AACf,aAAW,KAAK;AAChB,MAAI;AAEF,aADa,MAAM,OAAO,gBAAgB,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAC9C;UACT;AACN,SAAM,OAAO,sCAAsC;YAC3C;AACR,cAAW,MAAM;;IAGrB,CAAC,QAAQ,MAAM,CAChB;AAED,iBAAgB;AACd,aAAW,KAAK,GAAG;IAClB,CAAC,KAAK,IAAI,WAAW,CAAC;AAEzB,KAAI,QACF,QACE,oBAACD,QAAD;EAAM,OAAM;EAAS,SAAQ;EAAS,IAAG;YACvC,oBAACC,QAAD;GAAM,GAAE;aAAS;GAAiB,CAAA;EAC7B,CAAA;AAIX,KAAI,CAAC,OAAQ,QAAO;CAEpB,MAAM,WAAW,OAAO;CAExB,MAAM,cAAgC;EACpC;GACE,OAAO;GACP,OACE,oBAACA,QAAD;IAAM,MAAK;IAAK,IAAG;cAChB,OAAO;IACH,CAAA;GAET,UAAU,OAAO;GAClB;EACD;GACE,OAAO;GACP,OACE,oBAACA,QAAD;IAAM,MAAK;IAAK,IAAG;cAChB,OAAO;IACH,CAAA;GAEV;EACD;GACE,OAAO;GACP,OACE,oBAACA,QAAD;IAAM,MAAK;IAAK,IAAG;cAChB,OAAO;IACH,CAAA;GAEV;EACD;GACE,OAAO;GACP,OACE,oBAAC,OAAD;IACE,MAAK;IACL,SAAQ;IACR,OAAO,OAAO,SAAS,UAAU,SAAS;cAEzC,OAAO;IACF,CAAA;GAEX;EACD;GACE,OAAO;GACP,OAAO,OAAO;GACf;EACD;GACE,OAAO;GACP,OAAO,OAAO;GACd,QAAQ,CAAC,OAAO;GACjB;EACD;GACE,OAAO;GACP,OAAO,OAAO,WAAW,QAAQ;GACjC,QAAQ,CAAC,OAAO;GACjB;EACD;GACE,OAAO;GACP,OAAO,OAAO,YAAY,QAAQ;GAClC,QAAQ,CAAC,OAAO;GACjB;EACD;GACE,OAAO;GACP,OAAO,OAAO,EAAE,OAAO,WAAW,EAAE,MAAM,OAAO,CAAC,CAAC;GACpD;EACD;GACE,OAAO;GACP,OAAO,OAAO,YACV,OAAO,EAAE,OAAO,WAAW,EAAE,MAAM,OAAO,CAAC,CAAC,GAC5C,KAAA;GACJ,QAAQ,CAAC,OAAO;GACjB;EACD;GACE,OAAO;GACP,OACE,OAAO,cACN,OAAO,eAAe,OAAO,WAAW,aACvC,oBAACA,QAAD;IAAM,MAAK;IAAK,IAAG;cAChB,eAAe,OAAO,WAAW,OAAO,YAAY;IAChD,CAAA,GACL,KAAA;GACN,QAAQ,EACN,OAAO,cACN,OAAO,eAAe,OAAO,WAAW;GAE5C;EACF;AAED,QACE,qBAACD,QAAD;EAAM,WAAU;EAAS,KAAI;YAA7B;GAEE,qBAACA,QAAD;IAAM,OAAM;IAAS,KAAI;cAAzB;KACE,oBAACC,QAAD;MAAM,IAAI;MAAK,IAAG;gBACf,OAAO;MACH,CAAA;KACP,oBAAC,OAAD;MACE,MAAK;MACL,SAAQ;MACR,OAAO,OAAO,SAAS,UAAU,SAAS;gBAEzC,OAAO;MACF,CAAA;KACR,oBAACA,QAAD;MAAM,MAAK;MAAK,IAAG;MAAa,GAAE;gBAC/B,OAAO;MACH,CAAA;KACF;;GAGP,oBAACD,QAAD;IAAM,KAAI;cACR,oBAAC,cAAD;KACE,SAAQ;KACR,SAAQ;KACR,MAAK;KACL,MAAM;KACN,eAAe,WAAW,KAAK,GAAG;KAClC,CAAA;IACG,CAAA;GAGP,qBAAC,OAAD;IAAO,GAAE;IAAK,QAAO;IAAK,YAAA;cAA1B,CACE,oBAACC,QAAD;KAAM,MAAK;KAAK,IAAI;KAAK,IAAG;eAAK;KAE1B,CAAA,EACP,oBAAC,YAAD;KAAY,OAAO;KAAa,SAAS;KAAK,CAAA,CACxC;;GAGP,YACC,qBAAC,OAAD;IAAO,GAAE;IAAK,QAAO;IAAK,YAAA;cAA1B;KACE,oBAACA,QAAD;MAAM,MAAK;MAAK,IAAI;MAAK,IAAG;gBAAK;MAE1B,CAAA;KACN,SAAS,SAAS,WACjB,qBAACD,QAAD;MAAM,WAAU;MAAS,KAAI;gBAA7B;OACE,qBAACA,QAAD;QAAM,WAAU;QAAS,KAAK;kBAA9B,CACE,oBAACC,QAAD;SAAM,MAAK;SAAK,GAAE;SAAS,IAAG;SAAY,IAAI;mBAAK;SAE5C,CAAA,EACP,oBAACA,QAAD;SAAM,MAAK;mBAAM,OAAO,SAAS,MAAM,GAAG;SAAQ,CAAA,CAC7C;;OACP,qBAACD,QAAD;QAAM,WAAU;QAAS,KAAK;kBAA9B,CACE,oBAACC,QAAD;SAAM,MAAK;SAAK,GAAE;SAAS,IAAG;SAAY,IAAI;mBAAK;SAE5C,CAAA,EACP,oBAACA,QAAD;SAAM,MAAK;mBAAM,OAAO,SAAS,WAAW,GAAG;SAAQ,CAAA,CAClD;;OACP,qBAACD,QAAD;QAAM,WAAU;QAAS,KAAK;kBAA9B,CACE,oBAACC,QAAD;SAAM,MAAK;SAAK,GAAE;SAAS,IAAG;SAAY,IAAI;mBAAK;SAE5C,CAAA,EACP,oBAAC,OAAD;SAAO,GAAE;SAAK,QAAO;SAAK,YAAA;mBACxB,oBAACA,QAAD;UACE,MAAK;UACL,OAAO;WACL,YAAY;WACZ,WAAW;WACZ;oBAEA,OAAO,SAAS,QAAQ,GAAG;UACvB,CAAA;SACD,CAAA,CACH;;OACF;;KAER,SAAS,SAAS,SACjB,qBAACD,QAAD;MAAM,WAAU;MAAS,KAAI;gBAA7B,CACE,qBAACA,QAAD;OAAM,WAAU;OAAS,KAAK;iBAA9B,CACE,oBAACC,QAAD;QAAM,MAAK;QAAK,GAAE;QAAS,IAAG;QAAY,IAAI;kBAAK;QAE5C,CAAA,EACP,oBAACA,QAAD;QAAM,MAAK;kBAAM,OAAO,SAAS,MAAM,GAAG;QAAQ,CAAA,CAC7C;UACP,qBAACD,QAAD;OAAM,WAAU;OAAS,KAAK;iBAA9B,CACE,oBAACC,QAAD;QAAM,MAAK;QAAK,GAAE;QAAS,IAAG;QAAY,IAAI;kBAAK;QAE5C,CAAA,EACP,oBAAC,OAAD;QAAO,GAAE;QAAK,QAAO;QAAK,YAAA;kBACxB,oBAACA,QAAD;SACE,MAAK;SACL,OAAO;UACL,YAAY;UACZ,WAAW;UACZ;mBAEA,OAAO,SAAS,WAAW,GAAG;SAC1B,CAAA;QACD,CAAA,CACH;SACF;;KAEH;;GAIT,OAAO,aAAa,OAAO,KAAK,OAAO,UAAU,CAAC,SAAS,KAC1D,qBAAC,OAAD;IAAO,GAAE;IAAK,QAAO;IAAK,YAAA;cAA1B,CACE,oBAACA,QAAD;KAAM,MAAK;KAAK,IAAI;KAAK,IAAG;eAAK;KAE1B,CAAA,EACP,oBAAC,MAAD;KAAM,OAAA;eAAO,KAAK,UAAU,OAAO,WAAW,MAAM,EAAE;KAAQ,CAAA,CACxD;;GAIT,OAAO,SACN,qBAAC,OAAD;IAAO,GAAE;IAAK,QAAO;IAAK,YAAA;cAA1B,CACE,oBAACA,QAAD;KAAM,MAAK;KAAK,IAAI;KAAK,IAAG;eAAK;KAE1B,CAAA,EACP,oBAAC,OAAD;KAAO,GAAE;KAAK,QAAO;KAAK,YAAA;eACxB,oBAACA,QAAD;MACE,MAAK;MACL,OAAO;OACL,YAAY;OACZ,WAAW;OACZ;gBAEA,OAAO;MACH,CAAA;KACD,CAAA,CACF;;GAIT,OAAO,QAAQ,OAAO,KAAK,SAAS,KACnC,qBAAC,OAAD;IAAO,GAAE;IAAK,QAAO;IAAK,YAAA;cAA1B,CACE,qBAACA,QAAD;KAAM,MAAK;KAAK,IAAI;KAAK,IAAG;eAA5B;MAAiC;MACxB,OAAO,KAAK;MAAO;MACrB;QACP,oBAACD,QAAD;KACE,WAAU;KACV,OAAO;MAAE,WAAW;MAAK,WAAW;MAAQ;eAE3C,OAAO,KAAK,KAAK,KAAe,MAC/B,qBAACA,QAAD;MAAc,KAAI;MAAK,IAAI;gBAA3B;OACE,oBAAC,OAAD;QAAO,MAAK;QAAK,SAAQ;kBACtB,IAAI;QACC,CAAA;OACR,oBAACC,QAAD;QAAM,MAAK;QAAK,GAAE;QAAS,IAAG;kBAC3B,IAAI,KAAK,IAAI,UAAU,CAAC,oBAAoB;QACxC,CAAA;OACP,oBAACA,QAAD;QAAM,MAAK;kBAAM,IAAI;QAAe,CAAA;OAC/B;QARI,EAQJ,CACP;KACG,CAAA,CACD;;GAEL"}