@atlashub/smartstack 3.21.0 → 3.24.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.
- package/dist/chunks/{AgentSkillsPage-7si3Ng8e.js → AgentSkillsPage-BWQSCYl-.js} +2 -2
- package/dist/chunks/{AgentSkillsPage-7si3Ng8e.js.map → AgentSkillsPage-BWQSCYl-.js.map} +1 -1
- package/dist/chunks/{AgentSkillsPage-D0cD1QdM.js → AgentSkillsPage-IQcMnBaD.js} +2 -2
- package/dist/chunks/{AgentSkillsPage-D0cD1QdM.js.map → AgentSkillsPage-IQcMnBaD.js.map} +1 -1
- package/dist/chunks/{AgentWorkloadPage-H_7ze33H.js → AgentWorkloadPage-DqrjkvWL.js} +2 -2
- package/dist/chunks/{AgentWorkloadPage-H_7ze33H.js.map → AgentWorkloadPage-DqrjkvWL.js.map} +1 -1
- package/dist/chunks/{AgentWorkloadPage-D4d86cdV.js → AgentWorkloadPage-w-HiyFYP.js} +2 -2
- package/dist/chunks/{AgentWorkloadPage-D4d86cdV.js.map → AgentWorkloadPage-w-HiyFYP.js.map} +1 -1
- package/dist/chunks/{ApiCatalogDetailPage-2ktkRrCb.js → ApiCatalogDetailPage-D3L8Yf4G.js} +3 -3
- package/dist/chunks/{ApiCatalogDetailPage-2ktkRrCb.js.map → ApiCatalogDetailPage-D3L8Yf4G.js.map} +1 -1
- package/dist/chunks/{ApiCatalogDetailPage-BQ53xuwD.js → ApiCatalogDetailPage-MPT3Kz6H.js} +2 -2
- package/dist/chunks/{ApiCatalogDetailPage-BQ53xuwD.js.map → ApiCatalogDetailPage-MPT3Kz6H.js.map} +1 -1
- package/dist/chunks/{ApiCatalogPage-BEqTDJz8.js → ApiCatalogPage-D4Hg3uiS.js} +2 -2
- package/dist/chunks/{ApiCatalogPage-BEqTDJz8.js.map → ApiCatalogPage-D4Hg3uiS.js.map} +1 -1
- package/dist/chunks/{ApiCatalogPage-BBkWSLI8.js → ApiCatalogPage-DRg5Cz0r.js} +2 -2
- package/dist/chunks/{ApiCatalogPage-BBkWSLI8.js.map → ApiCatalogPage-DRg5Cz0r.js.map} +1 -1
- package/dist/chunks/{ApplicationDetailPage-BYJ2YMPq.js → ApplicationDetailPage-Caizuyn2.js} +2 -2
- package/dist/chunks/{ApplicationDetailPage-BYJ2YMPq.js.map → ApplicationDetailPage-Caizuyn2.js.map} +1 -1
- package/dist/chunks/{ApplicationDetailPage-D8-bf1as.js → ApplicationDetailPage-CuCW6aMB.js} +4 -4
- package/dist/chunks/{ApplicationDetailPage-D8-bf1as.js.map → ApplicationDetailPage-CuCW6aMB.js.map} +1 -1
- package/dist/chunks/{ApplicationsDashboardPage-BBlLms2r.js → ApplicationsDashboardPage-B2MW8-Kc.js} +2 -2
- package/dist/chunks/{ApplicationsDashboardPage-BBlLms2r.js.map → ApplicationsDashboardPage-B2MW8-Kc.js.map} +1 -1
- package/dist/chunks/{ApplicationsDashboardPage-DTWZxJJM.js → ApplicationsDashboardPage-BDIjFIYZ.js} +3 -3
- package/dist/chunks/{ApplicationsDashboardPage-DTWZxJJM.js.map → ApplicationsDashboardPage-BDIjFIYZ.js.map} +1 -1
- package/dist/chunks/{ApplicationsGridPage-BQaMsK1K.js → ApplicationsGridPage-DV-FihKj.js} +2 -2
- package/dist/chunks/{ApplicationsGridPage-BQaMsK1K.js.map → ApplicationsGridPage-DV-FihKj.js.map} +1 -1
- package/dist/chunks/{ApplicationsGridPage-DbVcvezt.js → ApplicationsGridPage-DXsTfXPI.js} +2 -2
- package/dist/chunks/{ApplicationsGridPage-DbVcvezt.js.map → ApplicationsGridPage-DXsTfXPI.js.map} +1 -1
- package/dist/chunks/{ApplicationsListPage-DYKM2Yeo.js → ApplicationsListPage--CGkyBuJ.js} +2 -2
- package/dist/chunks/{ApplicationsListPage-DYKM2Yeo.js.map → ApplicationsListPage--CGkyBuJ.js.map} +1 -1
- package/dist/chunks/{ApplicationsListPage-C91v2rZt.js → ApplicationsListPage-JUX823bh.js} +2 -2
- package/dist/chunks/{ApplicationsListPage-C91v2rZt.js.map → ApplicationsListPage-JUX823bh.js.map} +1 -1
- package/dist/chunks/{ApplicationsPage-BCbgotIx.js → ApplicationsPage-6zgFye6w.js} +2 -2
- package/dist/chunks/{ApplicationsPage-BCbgotIx.js.map → ApplicationsPage-6zgFye6w.js.map} +1 -1
- package/dist/chunks/{ApplicationsPage-CW3-Hjlu.js → ApplicationsPage-CQPuuiO6.js} +4 -4
- package/dist/chunks/{ApplicationsPage-CW3-Hjlu.js.map → ApplicationsPage-CQPuuiO6.js.map} +1 -1
- package/dist/chunks/{AssignmentRulesPage-D8vfGDBN.js → AssignmentRulesPage-CFffeEbo.js} +2 -2
- package/dist/chunks/{AssignmentRulesPage-D8vfGDBN.js.map → AssignmentRulesPage-CFffeEbo.js.map} +1 -1
- package/dist/chunks/{AssignmentRulesPage-CxktlEMB.js → AssignmentRulesPage-D78UeUId.js} +2 -2
- package/dist/chunks/{AssignmentRulesPage-CxktlEMB.js.map → AssignmentRulesPage-D78UeUId.js.map} +1 -1
- package/dist/chunks/{AssignmentsPage-DmfBYQAD.js → AssignmentsPage-Cww2ifZF.js} +2 -2
- package/dist/chunks/{AssignmentsPage-DmfBYQAD.js.map → AssignmentsPage-Cww2ifZF.js.map} +1 -1
- package/dist/chunks/{AssignmentsPage-sRCCBmRc.js → AssignmentsPage-DE_QS2LO.js} +2 -2
- package/dist/chunks/{AssignmentsPage-sRCCBmRc.js.map → AssignmentsPage-DE_QS2LO.js.map} +1 -1
- package/dist/chunks/{AuthCallbackPage-C7XiZxKb.js → AuthCallbackPage-CA2nO6DG.js} +2 -2
- package/dist/chunks/{AuthCallbackPage-C7XiZxKb.js.map → AuthCallbackPage-CA2nO6DG.js.map} +1 -1
- package/dist/chunks/{AuthCallbackPage-BCe_bwJM.js → AuthCallbackPage-CDUAoX-N.js} +2 -2
- package/dist/chunks/{AuthCallbackPage-BCe_bwJM.js.map → AuthCallbackPage-CDUAoX-N.js.map} +1 -1
- package/dist/chunks/{ConfirmEmailPage-BUfGSqxF.js → ConfirmEmailPage-BqsILAYH.js} +2 -2
- package/dist/chunks/{ConfirmEmailPage-BUfGSqxF.js.map → ConfirmEmailPage-BqsILAYH.js.map} +1 -1
- package/dist/chunks/{ConfirmEmailPage-Buj4x-rx.js → ConfirmEmailPage-INeHCuMB.js} +2 -2
- package/dist/chunks/{ConfirmEmailPage-Buj4x-rx.js.map → ConfirmEmailPage-INeHCuMB.js.map} +1 -1
- package/dist/chunks/{CreateSupportTicketPage-CKDX_HQm.js → CreateSupportTicketPage-BWeuV2aU.js} +2 -2
- package/dist/chunks/{CreateSupportTicketPage-CKDX_HQm.js.map → CreateSupportTicketPage-BWeuV2aU.js.map} +1 -1
- package/dist/chunks/{CreateSupportTicketPage-0LgY-_pu.js → CreateSupportTicketPage-OBwF4v7b.js} +2 -2
- package/dist/chunks/{CreateSupportTicketPage-0LgY-_pu.js.map → CreateSupportTicketPage-OBwF4v7b.js.map} +1 -1
- package/dist/chunks/{DashboardPage-CUZ80NGV.js → DashboardPage-CKHqWrdS.js} +3 -3
- package/dist/chunks/{DashboardPage-CUZ80NGV.js.map → DashboardPage-CKHqWrdS.js.map} +1 -1
- package/dist/chunks/{DashboardPage-CaNOAstg.js → DashboardPage-COmc9b__.js} +3 -3
- package/dist/chunks/{DashboardPage-CaNOAstg.js.map → DashboardPage-COmc9b__.js.map} +1 -1
- package/dist/chunks/{DashboardPage-B48_rQFi.js → DashboardPage-CfKZHiSj.js} +2 -2
- package/dist/chunks/{DashboardPage-B48_rQFi.js.map → DashboardPage-CfKZHiSj.js.map} +1 -1
- package/dist/chunks/{DashboardPage-CUZV1J9t.js → DashboardPage-CwEZZ3jx.js} +2 -2
- package/dist/chunks/{DashboardPage-CUZV1J9t.js.map → DashboardPage-CwEZZ3jx.js.map} +1 -1
- package/dist/chunks/{EscalationConfigPage-CdzAbnGy.js → EscalationConfigPage--7lgZ0kJ.js} +2 -2
- package/dist/chunks/{EscalationConfigPage-CdzAbnGy.js.map → EscalationConfigPage--7lgZ0kJ.js.map} +1 -1
- package/dist/chunks/{EscalationConfigPage-CYGIl_e6.js → EscalationConfigPage-DPyiBcqV.js} +2 -2
- package/dist/chunks/{EscalationConfigPage-CYGIl_e6.js.map → EscalationConfigPage-DPyiBcqV.js.map} +1 -1
- package/dist/chunks/{ForceChangePasswordPage-lRpkwcX7.js → ForceChangePasswordPage-BE-6umub.js} +2 -2
- package/dist/chunks/{ForceChangePasswordPage-lRpkwcX7.js.map → ForceChangePasswordPage-BE-6umub.js.map} +1 -1
- package/dist/chunks/{ForceChangePasswordPage-CvmYAV3r.js → ForceChangePasswordPage-CnsYoWmV.js} +2 -2
- package/dist/chunks/{ForceChangePasswordPage-CvmYAV3r.js.map → ForceChangePasswordPage-CnsYoWmV.js.map} +1 -1
- package/dist/chunks/{ForgotPasswordPage-0u49E4Pw.js → ForgotPasswordPage-CSq4DnFF.js} +2 -2
- package/dist/chunks/{ForgotPasswordPage-0u49E4Pw.js.map → ForgotPasswordPage-CSq4DnFF.js.map} +1 -1
- package/dist/chunks/{ForgotPasswordPage-CxQUqKOm.js → ForgotPasswordPage-DZLVolAC.js} +2 -2
- package/dist/chunks/{ForgotPasswordPage-CxQUqKOm.js.map → ForgotPasswordPage-DZLVolAC.js.map} +1 -1
- package/dist/chunks/{GroupDetailPage-DFBvVO1S.js → GroupDetailPage-Bf9Wb_2j.js} +5 -5
- package/dist/chunks/{GroupDetailPage-DFBvVO1S.js.map → GroupDetailPage-Bf9Wb_2j.js.map} +1 -1
- package/dist/chunks/{GroupDetailPage-B2FkKrGG.js → GroupDetailPage-R-hf3rJ7.js} +2 -2
- package/dist/chunks/{GroupDetailPage-B2FkKrGG.js.map → GroupDetailPage-R-hf3rJ7.js.map} +1 -1
- package/dist/chunks/{MyAccessRequestsPage-C9IX4c0K.js → MyAccessRequestsPage-BIisvWM6.js} +2 -2
- package/dist/chunks/{MyAccessRequestsPage-C9IX4c0K.js.map → MyAccessRequestsPage-BIisvWM6.js.map} +1 -1
- package/dist/chunks/{MyAccessRequestsPage-D6pVULNM.js → MyAccessRequestsPage-BLSV7Tbx.js} +2 -2
- package/dist/chunks/{MyAccessRequestsPage-D6pVULNM.js.map → MyAccessRequestsPage-BLSV7Tbx.js.map} +1 -1
- package/dist/chunks/{MyTenantsPage-BEcYYdGR.js → MyTenantsPage-D-7k9CP1.js} +3 -3
- package/dist/chunks/{MyTenantsPage-BEcYYdGR.js.map → MyTenantsPage-D-7k9CP1.js.map} +1 -1
- package/dist/chunks/{MyTenantsPage-D9f85zjF.js → MyTenantsPage-DqGW6aDt.js} +2 -2
- package/dist/chunks/{MyTenantsPage-D9f85zjF.js.map → MyTenantsPage-DqGW6aDt.js.map} +1 -1
- package/dist/chunks/{MyTicketsPage-DJR8h6y1.js → MyTicketsPage--DgDsnZA.js} +2 -2
- package/dist/chunks/{MyTicketsPage-DJR8h6y1.js.map → MyTicketsPage--DgDsnZA.js.map} +1 -1
- package/dist/chunks/{MyTicketsPage-DiOUExKJ.js → MyTicketsPage-CqJ3Aqob.js} +2 -2
- package/dist/chunks/{MyTicketsPage-DiOUExKJ.js.map → MyTicketsPage-CqJ3Aqob.js.map} +1 -1
- package/dist/chunks/{NavigationAppsPage-CeHbxfZw.js → NavigationAppsPage-Bebis_RT.js} +2 -2
- package/dist/chunks/{NavigationAppsPage-CeHbxfZw.js.map → NavigationAppsPage-Bebis_RT.js.map} +1 -1
- package/dist/chunks/{NavigationAppsPage-If7tmCFY.js → NavigationAppsPage-THNPOAjv.js} +2 -2
- package/dist/chunks/{NavigationAppsPage-If7tmCFY.js.map → NavigationAppsPage-THNPOAjv.js.map} +1 -1
- package/dist/chunks/{NotificationsPage-C29Lln5o.js → NotificationsPage-CAbNW_Cn.js} +2 -2
- package/dist/chunks/{NotificationsPage-C29Lln5o.js.map → NotificationsPage-CAbNW_Cn.js.map} +1 -1
- package/dist/chunks/{NotificationsPage-BiaLRb0s.js → NotificationsPage-DxwizUhL.js} +2 -2
- package/dist/chunks/{NotificationsPage-BiaLRb0s.js.map → NotificationsPage-DxwizUhL.js.map} +1 -1
- package/dist/chunks/{OnboardingWizardPage-DQrBKNBq.js → OnboardingWizardPage-C6HlbJ3K.js} +2 -2
- package/dist/chunks/{OnboardingWizardPage-DQrBKNBq.js.map → OnboardingWizardPage-C6HlbJ3K.js.map} +1 -1
- package/dist/chunks/{OnboardingWizardPage-BQah4cI8.js → OnboardingWizardPage-CyC2zONO.js} +2 -2
- package/dist/chunks/{OnboardingWizardPage-BQah4cI8.js.map → OnboardingWizardPage-CyC2zONO.js.map} +1 -1
- package/dist/chunks/{PermissionDetailPage-Ckjdjvf9.js → PermissionDetailPage-BDHiNgky.js} +2 -2
- package/dist/chunks/{PermissionDetailPage-Ckjdjvf9.js.map → PermissionDetailPage-BDHiNgky.js.map} +1 -1
- package/dist/chunks/{PermissionDetailPage-Dh8v7mGj.js → PermissionDetailPage-C5K17ydY.js} +2 -2
- package/dist/chunks/{PermissionDetailPage-Dh8v7mGj.js.map → PermissionDetailPage-C5K17ydY.js.map} +1 -1
- package/dist/chunks/{PermissionsPage-l0PnY-EE.js → PermissionsPage-COI5LJPo.js} +2 -2
- package/dist/chunks/{PermissionsPage-l0PnY-EE.js.map → PermissionsPage-COI5LJPo.js.map} +1 -1
- package/dist/chunks/{PermissionsPage-DLy9U3P3.js → PermissionsPage-CkOwH2_d.js} +2 -2
- package/dist/chunks/{PermissionsPage-DLy9U3P3.js.map → PermissionsPage-CkOwH2_d.js.map} +1 -1
- package/dist/chunks/{PortalDashboardPage-DFBx38-x.js → PortalDashboardPage-CoEC4CmC.js} +2 -2
- package/dist/chunks/{PortalDashboardPage-DFBx38-x.js.map → PortalDashboardPage-CoEC4CmC.js.map} +1 -1
- package/dist/chunks/{PortalDashboardPage-rQYhrX0q.js → PortalDashboardPage-DrYymEf-.js} +2 -2
- package/dist/chunks/{PortalDashboardPage-rQYhrX0q.js.map → PortalDashboardPage-DrYymEf-.js.map} +1 -1
- package/dist/chunks/{PreferencesPage-BBu8yZQB.js → PreferencesPage-CJRaU3ba.js} +2 -2
- package/dist/chunks/{PreferencesPage-BBu8yZQB.js.map → PreferencesPage-CJRaU3ba.js.map} +1 -1
- package/dist/chunks/{PreferencesPage-B81MsNV1.js → PreferencesPage-Cqr9mAab.js} +2 -2
- package/dist/chunks/{PreferencesPage-B81MsNV1.js.map → PreferencesPage-Cqr9mAab.js.map} +1 -1
- package/dist/chunks/{ProfilePage-DDrl10zj.js → ProfilePage-BZVpg6-l.js} +2 -2
- package/dist/chunks/{ProfilePage-DDrl10zj.js.map → ProfilePage-BZVpg6-l.js.map} +1 -1
- package/dist/chunks/{ProfilePage-DPoXwdnc.js → ProfilePage-Cu_FITeL.js} +2 -2
- package/dist/chunks/{ProfilePage-DPoXwdnc.js.map → ProfilePage-Cu_FITeL.js.map} +1 -1
- package/dist/chunks/{ReferencesManagementPage-eFsKjIEK.js → ReferencesManagementPage-DUlVk9Ps.js} +3 -3
- package/dist/chunks/{ReferencesManagementPage-eFsKjIEK.js.map → ReferencesManagementPage-DUlVk9Ps.js.map} +1 -1
- package/dist/chunks/{ReferencesManagementPage-DhVsuElE.js → ReferencesManagementPage-ZCuYtqd7.js} +2 -2
- package/dist/chunks/{ReferencesManagementPage-DhVsuElE.js.map → ReferencesManagementPage-ZCuYtqd7.js.map} +1 -1
- package/dist/chunks/{RegisterPage-CiQib3-6.js → RegisterPage-C4xmVwh9.js} +2 -2
- package/dist/chunks/{RegisterPage-CiQib3-6.js.map → RegisterPage-C4xmVwh9.js.map} +1 -1
- package/dist/chunks/{RegisterPage-bXCcJD88.js → RegisterPage-DGyzoIHT.js} +2 -2
- package/dist/chunks/{RegisterPage-bXCcJD88.js.map → RegisterPage-DGyzoIHT.js.map} +1 -1
- package/dist/chunks/{ResetPasswordPage-Dqiahhnj.js → ResetPasswordPage-DqDD6VPR.js} +2 -2
- package/dist/chunks/{ResetPasswordPage-Dqiahhnj.js.map → ResetPasswordPage-DqDD6VPR.js.map} +1 -1
- package/dist/chunks/{ResetPasswordPage-CubPG3yv.js → ResetPasswordPage-Glu-aeqv.js} +2 -2
- package/dist/chunks/{ResetPasswordPage-CubPG3yv.js.map → ResetPasswordPage-Glu-aeqv.js.map} +1 -1
- package/dist/chunks/{ResolutionModal-Bg7XZmR1.js → ResolutionModal-CxjANAOP.js} +2 -2
- package/dist/chunks/{ResolutionModal-Bg7XZmR1.js.map → ResolutionModal-CxjANAOP.js.map} +1 -1
- package/dist/chunks/{ResolutionModal-DqRk_T0n.js → ResolutionModal-Duat18qV.js} +2 -2
- package/dist/chunks/{ResolutionModal-DqRk_T0n.js.map → ResolutionModal-Duat18qV.js.map} +1 -1
- package/dist/chunks/{RoleDetailPage-Blau6_4c.js → RoleDetailPage-BQffUSnt.js} +3 -3
- package/dist/chunks/{RoleDetailPage-Blau6_4c.js.map → RoleDetailPage-BQffUSnt.js.map} +1 -1
- package/dist/chunks/{RoleDetailPage-CiRVxxIP.js → RoleDetailPage-JTm5lD1_.js} +2 -2
- package/dist/chunks/{RoleDetailPage-CiRVxxIP.js.map → RoleDetailPage-JTm5lD1_.js.map} +1 -1
- package/dist/chunks/{RolesPage-Pm-RN3lP.js → RolesPage-B9rRzciI.js} +2 -2
- package/dist/chunks/{RolesPage-Pm-RN3lP.js.map → RolesPage-B9rRzciI.js.map} +1 -1
- package/dist/chunks/{RolesPage-Cb8joqdJ.js → RolesPage-BN8_zMOC.js} +2 -2
- package/dist/chunks/{RolesPage-Cb8joqdJ.js.map → RolesPage-BN8_zMOC.js.map} +1 -1
- package/dist/chunks/{SlaConfigPage-B86McKM6.js → SlaConfigPage-B7kZNig4.js} +2 -2
- package/dist/chunks/{SlaConfigPage-B86McKM6.js.map → SlaConfigPage-B7kZNig4.js.map} +1 -1
- package/dist/chunks/{SlaConfigPage-BY7gvYU6.js → SlaConfigPage-okvZfA_K.js} +2 -2
- package/dist/chunks/{SlaConfigPage-BY7gvYU6.js.map → SlaConfigPage-okvZfA_K.js.map} +1 -1
- package/dist/chunks/{SupportPermissionsPage-BYxcLMSd.js → SupportPermissionsPage-DGAPqJbl.js} +2 -2
- package/dist/chunks/{SupportPermissionsPage-BYxcLMSd.js.map → SupportPermissionsPage-DGAPqJbl.js.map} +1 -1
- package/dist/chunks/{SupportPermissionsPage-MXqXNJIZ.js → SupportPermissionsPage-Dg_wLOme.js} +2 -2
- package/dist/chunks/{SupportPermissionsPage-MXqXNJIZ.js.map → SupportPermissionsPage-Dg_wLOme.js.map} +1 -1
- package/dist/chunks/{TemplatesPage-BDguJ401.js → TemplatesPage-DT9fhlAU.js} +2 -2
- package/dist/chunks/{TemplatesPage-BDguJ401.js.map → TemplatesPage-DT9fhlAU.js.map} +1 -1
- package/dist/chunks/{TemplatesPage-DdnGgioU.js → TemplatesPage-DiEk538p.js} +2 -2
- package/dist/chunks/{TemplatesPage-DdnGgioU.js.map → TemplatesPage-DiEk538p.js.map} +1 -1
- package/dist/chunks/{TenantCard-ffwWsgFQ.js → TenantCard-BbSYk9_Z.js} +2 -2
- package/dist/chunks/{TenantCard-ffwWsgFQ.js.map → TenantCard-BbSYk9_Z.js.map} +1 -1
- package/dist/chunks/{TenantCard-CUjb6og9.js → TenantCard-CEkiKxcZ.js} +2 -2
- package/dist/chunks/{TenantCard-CUjb6og9.js.map → TenantCard-CEkiKxcZ.js.map} +1 -1
- package/dist/chunks/{TenantScopeSelector-Dz7i1I43.js → TenantScopeSelector-BWfYxvEa.js} +2 -2
- package/dist/chunks/{TenantScopeSelector-Dz7i1I43.js.map → TenantScopeSelector-BWfYxvEa.js.map} +1 -1
- package/dist/chunks/{TenantScopeSelector-Cym_Zyps.js → TenantScopeSelector-D-BKgQPV.js} +2 -2
- package/dist/chunks/{TenantScopeSelector-Cym_Zyps.js.map → TenantScopeSelector-D-BKgQPV.js.map} +1 -1
- package/dist/chunks/{TicketDetailPage-GOh9GX7E.js → TicketDetailPage-C1mNS9Up.js} +2 -2
- package/dist/chunks/{TicketDetailPage-GOh9GX7E.js.map → TicketDetailPage-C1mNS9Up.js.map} +1 -1
- package/dist/chunks/{TicketDetailPage-Du8WMyqf.js → TicketDetailPage-ieVDRh42.js} +2 -2
- package/dist/chunks/{TicketDetailPage-Du8WMyqf.js.map → TicketDetailPage-ieVDRh42.js.map} +1 -1
- package/dist/chunks/{TicketsPage-Bqd6moQy.js → TicketsPage-CnuWsnIW.js} +2 -2
- package/dist/chunks/{TicketsPage-Bqd6moQy.js.map → TicketsPage-CnuWsnIW.js.map} +1 -1
- package/dist/chunks/{TicketsPage-WdU4Bb7M.js → TicketsPage-jjyY15_D.js} +2 -2
- package/dist/chunks/{TicketsPage-WdU4Bb7M.js.map → TicketsPage-jjyY15_D.js.map} +1 -1
- package/dist/chunks/{UserCreateTicketPage-Cm1emgwR.js → UserCreateTicketPage-B8Tvf-ag.js} +2 -2
- package/dist/chunks/{UserCreateTicketPage-Cm1emgwR.js.map → UserCreateTicketPage-B8Tvf-ag.js.map} +1 -1
- package/dist/chunks/{UserCreateTicketPage-BPw-5Y_D.js → UserCreateTicketPage-DnOsDlfO.js} +2 -2
- package/dist/chunks/{UserCreateTicketPage-BPw-5Y_D.js.map → UserCreateTicketPage-DnOsDlfO.js.map} +1 -1
- package/dist/chunks/{UserDashboardPage-BP5WeXPS.js → UserDashboardPage-BrtkJ-NB.js} +2 -2
- package/dist/chunks/{UserDashboardPage-BP5WeXPS.js.map → UserDashboardPage-BrtkJ-NB.js.map} +1 -1
- package/dist/chunks/{UserDashboardPage-B53C8fUq.js → UserDashboardPage-KLB5CQP5.js} +2 -2
- package/dist/chunks/{UserDashboardPage-B53C8fUq.js.map → UserDashboardPage-KLB5CQP5.js.map} +1 -1
- package/dist/chunks/{UserDetailPage-B110bmGX.js → UserDetailPage-U7smBQoF.js} +5 -5
- package/dist/chunks/{UserDetailPage-B110bmGX.js.map → UserDetailPage-U7smBQoF.js.map} +1 -1
- package/dist/chunks/{UserDetailPage-CV2VCE46.js → UserDetailPage-_J6lcKAU.js} +2 -2
- package/dist/chunks/{UserDetailPage-CV2VCE46.js.map → UserDetailPage-_J6lcKAU.js.map} +1 -1
- package/dist/chunks/{UserTicketDetailPage-CCNJON1V.js → UserTicketDetailPage-CWoYQgH-.js} +2 -2
- package/dist/chunks/{UserTicketDetailPage-CCNJON1V.js.map → UserTicketDetailPage-CWoYQgH-.js.map} +1 -1
- package/dist/chunks/{UserTicketDetailPage-V0mLXrox.js → UserTicketDetailPage-DkufSlvZ.js} +2 -2
- package/dist/chunks/{UserTicketDetailPage-V0mLXrox.js.map → UserTicketDetailPage-DkufSlvZ.js.map} +1 -1
- package/dist/chunks/{UsersGroupsPage-CmdaU-z-.js → UsersGroupsPage-C38s2-Rq.js} +3 -3
- package/dist/chunks/{UsersGroupsPage-CmdaU-z-.js.map → UsersGroupsPage-C38s2-Rq.js.map} +1 -1
- package/dist/chunks/{UsersGroupsPage-BgfAMgEP.js → UsersGroupsPage-Dq3rAteo.js} +2 -2
- package/dist/chunks/{UsersGroupsPage-BgfAMgEP.js.map → UsersGroupsPage-Dq3rAteo.js.map} +1 -1
- package/dist/chunks/{UsersPage-Bg7033pp.js → UsersPage-B5C5KEUR.js} +2 -2
- package/dist/chunks/{UsersPage-Bg7033pp.js.map → UsersPage-B5C5KEUR.js.map} +1 -1
- package/dist/chunks/{UsersPage-TYAfwPY1.js → UsersPage-CXC9Hvq6.js} +2 -2
- package/dist/chunks/{UsersPage-TYAfwPY1.js.map → UsersPage-CXC9Hvq6.js.map} +1 -1
- package/dist/chunks/{accessRequestsApi-DZeDvzwv.js → accessRequestsApi-B-4TJ5_U.js} +2 -2
- package/dist/chunks/{accessRequestsApi-DZeDvzwv.js.map → accessRequestsApi-B-4TJ5_U.js.map} +1 -1
- package/dist/chunks/{accessRequestsApi-ZXFPCid2.js → accessRequestsApi-DZSfThpd.js} +2 -2
- package/dist/chunks/{accessRequestsApi-ZXFPCid2.js.map → accessRequestsApi-DZSfThpd.js.map} +1 -1
- package/dist/chunks/{aiApi-CsH8DXgs.js → aiApi-B20Teu2v.js} +2 -2
- package/dist/chunks/{aiApi-CsH8DXgs.js.map → aiApi-B20Teu2v.js.map} +1 -1
- package/dist/chunks/{aiApi-CVPzFTXa.js → aiApi-DMGz-RPM.js} +2 -2
- package/dist/chunks/{aiApi-CVPzFTXa.js.map → aiApi-DMGz-RPM.js.map} +1 -1
- package/dist/chunks/{applicationAnalyticsApi-B8AhFYLr.js → applicationAnalyticsApi-Bwa75Fzd.js} +2 -2
- package/dist/chunks/{applicationAnalyticsApi-B8AhFYLr.js.map → applicationAnalyticsApi-Bwa75Fzd.js.map} +1 -1
- package/dist/chunks/{applicationAnalyticsApi-Ce_1qOk-.js → applicationAnalyticsApi-CLBqRPfN.js} +2 -2
- package/dist/chunks/{applicationAnalyticsApi-Ce_1qOk-.js.map → applicationAnalyticsApi-CLBqRPfN.js.map} +1 -1
- package/dist/chunks/{groupsApi-BgCk2fsp.js → groupsApi-QzXI-5xu.js} +2 -2
- package/dist/chunks/{groupsApi-BgCk2fsp.js.map → groupsApi-QzXI-5xu.js.map} +1 -1
- package/dist/chunks/{groupsApi-BIbG665N.js → groupsApi-hB9kSWEd.js} +2 -2
- package/dist/chunks/{groupsApi-BIbG665N.js.map → groupsApi-hB9kSWEd.js.map} +1 -1
- package/dist/chunks/{index-Cb3LotuT.js → index--NGcBYUu.js} +3 -3
- package/dist/chunks/{index-Cb3LotuT.js.map → index--NGcBYUu.js.map} +1 -1
- package/dist/chunks/{index-C33zcyF4.js → index--aPwOFjF.js} +2 -2
- package/dist/chunks/{index-C33zcyF4.js.map → index--aPwOFjF.js.map} +1 -1
- package/dist/chunks/{index-sMr9qND_.js → index-0VrOtwP0.js} +2 -2
- package/dist/chunks/{index-sMr9qND_.js.map → index-0VrOtwP0.js.map} +1 -1
- package/dist/chunks/{index-DDKetfKq.js → index-37U271aw.js} +2 -2
- package/dist/chunks/{index-DDKetfKq.js.map → index-37U271aw.js.map} +1 -1
- package/dist/chunks/{index-B9fS7ir6.js → index-B7qZTuQ-.js} +2 -2
- package/dist/chunks/{index-B9fS7ir6.js.map → index-B7qZTuQ-.js.map} +1 -1
- package/dist/chunks/{index-CdjBY7L8.js → index-Bedzmqr-.js} +2 -2
- package/dist/chunks/{index-CdjBY7L8.js.map → index-Bedzmqr-.js.map} +1 -1
- package/dist/chunks/{index-CHG_O1fS.js → index-Betxo5g5.js} +2 -2
- package/dist/chunks/{index-CHG_O1fS.js.map → index-Betxo5g5.js.map} +1 -1
- package/dist/chunks/{index-jiGu-H8x.js → index-BmaJz475.js} +2 -2
- package/dist/chunks/{index-jiGu-H8x.js.map → index-BmaJz475.js.map} +1 -1
- package/dist/chunks/{index-C53JoVNk.js → index-Buhqag3v.js} +2 -2
- package/dist/chunks/{index-C53JoVNk.js.map → index-Buhqag3v.js.map} +1 -1
- package/dist/chunks/{index-DO0Rw7hX.js → index-C3VxlfKq.js} +2 -2
- package/dist/chunks/{index-DO0Rw7hX.js.map → index-C3VxlfKq.js.map} +1 -1
- package/dist/chunks/{index-CSQ60fpG.js → index-CgpRo8Oe.js} +2 -2
- package/dist/chunks/{index-CSQ60fpG.js.map → index-CgpRo8Oe.js.map} +1 -1
- package/dist/chunks/{index-B-e-ELsf.js → index-DOY0w8Iu.js} +8 -8
- package/dist/chunks/{index-B-e-ELsf.js.map → index-DOY0w8Iu.js.map} +1 -1
- package/dist/chunks/{index-D2REDIRX.js → index-DwuvIOrQ.js} +2 -2
- package/dist/chunks/{index-D2REDIRX.js.map → index-DwuvIOrQ.js.map} +1 -1
- package/dist/chunks/{index-DCcl7Qof.js → index-DzedSLdI.js} +2 -2
- package/dist/chunks/{index-DCcl7Qof.js.map → index-DzedSLdI.js.map} +1 -1
- package/dist/chunks/{index-2wUhd9Lu.js → index-IgLVXPg8.js} +10 -10
- package/dist/chunks/index-IgLVXPg8.js.map +1 -0
- package/dist/chunks/{index-CwSaRXXg.js → index-lpIzhufD.js} +1916 -1900
- package/dist/chunks/index-lpIzhufD.js.map +1 -0
- package/dist/chunks/{index-Cuwn2q-f.js → index-mLUKwbGl.js} +4 -4
- package/dist/chunks/{index-Cuwn2q-f.js.map → index-mLUKwbGl.js.map} +1 -1
- package/dist/chunks/{index-B0mk2tNY.js → index-tO6MMIFB.js} +2 -2
- package/dist/chunks/{index-B0mk2tNY.js.map → index-tO6MMIFB.js.map} +1 -1
- package/dist/chunks/{tenantIconMap-BpNANQ5s.js → tenantIconMap-BQD9byc8.js} +2 -2
- package/dist/chunks/{tenantIconMap-BpNANQ5s.js.map → tenantIconMap-BQD9byc8.js.map} +1 -1
- package/dist/chunks/{tenantIconMap-DtdUgvJO.js → tenantIconMap-CTMuSt18.js} +2 -2
- package/dist/chunks/{tenantIconMap-DtdUgvJO.js.map → tenantIconMap-CTMuSt18.js.map} +1 -1
- package/dist/chunks/{ticketingApi-Bu4rKwLl.js → ticketingApi-BNIdox5t.js} +2 -2
- package/dist/chunks/{ticketingApi-Bu4rKwLl.js.map → ticketingApi-BNIdox5t.js.map} +1 -1
- package/dist/chunks/{ticketingApi-r4Avm9iS.js → ticketingApi-J0vC_t7r.js} +2 -2
- package/dist/chunks/{ticketingApi-r4Avm9iS.js.map → ticketingApi-J0vC_t7r.js.map} +1 -1
- package/dist/chunks/{useAccessRequests-B9bF4swg.js → useAccessRequests-DCNNLnxk.js} +3 -3
- package/dist/chunks/{useAccessRequests-B9bF4swg.js.map → useAccessRequests-DCNNLnxk.js.map} +1 -1
- package/dist/chunks/{useAccessRequests-JyPUX3Om.js → useAccessRequests-DT7X4FAK.js} +2 -2
- package/dist/chunks/{useAccessRequests-JyPUX3Om.js.map → useAccessRequests-DT7X4FAK.js.map} +1 -1
- package/dist/chunks/{useUserAccessRequests-DjPQenC2.js → useUserAccessRequests-BYbmG4c7.js} +2 -2
- package/dist/chunks/{useUserAccessRequests-DjPQenC2.js.map → useUserAccessRequests-BYbmG4c7.js.map} +1 -1
- package/dist/chunks/{useUserAccessRequests-BB6FHW14.js → useUserAccessRequests-CylKxRN6.js} +2 -2
- package/dist/chunks/{useUserAccessRequests-BB6FHW14.js.map → useUserAccessRequests-CylKxRN6.js.map} +1 -1
- package/dist/hooks/useCollapsibleState.d.ts +5 -2
- package/dist/hooks/useCollapsibleState.d.ts.map +1 -1
- package/dist/i18n/config.d.ts +1 -0
- package/dist/i18n/config.d.ts.map +1 -1
- package/dist/main.d.ts +0 -1
- package/dist/main.d.ts.map +1 -1
- package/dist/smartstack.cjs +1 -1
- package/dist/smartstack.js +1 -1
- package/dist/utils/permissions.d.ts.map +1 -1
- package/package.json +1 -1
- package/dist/chunks/index-2wUhd9Lu.js.map +0 -1
- package/dist/chunks/index-CwSaRXXg.js.map +0 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("react/jsx-runtime"),N=require("react"),K=require("react-i18next"),L=require("react-router-dom"),T=require("./index-
|
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("react/jsx-runtime"),N=require("react"),K=require("react-i18next"),L=require("react-router-dom"),T=require("./index-IgLVXPg8.js"),n=require("lucide-react"),V=[{id:"general",labelKey:"settings.tabs.general",icon:n.Settings,route:"/administration/configuration/settings/general"},{id:"file-upload",labelKey:"settings.tabs.fileUpload",icon:n.Upload,route:"/administration/configuration/settings/file-upload"},{id:"legal-file",labelKey:"settings.tabs.legalFile",icon:n.FileText,route:"/administration/configuration/settings/legal-file"}];function M(){const{t}=K.useTranslation("admin"),w=L.useNavigate(),y=L.useLocation(),[x,h]=N.useState([]),[m,u]=N.useState(!0),g=N.useCallback(async()=>{try{u(!0);const v=await T.adminApi.settings.getAll();h(v)}catch(v){console.error("Failed to load settings:",v)}finally{u(!1)}},[]);N.useEffect(()=>{g()},[g]);const k=()=>{const v=y.pathname;return V.find(p=>v.includes(p.id))?.id||"general"},S=v=>{const b=V.find(p=>p.id===v);b&&w(b.route)};return m?e.jsx("div",{className:"flex items-center justify-center min-h-[400px]",children:e.jsx(n.Loader2,{className:"w-8 h-8 animate-spin text-[var(--color-accent-600)]"})}):e.jsxs("div",{className:"space-y-6",children:[e.jsx(T.Breadcrumb,{items:[{label:t("header.title"),href:"/administration"},{label:t("settings.title","Settings")}]}),e.jsx("div",{className:"flex items-center justify-between",children:e.jsxs("div",{children:[e.jsx("h1",{className:"text-2xl font-bold text-[var(--text-primary)]",children:t("settings.title")}),e.jsx("p",{className:"text-[var(--text-secondary)] mt-1",children:t("settings.subtitle")})]})}),e.jsx("div",{className:"border-b border-[var(--border-color)]",children:e.jsx("nav",{className:"-mb-px flex space-x-8","aria-label":"Tabs",children:V.map(v=>{const b=v.icon,p=k()===v.id;return e.jsxs("button",{onClick:()=>S(v.id),className:`
|
|
2
2
|
group inline-flex items-center py-4 px-1 border-b-2 font-medium text-sm transition-colors
|
|
3
3
|
${p?"border-[var(--color-accent-500)] text-[var(--color-accent-600)]":"border-transparent text-[var(--text-secondary)] hover:text-[var(--text-primary)] hover:border-[var(--border-color)]"}
|
|
4
4
|
`,children:[e.jsx(b,{className:`mr-2 h-5 w-5 ${p?"text-[var(--color-accent-500)]":"text-[var(--text-tertiary)] group-hover:text-[var(--text-secondary)]"}`}),t(v.labelKey)]},v.id)})})}),e.jsx(L.Outlet,{context:{settings:x,refreshSettings:g}})]})}function U(){const{settings:t,refreshSettings:w}=L.useOutletContext(),[y,x]=N.useState({}),[h,m]=N.useState(null),[u,g]=N.useState(null),[k,S]=N.useState([]),[v,b]=N.useState(!1),p=(o,c,i)=>{x(f=>({...f,[`${o}.${c}`]:i}))},D=o=>{const c=`${o.category}.${o.key}`;return y[c]!==void 0&&y[c]!==o.value},C=o=>{const c=`${o.category}.${o.key}`;return y[c]??o.value},$=o=>`${o.category}.${o.key}`;return{settings:t,editedValues:y,saving:h,historyModal:u,history:k,loadingHistory:v,handleValueChange:p,hasChanges:D,getDisplayValue:C,getEditKey:$,handleSave:async o=>{const c=$(o),i=y[c];if(!(i===void 0||i===o.value))try{m(c),await T.adminApi.settings.update(o.category,o.key,i),x(f=>{const A={...f};return delete A[c],A}),await w()}catch(f){console.error("Failed to save setting:",f)}finally{m(null)}},handleReset:async o=>{const c=$(o);try{m(c),await T.adminApi.settings.reset(o.category,o.key),x(i=>{const f={...i};return delete f[c],f}),await w()}catch(i){console.error("Failed to reset setting:",i)}finally{m(null)}},openHistory:async(o,c)=>{g({category:o,key:c}),b(!0);try{const i=await T.adminApi.settings.getHistory(o,c);S(i)}catch(i){console.error("Failed to load history:",i)}finally{b(!1)}},closeHistory:()=>g(null)}}function R({category:t,settingKey:w,history:y,loading:x,onClose:h}){const{t:m}=K.useTranslation("admin");return e.jsx("div",{className:"fixed inset-0 z-50 overflow-y-auto",children:e.jsxs("div",{className:"flex items-center justify-center min-h-screen px-4",children:[e.jsx("button",{type:"button",className:"fixed inset-0 bg-black/50",onClick:h,"aria-label":"Close modal"}),e.jsxs("div",{className:"relative rounded-[var(--radius-card)] bg-[var(--bg-card)] shadow-xl max-w-2xl w-full p-6",children:[e.jsx("h3",{className:"text-lg font-semibold text-[var(--text-primary)] mb-4",children:m("settings.historyTitle",{key:`${t}.${w}`})}),x&&e.jsx("div",{className:"flex justify-center py-8",children:e.jsx(n.Loader2,{className:"w-8 h-8 animate-spin text-[var(--color-accent-600)]"})}),!x&&y.length===0&&e.jsx("p",{className:"text-center py-8 text-[var(--text-secondary)]",children:m("settings.noHistory")}),!x&&y.length>0&&e.jsx("div",{className:"space-y-3 max-h-96 overflow-y-auto",children:y.map(u=>e.jsxs("div",{className:"p-3 bg-[var(--bg-secondary)] rounded-lg",children:[e.jsxs("div",{className:"flex justify-between text-sm",children:[e.jsx("span",{className:"text-[var(--text-secondary)]",children:u.changedByUserName||m("settings.system")}),e.jsx("span",{className:"text-[var(--text-tertiary)]",children:new Date(u.changedAt).toLocaleString()})]}),e.jsxs("div",{className:"mt-2 text-sm",children:[e.jsx("span",{className:"text-[var(--error-text)] line-through",children:u.oldValue}),e.jsx("span",{className:"mx-2 text-[var(--text-tertiary)]",children:"→"}),e.jsx("span",{className:"text-[var(--success-text)]",children:u.newValue})]})]},u.id))}),e.jsx("div",{className:"mt-6 flex justify-end",children:e.jsx("button",{onClick:h,className:"px-4 py-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors",children:m("common.cancel")})})]})]})})}function I(){const{t}=K.useTranslation("admin"),{settings:w,saving:y,historyModal:x,history:h,loadingHistory:m,handleValueChange:u,hasChanges:g,getDisplayValue:k,getEditKey:S,handleSave:v,handleReset:b,openHistory:p,closeHistory:D}=U(),C=w.filter(s=>s.category==="Session"),$=w.filter(s=>s.category==="General"),E=s=>s==="Database"?"bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]":s==="AppSettings"?"bg-purple-500/10 text-purple-600 dark:text-purple-400":"bg-[var(--bg-secondary)] text-[var(--text-tertiary)]",F=(s,o,c)=>s.valueType==="Bool"?e.jsxs("label",{className:"relative inline-flex items-center cursor-pointer",children:[e.jsx("input",{type:"checkbox",checked:o==="true",onChange:i=>u(s.category,s.key,i.target.checked?"true":"false"),disabled:!s.isEditable||c,className:"sr-only peer"}),e.jsx("div",{className:"w-11 h-6 bg-[var(--bg-tertiary)] peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-[var(--color-accent-500)]/30 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-[var(--border-color)] after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[var(--color-accent-600)] peer-disabled:opacity-50 peer-disabled:cursor-not-allowed"})]}):s.valueType==="Int"||s.valueType==="Decimal"?e.jsx("input",{type:"number",value:o,onChange:i=>u(s.category,s.key,i.target.value),disabled:!s.isEditable||c,className:"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]"}):e.jsx("input",{type:s.isSensitive?"password":"text",value:s.isSensitive?"********":o,onChange:i=>u(s.category,s.key,i.target.value),disabled:!s.isEditable||c||s.isSensitive,className:"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]"}),a=s=>{const o=S(s),c=y===o,i=g(s),f=k(s);return e.jsxs("div",{className:"flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0",children:[e.jsxs("div",{className:"flex-1 min-w-0 pr-4",children:[e.jsxs("div",{className:"flex items-center gap-2",children:[e.jsx("span",{className:"font-medium text-[var(--text-primary)]",children:t(`settings.keys.${s.category}.${s.key}`,s.key)}),!s.isEditable&&e.jsx("span",{className:"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded",children:t("settings.readonly")}),e.jsx("span",{className:`px-2 py-0.5 text-xs rounded ${E(s.source)}`,children:s.source})]}),s.description&&e.jsx("p",{className:"text-sm text-[var(--text-secondary)] mt-1",children:t(`settings.descriptions.${s.category}.${s.key}`,s.description)}),s.modifiedAt&&e.jsxs("p",{className:"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1",children:[e.jsx(n.Clock,{className:"w-3 h-3"}),t("settings.modifiedBy",{user:s.modifiedByUserName,date:new Date(s.modifiedAt).toLocaleString()})]})]}),e.jsxs("div",{className:"flex items-center gap-2",children:[F(s,f,c),s.isEditable&&s.source==="Database"&&e.jsxs(e.Fragment,{children:[e.jsx("button",{onClick:()=>v(s),disabled:!i||c,className:"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors",title:t("common.save"),children:c?e.jsx(n.Loader2,{className:"w-4 h-4 animate-spin"}):e.jsx(n.Save,{className:"w-4 h-4"})}),e.jsx("button",{onClick:()=>b(s),disabled:s.value===s.defaultValue||c,className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors",title:t("settings.resetToDefault"),children:e.jsx(n.RotateCcw,{className:"w-4 h-4"})}),e.jsx("button",{onClick:()=>p(s.category,s.key),className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors",title:t("settings.viewHistory"),children:e.jsx(n.History,{className:"w-4 h-4"})})]})]})]},o)};return e.jsxs("div",{className:"space-y-6",children:[C.length>0&&e.jsxs("div",{className:"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm",children:[e.jsx("div",{className:"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]",children:e.jsxs("div",{className:"flex items-center gap-3",children:[e.jsx("div",{className:"p-2 rounded-lg bg-[var(--color-accent-500)]/20",children:e.jsx(n.Clock,{className:"w-5 h-5 text-[var(--color-accent-600)]"})}),e.jsxs("div",{children:[e.jsx("h3",{className:"text-lg font-semibold text-[var(--text-primary)]",children:t("settings.sections.session")}),e.jsx("p",{className:"text-sm text-[var(--text-secondary)]",children:t("settings.sections.sessionDescription")})]})]})}),e.jsx("div",{className:"px-6 py-2",children:C.map(a)})]}),$.length>0&&e.jsxs("div",{className:"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm",children:[e.jsx("div",{className:"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]",children:e.jsxs("div",{className:"flex items-center gap-3",children:[e.jsx("div",{className:"p-2 rounded-lg bg-[var(--color-accent-500)]/20",children:e.jsx(n.Settings,{className:"w-5 h-5 text-[var(--color-accent-600)]"})}),e.jsxs("div",{children:[e.jsx("h3",{className:"text-lg font-semibold text-[var(--text-primary)]",children:t("settings.sections.general")}),e.jsx("p",{className:"text-sm text-[var(--text-secondary)]",children:t("settings.sections.generalDescription")})]})]})}),e.jsx("div",{className:"px-6 py-2",children:$.map(a)})]}),C.length===0&&$.length===0&&e.jsx("div",{className:"text-center py-12 text-[var(--text-secondary)]",children:t("settings.noSettings")}),x&&e.jsx(R,{category:x.category,settingKey:x.key,history:h,loading:m,onClose:D})]})}function q(){const{t}=K.useTranslation("admin"),{settings:w,saving:y,historyModal:x,history:h,loadingHistory:m,handleValueChange:u,hasChanges:g,getDisplayValue:k,getEditKey:S,handleSave:v,handleReset:b,openHistory:p,closeHistory:D}=U(),C=w.filter(a=>a.category==="FileUpload"),$=a=>{try{return JSON.parse(a)}catch{return[]}},E=(a,s,o)=>a.valueType==="Int"?e.jsxs("div",{className:"flex items-center gap-2",children:[e.jsx("input",{type:"number",value:s,onChange:c=>u(a.category,a.key,c.target.value),disabled:!a.isEditable||o,className:"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]"}),a.key==="MaxFileSizeMB"&&e.jsx("span",{className:"text-sm text-[var(--text-secondary)]",children:"MB"})]}):e.jsx("input",{type:a.isSensitive?"password":"text",value:a.isSensitive?"********":s,onChange:c=>u(a.category,a.key,c.target.value),disabled:!a.isEditable||o||a.isSensitive,className:"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]"}),F=a=>{const s=S(a),o=y===s,c=g(a),i=k(a);if(a.valueType==="Json"&&a.key==="AllowedExtensions"){const f=$(i);return e.jsxs("div",{className:"py-4 border-b border-[var(--border-color)] last:border-0",children:[e.jsxs("div",{className:"flex items-center justify-between mb-3",children:[e.jsxs("div",{children:[e.jsxs("div",{className:"flex items-center gap-2",children:[e.jsx("span",{className:"font-medium text-[var(--text-primary)]",children:t(`settings.keys.${a.category}.${a.key}`,a.key)}),!a.isEditable&&e.jsx("span",{className:"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded",children:t("settings.readonly")}),e.jsx("span",{className:"px-2 py-0.5 text-xs bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)] rounded",children:a.source})]}),a.description&&e.jsx("p",{className:"text-sm text-[var(--text-secondary)] mt-1",children:t(`settings.descriptions.${a.category}.${a.key}`,a.description)})]}),a.isEditable&&a.source==="Database"&&e.jsx("button",{onClick:()=>p(a.category,a.key),className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors",title:t("settings.viewHistory"),children:e.jsx(n.History,{className:"w-4 h-4"})})]}),e.jsx("div",{className:"flex flex-wrap gap-2",children:f.map(A=>e.jsx("span",{className:"px-3 py-1 bg-[var(--bg-secondary)] text-[var(--text-secondary)] rounded-full text-sm",children:A},A))})]},s)}return e.jsxs("div",{className:"flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0",children:[e.jsxs("div",{className:"flex-1 min-w-0 pr-4",children:[e.jsxs("div",{className:"flex items-center gap-2",children:[e.jsx("span",{className:"font-medium text-[var(--text-primary)]",children:t(`settings.keys.${a.category}.${a.key}`,a.key)}),!a.isEditable&&e.jsx("span",{className:"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded",children:t("settings.readonly")}),e.jsx("span",{className:`px-2 py-0.5 text-xs rounded ${a.source==="Database"?"bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]":a.source==="AppSettings"?"bg-purple-500/10 text-purple-600 dark:text-purple-400":"bg-[var(--bg-secondary)] text-[var(--text-tertiary)]"}`,children:a.source})]}),a.description&&e.jsx("p",{className:"text-sm text-[var(--text-secondary)] mt-1",children:t(`settings.descriptions.${a.category}.${a.key}`,a.description)}),a.modifiedAt&&e.jsxs("p",{className:"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1",children:[e.jsx(n.Clock,{className:"w-3 h-3"}),t("settings.modifiedBy",{user:a.modifiedByUserName,date:new Date(a.modifiedAt).toLocaleString()})]})]}),e.jsxs("div",{className:"flex items-center gap-2",children:[E(a,i,o),a.isEditable&&a.source==="Database"&&e.jsxs(e.Fragment,{children:[e.jsx("button",{onClick:()=>v(a),disabled:!c||o,className:"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors",title:t("common.save"),children:o?e.jsx(n.Loader2,{className:"w-4 h-4 animate-spin"}):e.jsx(n.Save,{className:"w-4 h-4"})}),e.jsx("button",{onClick:()=>b(a),disabled:a.value===a.defaultValue||o,className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors",title:t("settings.resetToDefault"),children:e.jsx(n.RotateCcw,{className:"w-4 h-4"})}),e.jsx("button",{onClick:()=>p(a.category,a.key),className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors",title:t("settings.viewHistory"),children:e.jsx(n.History,{className:"w-4 h-4"})})]})]})]},s)};return e.jsxs("div",{className:"space-y-6",children:[e.jsxs("div",{className:"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm",children:[e.jsx("div",{className:"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]",children:e.jsxs("div",{className:"flex items-center gap-3",children:[e.jsx("div",{className:"p-2 rounded-lg bg-[var(--color-accent-500)]/20",children:e.jsx(n.Upload,{className:"w-5 h-5 text-[var(--color-accent-600)]"})}),e.jsxs("div",{children:[e.jsx("h3",{className:"text-lg font-semibold text-[var(--text-primary)]",children:t("settings.sections.fileUpload")}),e.jsx("p",{className:"text-sm text-[var(--text-secondary)]",children:t("settings.sections.fileUploadDescription")})]})]})}),e.jsx("div",{className:"px-6 py-2",children:C.length>0?C.map(F):e.jsx("div",{className:"text-center py-8 text-[var(--text-secondary)]",children:t("settings.noSettings")})})]}),x&&e.jsx(R,{category:x.category,settingKey:x.key,history:h,loading:m,onClose:D})]})}function P(){const{t}=K.useTranslation("admin"),{settings:w,refreshSettings:y}=L.useOutletContext(),[x,h]=N.useState({}),[m,u]=N.useState(null),[g,k]=N.useState(null),[S,v]=N.useState([]),[b,p]=N.useState(!1),D=w.filter(r=>r.category==="LegalFile"),C=(r,l,d)=>{h(j=>({...j,[`${r}.${l}`]:d}))},$=r=>{const l=`${r.category}.${r.key}`;return x[l]!==void 0&&x[l]!==r.value},E=r=>{const l=`${r.category}.${r.key}`;return x[l]??r.value},F=async r=>{const l=`${r.category}.${r.key}`,d=x[l];if(!(d===void 0||d===r.value))try{u(l),await T.adminApi.settings.update(r.category,r.key,d),h(j=>{const H={...j};return delete H[l],H}),await y()}catch(j){console.error("Failed to save setting:",j)}finally{u(null)}},a=async r=>{const l=`${r.category}.${r.key}`;try{u(l),await T.adminApi.settings.reset(r.category,r.key),h(d=>{const j={...d};return delete j[l],j}),await y()}catch(d){console.error("Failed to reset setting:",d)}finally{u(null)}},s=async(r,l)=>{k({category:r,key:l}),p(!0);try{const d=await T.adminApi.settings.getHistory(r,l);v(d)}catch(d){console.error("Failed to load history:",d)}finally{p(!1)}},o=r=>{const l=Math.floor(r/365),d=r%365;return d===0?`${l} ${t("settings.years")}`:`${l} ${t("settings.years")} ${d} ${t("settings.days")}`},c=r=>!r.isEditable&&(r.key==="RetentionDays"||r.key==="EnableWorm"),i=(r,l,d,j)=>r.valueType==="Bool"?e.jsxs("label",{className:"relative inline-flex cursor-pointer",children:[e.jsx("span",{className:"sr-only",children:r.key}),e.jsx("input",{type:"checkbox",checked:l==="true",onChange:H=>j(r.category,r.key,H.target.checked?"true":"false"),disabled:!r.isEditable||d,className:"sr-only peer","aria-label":r.key}),e.jsx("div",{className:"w-11 h-6 bg-[var(--bg-tertiary)] peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-[var(--color-accent-500)]/30 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-[var(--border-color)] after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[var(--color-accent-600)] peer-disabled:opacity-50 peer-disabled:cursor-not-allowed"})]}):r.valueType==="Int"?e.jsxs("div",{className:"flex items-center gap-2",children:[e.jsx("input",{type:"number",value:l,onChange:H=>j(r.category,r.key,H.target.value),disabled:!r.isEditable||d,className:"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]"}),(r.key==="RetentionDays"||r.key==="ArchiveAfterDays")&&e.jsxs("span",{className:"text-sm text-[var(--text-secondary)]",children:["(",o(parseInt(l)||0),")"]})]}):e.jsx("input",{type:r.isSensitive?"password":"text",value:r.isSensitive?"********":l,onChange:H=>j(r.category,r.key,H.target.value),disabled:!r.isEditable||d||r.isSensitive,className:"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]"}),f=r=>r==="Database"?"bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]":r==="AppSettings"?"bg-purple-500/10 text-purple-600 dark:text-purple-400":"bg-[var(--bg-secondary)] text-[var(--text-tertiary)]",A=r=>{const l=`${r.category}.${r.key}`,d=m===l,j=$(r),H=E(r),B=c(r);return e.jsxs("div",{className:`flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0 ${B?"bg-[var(--warning-bg)] -mx-6 px-6":""}`,children:[e.jsxs("div",{className:"flex-1 min-w-0 pr-4",children:[e.jsxs("div",{className:"flex items-center gap-2",children:[e.jsx("span",{className:"font-medium text-[var(--text-primary)]",children:t(`settings.keys.${r.category}.${r.key}`,r.key)}),B&&e.jsxs("span",{className:"px-2 py-0.5 text-xs bg-[var(--warning-bg)] text-[var(--warning-text)] border border-[var(--warning-border)] rounded flex items-center gap-1",children:[e.jsx(n.Shield,{className:"w-3 h-3"}),t("settings.legallyProtected")]}),!r.isEditable&&!B&&e.jsx("span",{className:"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded",children:t("settings.readonly")}),e.jsx("span",{className:`px-2 py-0.5 text-xs rounded ${f(r.source)}`,children:r.source})]}),r.description&&e.jsx("p",{className:"text-sm text-[var(--text-secondary)] mt-1",children:t(`settings.descriptions.${r.category}.${r.key}`,r.description)}),r.modifiedAt&&e.jsxs("p",{className:"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1",children:[e.jsx(n.Clock,{className:"w-3 h-3"}),t("settings.modifiedBy",{user:r.modifiedByUserName,date:new Date(r.modifiedAt).toLocaleString()})]})]}),e.jsxs("div",{className:"flex items-center gap-2",children:[i(r,H,d,C),r.isEditable&&r.source==="Database"&&e.jsxs(e.Fragment,{children:[e.jsx("button",{onClick:()=>F(r),disabled:!j||d,className:"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors",title:t("common.save"),children:d?e.jsx(n.Loader2,{className:"w-4 h-4 animate-spin"}):e.jsx(n.Save,{className:"w-4 h-4"})}),e.jsx("button",{onClick:()=>a(r),disabled:r.value===r.defaultValue||d,className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors",title:t("settings.resetToDefault"),children:e.jsx(n.RotateCcw,{className:"w-4 h-4"})}),e.jsx("button",{onClick:()=>s(r.category,r.key),className:"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors",title:t("settings.viewHistory"),children:e.jsx(n.History,{className:"w-4 h-4"})})]})]})]},l)};return e.jsxs("div",{className:"space-y-6",children:[e.jsx("div",{className:"bg-[var(--warning-bg)] border border-[var(--warning-border)] rounded-[var(--radius-card)] p-4",children:e.jsxs("div",{className:"flex items-start gap-3",children:[e.jsx(n.AlertTriangle,{className:"w-5 h-5 text-[var(--warning-text)] flex-shrink-0 mt-0.5"}),e.jsxs("div",{children:[e.jsx("h4",{className:"font-medium text-[var(--warning-text)]",children:t("settings.legalWarningTitle")}),e.jsx("p",{className:"text-sm text-[var(--warning-text)] opacity-90 mt-1",children:t("settings.legalWarningDescription")})]})]})}),e.jsxs("div",{className:"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm",children:[e.jsx("div",{className:"px-6 py-4 bg-gradient-to-r from-amber-500/10 to-amber-600/5 border-b border-[var(--border-color)]",children:e.jsxs("div",{className:"flex items-center gap-3",children:[e.jsx("div",{className:"p-2 rounded-lg bg-amber-500/20",children:e.jsx(n.FileText,{className:"w-5 h-5 text-amber-600"})}),e.jsxs("div",{children:[e.jsx("h3",{className:"text-lg font-semibold text-[var(--text-primary)]",children:t("settings.sections.legalFile")}),e.jsx("p",{className:"text-sm text-[var(--text-secondary)]",children:t("settings.sections.legalFileDescription")})]})]})}),e.jsx("div",{className:"px-6 py-2",children:D.length>0?D.map(A):e.jsx("div",{className:"text-center py-8 text-[var(--text-secondary)]",children:t("settings.noSettings")})})]}),g&&e.jsx("div",{className:"fixed inset-0 z-50 overflow-y-auto",children:e.jsxs("div",{className:"flex items-center justify-center min-h-screen px-4",children:[e.jsx("button",{type:"button",className:"fixed inset-0 bg-black/50",onClick:()=>k(null),"aria-label":"Close modal"}),e.jsxs("div",{className:"relative rounded-[var(--radius-card)] bg-[var(--bg-card)] shadow-xl max-w-2xl w-full p-6",children:[e.jsx("h3",{className:"text-lg font-semibold text-[var(--text-primary)] mb-4",children:t("settings.historyTitle",{key:`${g.category}.${g.key}`})}),b&&e.jsx("div",{className:"flex justify-center py-8",children:e.jsx(n.Loader2,{className:"w-8 h-8 animate-spin text-[var(--color-accent-600)]"})}),!b&&S.length===0&&e.jsx("p",{className:"text-center py-8 text-[var(--text-secondary)]",children:t("settings.noHistory")}),!b&&S.length>0&&e.jsx("div",{className:"space-y-3 max-h-96 overflow-y-auto",children:S.map(r=>e.jsxs("div",{className:"p-3 bg-[var(--bg-secondary)] rounded-lg",children:[e.jsxs("div",{className:"flex justify-between text-sm",children:[e.jsx("span",{className:"text-[var(--text-secondary)]",children:r.changedByUserName||t("settings.system")}),e.jsx("span",{className:"text-[var(--text-tertiary)]",children:new Date(r.changedAt).toLocaleString()})]}),e.jsxs("div",{className:"mt-2 text-sm",children:[e.jsx("span",{className:"text-[var(--error-text)] line-through",children:r.oldValue}),e.jsx("span",{className:"mx-2 text-[var(--text-tertiary)]",children:"→"}),e.jsx("span",{className:"text-[var(--success-text)]",children:r.newValue})]})]},r.id))}),e.jsx("div",{className:"mt-6 flex justify-end",children:e.jsx("button",{onClick:()=>k(null),className:"px-4 py-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors",children:t("common.cancel")})})]})]})})]})}exports.FileUploadSettingsSection=q;exports.GeneralSettingsSection=I;exports.LegalFileSettingsSection=P;exports.SettingsPage=M;
|
|
5
|
-
//# sourceMappingURL=index-
|
|
5
|
+
//# sourceMappingURL=index-Betxo5g5.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-CHG_O1fS.js","sources":["../../src/pages/platform/administration/configuration/settings/SettingsPage.tsx","../../src/hooks/useSettingsSection.ts","../../src/components/platform/administration/settings/SettingsHistoryModal.tsx","../../src/pages/platform/administration/configuration/settings/GeneralSettingsSection.tsx","../../src/pages/platform/administration/configuration/settings/FileUploadSettingsSection.tsx","../../src/pages/platform/administration/configuration/settings/LegalFileSettingsSection.tsx"],"sourcesContent":["import { useState, useEffect, useCallback } from 'react';\r\nimport type { ReactElement } from 'react';\r\nimport { useTranslation } from 'react-i18next';\r\nimport { useNavigate, useLocation, Outlet } from 'react-router-dom';\r\nimport { Breadcrumb } from '@/components/ui/Breadcrumb';\r\nimport { Settings, Upload, FileText, Loader2 } from 'lucide-react';\r\nimport { adminApi, type SettingViewModel } from '@/services/api/adminApi';\r\n\r\ninterface TabConfig {\r\n id: string;\r\n labelKey: string;\r\n icon: React.ElementType;\r\n route: string;\r\n}\r\n\r\nconst TABS: TabConfig[] = [\r\n { id: 'general', labelKey: 'settings.tabs.general', icon: Settings, route: '/administration/configuration/settings/general' },\r\n { id: 'file-upload', labelKey: 'settings.tabs.fileUpload', icon: Upload, route: '/administration/configuration/settings/file-upload' },\r\n { id: 'legal-file', labelKey: 'settings.tabs.legalFile', icon: FileText, route: '/administration/configuration/settings/legal-file' }\r\n];\r\n\r\nexport function SettingsPage(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const navigate = useNavigate();\r\n const location = useLocation();\r\n const [settings, setSettings] = useState<SettingViewModel[]>([]);\r\n const [loading, setLoading] = useState(true);\r\n\r\n const loadSettings = useCallback(async () => {\r\n try {\r\n setLoading(true);\r\n const result = await adminApi.settings.getAll();\r\n setSettings(result);\r\n } catch (err) {\r\n console.error('Failed to load settings:', err);\r\n } finally {\r\n setLoading(false);\r\n }\r\n }, []);\r\n\r\n useEffect(() => {\r\n loadSettings();\r\n }, [loadSettings]);\r\n\r\n const getCurrentTab = () => {\r\n const path = location.pathname;\r\n const tab = TABS.find(t => path.includes(t.id));\r\n return tab?.id || 'general';\r\n };\r\n\r\n const handleTabChange = (tabId: string) => {\r\n const tab = TABS.find(t => t.id === tabId);\r\n if (tab) {\r\n navigate(tab.route);\r\n }\r\n };\r\n\r\n if (loading) {\r\n return (\r\n <div className=\"flex items-center justify-center min-h-[400px]\">\r\n <Loader2 className=\"w-8 h-8 animate-spin text-[var(--color-accent-600)]\" />\r\n </div>\r\n );\r\n }\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n <Breadcrumb\r\n items={[\r\n { label: t('header.title'), href: '/administration' },\r\n { label: t('settings.title', 'Settings') }\r\n ]}\r\n />\r\n\r\n {/* Header */}\r\n <div className=\"flex items-center justify-between\">\r\n <div>\r\n <h1 className=\"text-2xl font-bold text-[var(--text-primary)]\">\r\n {t('settings.title')}\r\n </h1>\r\n <p className=\"text-[var(--text-secondary)] mt-1\">\r\n {t('settings.subtitle')}\r\n </p>\r\n </div>\r\n </div>\r\n\r\n {/* Tabs */}\r\n <div className=\"border-b border-[var(--border-color)]\">\r\n <nav className=\"-mb-px flex space-x-8\" aria-label=\"Tabs\">\r\n {TABS.map((tab) => {\r\n const Icon = tab.icon;\r\n const isActive = getCurrentTab() === tab.id;\r\n return (\r\n <button\r\n key={tab.id}\r\n onClick={() => handleTabChange(tab.id)}\r\n className={`\r\n group inline-flex items-center py-4 px-1 border-b-2 font-medium text-sm transition-colors\r\n ${isActive\r\n ? 'border-[var(--color-accent-500)] text-[var(--color-accent-600)]'\r\n : 'border-transparent text-[var(--text-secondary)] hover:text-[var(--text-primary)] hover:border-[var(--border-color)]'\r\n }\r\n `}\r\n >\r\n <Icon className={`mr-2 h-5 w-5 ${isActive ? 'text-[var(--color-accent-500)]' : 'text-[var(--text-tertiary)] group-hover:text-[var(--text-secondary)]'}`} />\r\n {t(tab.labelKey)}\r\n </button>\r\n );\r\n })}\r\n </nav>\r\n </div>\r\n\r\n {/* Content */}\r\n <Outlet context={{ settings, refreshSettings: loadSettings }} />\r\n </div>\r\n );\r\n}\r\n\r\nexport default SettingsPage;\r\n","import { useState } from 'react';\r\nimport { useOutletContext } from 'react-router-dom';\r\nimport { adminApi, type SettingViewModel, type SettingHistoryDto } from '@/services/api/adminApi';\r\n\r\ninterface SettingsContext {\r\n settings: SettingViewModel[];\r\n refreshSettings: () => Promise<void>;\r\n}\r\n\r\nexport function useSettingsSection(): {\n settings: SettingViewModel[];\n editedValues: Record<string, string>;\n saving: string | null;\n historyModal: { category: string; key: string } | null;\n history: SettingHistoryDto[];\n loadingHistory: boolean;\n handleValueChange: (category: string, key: string, value: string) => void;\n hasChanges: (setting: SettingViewModel) => boolean;\n getDisplayValue: (setting: SettingViewModel) => string;\n getEditKey: (setting: SettingViewModel) => string;\n handleSave: (setting: SettingViewModel) => Promise<void>;\n handleReset: (setting: SettingViewModel) => Promise<void>;\n openHistory: (category: string, key: string) => Promise<void>;\n closeHistory: () => void;\n} {\r\n const { settings, refreshSettings } = useOutletContext<SettingsContext>();\r\n const [editedValues, setEditedValues] = useState<Record<string, string>>({});\r\n const [saving, setSaving] = useState<string | null>(null);\r\n const [historyModal, setHistoryModal] = useState<{ category: string; key: string } | null>(null);\r\n const [history, setHistory] = useState<SettingHistoryDto[]>([]);\r\n const [loadingHistory, setLoadingHistory] = useState(false);\r\n\r\n const handleValueChange = (category: string, key: string, value: string) => {\r\n setEditedValues(prev => ({\r\n ...prev,\r\n [`${category}.${key}`]: value\r\n }));\r\n };\r\n\r\n const hasChanges = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] !== undefined && editedValues[editKey] !== setting.value;\r\n };\r\n\r\n const getDisplayValue = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] ?? setting.value;\r\n };\r\n\r\n const getEditKey = (setting: SettingViewModel) => `${setting.category}.${setting.key}`;\r\n\r\n const handleSave = async (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n const newValue = editedValues[editKey];\r\n if (newValue === undefined || newValue === setting.value) return;\r\n\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.update(setting.category, setting.key, newValue);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to save setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const handleReset = async (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.reset(setting.category, setting.key);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to reset setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const openHistory = async (category: string, key: string) => {\r\n setHistoryModal({ category, key });\r\n setLoadingHistory(true);\r\n try {\r\n const result = await adminApi.settings.getHistory(category, key);\r\n setHistory(result);\r\n } catch (err) {\r\n console.error('Failed to load history:', err);\r\n } finally {\r\n setLoadingHistory(false);\r\n }\r\n };\r\n\r\n const closeHistory = () => setHistoryModal(null);\r\n\r\n return {\r\n settings,\r\n editedValues,\r\n saving,\r\n historyModal,\r\n history,\r\n loadingHistory,\r\n handleValueChange,\r\n hasChanges,\r\n getDisplayValue,\r\n getEditKey,\r\n handleSave,\r\n handleReset,\r\n openHistory,\r\n closeHistory,\r\n };\r\n}\r\n\r\nexport type { SettingsContext };\r\n","import type { ReactElement } from 'react';\r\nimport { useTranslation } from 'react-i18next';\r\nimport { Loader2 } from 'lucide-react';\r\nimport type { SettingHistoryDto } from '@/services/api/adminApi';\r\n\r\ninterface SettingsHistoryModalProps {\r\n readonly category: string;\r\n readonly settingKey: string;\r\n readonly history: SettingHistoryDto[];\r\n readonly loading: boolean;\r\n readonly onClose: () => void;\r\n}\r\n\r\nexport function SettingsHistoryModal({ category, settingKey, history, loading, onClose }: SettingsHistoryModalProps): ReactElement {\r\n const { t } = useTranslation('admin');\r\n\r\n return (\r\n <div className=\"fixed inset-0 z-50 overflow-y-auto\">\r\n <div className=\"flex items-center justify-center min-h-screen px-4\">\r\n <button type=\"button\" className=\"fixed inset-0 bg-black/50\" onClick={onClose} aria-label=\"Close modal\" />\r\n <div className=\"relative rounded-[var(--radius-card)] bg-[var(--bg-card)] shadow-xl max-w-2xl w-full p-6\">\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)] mb-4\">\r\n {t('settings.historyTitle', { key: `${category}.${settingKey}` })}\r\n </h3>\r\n\r\n {loading && (\r\n <div className=\"flex justify-center py-8\">\r\n <Loader2 className=\"w-8 h-8 animate-spin text-[var(--color-accent-600)]\" />\r\n </div>\r\n )}\r\n {!loading && history.length === 0 && (\r\n <p className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noHistory')}\r\n </p>\r\n )}\r\n {!loading && history.length > 0 && (\r\n <div className=\"space-y-3 max-h-96 overflow-y-auto\">\r\n {history.map((entry) => (\r\n <div key={entry.id} className=\"p-3 bg-[var(--bg-secondary)] rounded-lg\">\r\n <div className=\"flex justify-between text-sm\">\r\n <span className=\"text-[var(--text-secondary)]\">\r\n {entry.changedByUserName || t('settings.system')}\r\n </span>\r\n <span className=\"text-[var(--text-tertiary)]\">\r\n {new Date(entry.changedAt).toLocaleString()}\r\n </span>\r\n </div>\r\n <div className=\"mt-2 text-sm\">\r\n <span className=\"text-[var(--error-text)] line-through\">{entry.oldValue}</span>\r\n <span className=\"mx-2 text-[var(--text-tertiary)]\">→</span>\r\n <span className=\"text-[var(--success-text)]\">{entry.newValue}</span>\r\n </div>\r\n </div>\r\n ))}\r\n </div>\r\n )}\r\n\r\n <div className=\"mt-6 flex justify-end\">\r\n <button\r\n onClick={onClose}\r\n className=\"px-4 py-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n >\r\n {t('common.cancel')}\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n );\r\n}\r\n","import type { ReactElement } from 'react';\nimport { useTranslation } from 'react-i18next';\r\nimport { Save, RotateCcw, Loader2, Clock, History, Settings as SettingsIcon } from 'lucide-react';\r\nimport type { SettingViewModel } from '@/services/api/adminApi';\r\nimport { useSettingsSection } from '@/hooks/useSettingsSection';\r\nimport { SettingsHistoryModal } from '@/components/platform/administration/settings/SettingsHistoryModal';\r\n\r\nexport function GeneralSettingsSection(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const {\r\n settings,\r\n saving,\r\n historyModal,\r\n history,\r\n loadingHistory,\r\n handleValueChange,\r\n hasChanges,\r\n getDisplayValue,\r\n getEditKey,\r\n handleSave,\r\n handleReset,\r\n openHistory,\r\n closeHistory,\r\n } = useSettingsSection();\r\n\r\n // Filter settings for Session and General categories\r\n const sessionSettings = settings.filter(s => s.category === 'Session');\r\n const generalSettings = settings.filter(s => s.category === 'General');\r\n\r\n const getSourceBadgeClass = (source: string): string => {\r\n if (source === 'Database') {\r\n return 'bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]';\r\n }\r\n if (source === 'AppSettings') {\r\n return 'bg-purple-500/10 text-purple-600 dark:text-purple-400';\r\n }\r\n return 'bg-[var(--bg-secondary)] text-[var(--text-tertiary)]';\r\n };\r\n\r\n const getSettingInput = (setting: SettingViewModel, displayValue: string, isEditing: boolean): React.ReactNode => {\r\n if (setting.valueType === 'Bool') {\r\n return (\r\n <label className=\"relative inline-flex items-center cursor-pointer\">\r\n <input\r\n type=\"checkbox\"\r\n checked={displayValue === 'true'}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.checked ? 'true' : 'false')}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"sr-only peer\"\r\n />\r\n <div className=\"w-11 h-6 bg-[var(--bg-tertiary)] peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-[var(--color-accent-500)]/30 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-[var(--border-color)] after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[var(--color-accent-600)] peer-disabled:opacity-50 peer-disabled:cursor-not-allowed\"></div>\r\n </label>\r\n );\r\n }\r\n if (setting.valueType === 'Int' || setting.valueType === 'Decimal') {\r\n return (\r\n <input\r\n type=\"number\"\r\n value={displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n }\r\n return (\r\n <input\r\n type={setting.isSensitive ? 'password' : 'text'}\r\n value={setting.isSensitive ? '********' : displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing || setting.isSensitive}\r\n className=\"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n };\r\n\r\n const renderSetting = (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n const isEditing = saving === editKey;\r\n const changed = hasChanges(setting);\r\n const displayValue = getDisplayValue(setting);\r\n\r\n return (\r\n <div\r\n key={editKey}\r\n className=\"flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0\"\r\n >\r\n <div className=\"flex-1 min-w-0 pr-4\">\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {!setting.isEditable && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className={`px-2 py-0.5 text-xs rounded ${getSourceBadgeClass(setting.source)}`}>\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n {setting.modifiedAt && (\r\n <p className=\"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1\">\r\n <Clock className=\"w-3 h-3\" />\r\n {t('settings.modifiedBy', { user: setting.modifiedByUserName, date: new Date(setting.modifiedAt).toLocaleString() })}\r\n </p>\r\n )}\r\n </div>\r\n\r\n <div className=\"flex items-center gap-2\">\r\n {getSettingInput(setting, displayValue, isEditing)}\r\n\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <>\r\n <button\r\n onClick={() => handleSave(setting)}\r\n disabled={!changed || isEditing}\r\n className=\"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('common.save')}\r\n >\r\n {isEditing ? <Loader2 className=\"w-4 h-4 animate-spin\" /> : <Save className=\"w-4 h-4\" />}\r\n </button>\r\n <button\r\n onClick={() => handleReset(setting)}\r\n disabled={setting.value === setting.defaultValue || isEditing}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('settings.resetToDefault')}\r\n >\r\n <RotateCcw className=\"w-4 h-4\" />\r\n </button>\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n </>\r\n )}\r\n </div>\r\n </div>\r\n );\r\n };\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n {/* Session Settings */}\r\n {sessionSettings.length > 0 && (\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-[var(--color-accent-500)]/20\">\r\n <Clock className=\"w-5 h-5 text-[var(--color-accent-600)]\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.session')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.sessionDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {sessionSettings.map(renderSetting)}\r\n </div>\r\n </div>\r\n )}\r\n\r\n {/* General Settings */}\r\n {generalSettings.length > 0 && (\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-[var(--color-accent-500)]/20\">\r\n <SettingsIcon className=\"w-5 h-5 text-[var(--color-accent-600)]\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.general')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.generalDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {generalSettings.map(renderSetting)}\r\n </div>\r\n </div>\r\n )}\r\n\r\n {sessionSettings.length === 0 && generalSettings.length === 0 && (\r\n <div className=\"text-center py-12 text-[var(--text-secondary)]\">\r\n {t('settings.noSettings')}\r\n </div>\r\n )}\r\n\r\n {historyModal && (\r\n <SettingsHistoryModal\r\n category={historyModal.category}\r\n settingKey={historyModal.key}\r\n history={history}\r\n loading={loadingHistory}\r\n onClose={closeHistory}\r\n />\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default GeneralSettingsSection;\r\n","import type { ReactElement } from 'react';\nimport { useTranslation } from 'react-i18next';\r\nimport { Save, RotateCcw, Loader2, Upload, History, Clock } from 'lucide-react';\r\nimport type { SettingViewModel } from '@/services/api/adminApi';\r\nimport { useSettingsSection } from '@/hooks/useSettingsSection';\r\nimport { SettingsHistoryModal } from '@/components/platform/administration/settings/SettingsHistoryModal';\r\n\r\nexport function FileUploadSettingsSection(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const {\r\n settings,\r\n saving,\r\n historyModal,\r\n history,\r\n loadingHistory,\r\n handleValueChange,\r\n hasChanges,\r\n getDisplayValue,\r\n getEditKey,\r\n handleSave,\r\n handleReset,\r\n openHistory,\r\n closeHistory,\r\n } = useSettingsSection();\r\n\r\n // Filter settings for FileUpload category\r\n const fileUploadSettings = settings.filter(s => s.category === 'FileUpload');\r\n\r\n // Parse JSON extensions for display\r\n const parseExtensions = (value: string): string[] => {\r\n try {\r\n return JSON.parse(value);\r\n } catch {\r\n return [];\r\n }\r\n };\r\n\r\n const renderSettingInput = (setting: SettingViewModel, displayValue: string, isEditing: boolean) => {\r\n if (setting.valueType === 'Int') {\r\n return (\r\n <div className=\"flex items-center gap-2\">\r\n <input\r\n type=\"number\"\r\n value={displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n {setting.key === 'MaxFileSizeMB' && (\r\n <span className=\"text-sm text-[var(--text-secondary)]\">MB</span>\r\n )}\r\n </div>\r\n );\r\n }\r\n return (\r\n <input\r\n type={setting.isSensitive ? 'password' : 'text'}\r\n value={setting.isSensitive ? '********' : displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing || setting.isSensitive}\r\n className=\"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n };\r\n\r\n const renderSetting = (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n const isEditing = saving === editKey;\r\n const changed = hasChanges(setting);\r\n const displayValue = getDisplayValue(setting);\r\n\r\n // Special rendering for JSON arrays (extensions)\r\n if (setting.valueType === 'Json' && setting.key === 'AllowedExtensions') {\r\n const extensions = parseExtensions(displayValue);\r\n return (\r\n <div key={editKey} className=\"py-4 border-b border-[var(--border-color)] last:border-0\">\r\n <div className=\"flex items-center justify-between mb-3\">\r\n <div>\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {!setting.isEditable && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)] rounded\">\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n </div>\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n )}\r\n </div>\r\n <div className=\"flex flex-wrap gap-2\">\r\n {extensions.map((ext) => (\r\n <span\r\n key={ext}\r\n className=\"px-3 py-1 bg-[var(--bg-secondary)] text-[var(--text-secondary)] rounded-full text-sm\"\r\n >\r\n {ext}\r\n </span>\r\n ))}\r\n </div>\r\n </div>\r\n );\r\n }\r\n\r\n return (\r\n <div\r\n key={editKey}\r\n className=\"flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0\"\r\n >\r\n <div className=\"flex-1 min-w-0 pr-4\">\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {!setting.isEditable && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className={`px-2 py-0.5 text-xs rounded ${\r\n setting.source === 'Database'\r\n ? 'bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]'\r\n : setting.source === 'AppSettings'\r\n ? 'bg-purple-500/10 text-purple-600 dark:text-purple-400'\r\n : 'bg-[var(--bg-secondary)] text-[var(--text-tertiary)]'\r\n }`}>\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n {setting.modifiedAt && (\r\n <p className=\"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1\">\r\n <Clock className=\"w-3 h-3\" />\r\n {t('settings.modifiedBy', { user: setting.modifiedByUserName, date: new Date(setting.modifiedAt).toLocaleString() })}\r\n </p>\r\n )}\r\n </div>\r\n\r\n <div className=\"flex items-center gap-2\">\r\n {renderSettingInput(setting, displayValue, isEditing)}\r\n\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <>\r\n <button\r\n onClick={() => handleSave(setting)}\r\n disabled={!changed || isEditing}\r\n className=\"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('common.save')}\r\n >\r\n {isEditing ? <Loader2 className=\"w-4 h-4 animate-spin\" /> : <Save className=\"w-4 h-4\" />}\r\n </button>\r\n <button\r\n onClick={() => handleReset(setting)}\r\n disabled={setting.value === setting.defaultValue || isEditing}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('settings.resetToDefault')}\r\n >\r\n <RotateCcw className=\"w-4 h-4\" />\r\n </button>\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n </>\r\n )}\r\n </div>\r\n </div>\r\n );\r\n };\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-[var(--color-accent-500)]/20\">\r\n <Upload className=\"w-5 h-5 text-[var(--color-accent-600)]\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.fileUpload')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.fileUploadDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {fileUploadSettings.length > 0 ? (\r\n fileUploadSettings.map(renderSetting)\r\n ) : (\r\n <div className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noSettings')}\r\n </div>\r\n )}\r\n </div>\r\n </div>\r\n\r\n {historyModal && (\r\n <SettingsHistoryModal\r\n category={historyModal.category}\r\n settingKey={historyModal.key}\r\n history={history}\r\n loading={loadingHistory}\r\n onClose={closeHistory}\r\n />\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default FileUploadSettingsSection;\r\n","import { useState } from 'react';\r\nimport type { ReactElement } from 'react';\r\nimport { useTranslation } from 'react-i18next';\r\nimport { useOutletContext } from 'react-router-dom';\r\nimport { Save, RotateCcw, Loader2, FileText, History, Clock, AlertTriangle, Shield } from 'lucide-react';\r\nimport { adminApi, type SettingViewModel, type SettingHistoryDto } from '@/services/api/adminApi';\r\n\r\ninterface SettingsContext {\r\n settings: SettingViewModel[];\r\n refreshSettings: () => Promise<void>;\r\n}\r\n\r\nexport function LegalFileSettingsSection(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const { settings, refreshSettings } = useOutletContext<SettingsContext>();\r\n const [editedValues, setEditedValues] = useState<Record<string, string>>({});\r\n const [saving, setSaving] = useState<string | null>(null);\r\n const [historyModal, setHistoryModal] = useState<{ category: string; key: string } | null>(null);\r\n const [history, setHistory] = useState<SettingHistoryDto[]>([]);\r\n const [loadingHistory, setLoadingHistory] = useState(false);\r\n\r\n // Filter settings for LegalFile category\r\n const legalFileSettings = settings.filter(s => s.category === 'LegalFile');\r\n\r\n const handleValueChange = (category: string, key: string, value: string) => {\r\n setEditedValues(prev => ({\r\n ...prev,\r\n [`${category}.${key}`]: value\r\n }));\r\n };\r\n\r\n const hasChanges = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] !== undefined && editedValues[editKey] !== setting.value;\r\n };\r\n\r\n const getDisplayValue = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] ?? setting.value;\r\n };\r\n\r\n const handleSave = async (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n const newValue = editedValues[editKey];\r\n if (newValue === undefined || newValue === setting.value) return;\r\n\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.update(setting.category, setting.key, newValue);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to save setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const handleReset = async (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.reset(setting.category, setting.key);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to reset setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const openHistory = async (category: string, key: string) => {\r\n setHistoryModal({ category, key });\r\n setLoadingHistory(true);\r\n try {\r\n const result = await adminApi.settings.getHistory(category, key);\r\n setHistory(result);\r\n } catch (err) {\r\n console.error('Failed to load history:', err);\r\n } finally {\r\n setLoadingHistory(false);\r\n }\r\n };\r\n\r\n // Convert days to years for display\r\n const daysToYears = (days: number): string => {\r\n const years = Math.floor(days / 365);\r\n const remainingDays = days % 365;\r\n if (remainingDays === 0) {\r\n return `${years} ${t('settings.years')}`;\r\n }\r\n return `${years} ${t('settings.years')} ${remainingDays} ${t('settings.days')}`;\r\n };\r\n\r\n const getLegallyProtected = (setting: SettingViewModel): boolean => {\r\n return !setting.isEditable && (setting.key === 'RetentionDays' || setting.key === 'EnableWorm');\r\n };\r\n\r\n const renderSettingInput = (setting: SettingViewModel, displayValue: string, isEditing: boolean, handleValueChange: (category: string, key: string, value: string) => void) => {\r\n if (setting.valueType === 'Bool') {\r\n return (\r\n <label className=\"relative inline-flex cursor-pointer\">\r\n <span className=\"sr-only\">{setting.key}</span>\r\n <input\r\n type=\"checkbox\"\r\n checked={displayValue === 'true'}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.checked ? 'true' : 'false')}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"sr-only peer\"\r\n aria-label={setting.key}\r\n />\r\n <div className=\"w-11 h-6 bg-[var(--bg-tertiary)] peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-[var(--color-accent-500)]/30 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-[var(--border-color)] after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[var(--color-accent-600)] peer-disabled:opacity-50 peer-disabled:cursor-not-allowed\"></div>\r\n </label>\r\n );\r\n }\r\n if (setting.valueType === 'Int') {\r\n return (\r\n <div className=\"flex items-center gap-2\">\r\n <input\r\n type=\"number\"\r\n value={displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n {(setting.key === 'RetentionDays' || setting.key === 'ArchiveAfterDays') && (\r\n <span className=\"text-sm text-[var(--text-secondary)]\">\r\n ({daysToYears(parseInt(displayValue) || 0)})\r\n </span>\r\n )}\r\n </div>\r\n );\r\n }\r\n return (\r\n <input\r\n type={setting.isSensitive ? 'password' : 'text'}\r\n value={setting.isSensitive ? '********' : displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing || setting.isSensitive}\r\n className=\"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n };\r\n\r\n const getSourceBadgeClass = (source: string): string => {\r\n if (source === 'Database') return 'bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]';\r\n if (source === 'AppSettings') return 'bg-purple-500/10 text-purple-600 dark:text-purple-400';\r\n return 'bg-[var(--bg-secondary)] text-[var(--text-tertiary)]';\r\n };\r\n\r\n const renderSetting = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n const isEditing = saving === editKey;\r\n const changed = hasChanges(setting);\r\n const displayValue = getDisplayValue(setting);\r\n const isLegallyProtected = getLegallyProtected(setting);\r\n\r\n return (\r\n <div\r\n key={editKey}\r\n className={`flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0 ${\r\n isLegallyProtected ? 'bg-[var(--warning-bg)] -mx-6 px-6' : ''\r\n }`}\r\n >\r\n <div className=\"flex-1 min-w-0 pr-4\">\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {isLegallyProtected && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--warning-bg)] text-[var(--warning-text)] border border-[var(--warning-border)] rounded flex items-center gap-1\">\r\n <Shield className=\"w-3 h-3\" />\r\n {t('settings.legallyProtected')}\r\n </span>\r\n )}\r\n {!setting.isEditable && !isLegallyProtected && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className={`px-2 py-0.5 text-xs rounded ${getSourceBadgeClass(setting.source)}`}>\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n {setting.modifiedAt && (\r\n <p className=\"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1\">\r\n <Clock className=\"w-3 h-3\" />\r\n {t('settings.modifiedBy', { user: setting.modifiedByUserName, date: new Date(setting.modifiedAt).toLocaleString() })}\r\n </p>\r\n )}\r\n </div>\r\n\r\n <div className=\"flex items-center gap-2\">\r\n {renderSettingInput(setting, displayValue, isEditing, handleValueChange)}\r\n\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <>\r\n <button\r\n onClick={() => handleSave(setting)}\r\n disabled={!changed || isEditing}\r\n className=\"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('common.save')}\r\n >\r\n {isEditing ? <Loader2 className=\"w-4 h-4 animate-spin\" /> : <Save className=\"w-4 h-4\" />}\r\n </button>\r\n <button\r\n onClick={() => handleReset(setting)}\r\n disabled={setting.value === setting.defaultValue || isEditing}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('settings.resetToDefault')}\r\n >\r\n <RotateCcw className=\"w-4 h-4\" />\r\n </button>\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n </>\r\n )}\r\n </div>\r\n </div>\r\n );\r\n };\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n {/* Legal Warning Banner */}\r\n <div className=\"bg-[var(--warning-bg)] border border-[var(--warning-border)] rounded-[var(--radius-card)] p-4\">\r\n <div className=\"flex items-start gap-3\">\r\n <AlertTriangle className=\"w-5 h-5 text-[var(--warning-text)] flex-shrink-0 mt-0.5\" />\r\n <div>\r\n <h4 className=\"font-medium text-[var(--warning-text)]\">\r\n {t('settings.legalWarningTitle')}\r\n </h4>\r\n <p className=\"text-sm text-[var(--warning-text)] opacity-90 mt-1\">\r\n {t('settings.legalWarningDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-amber-500/10 to-amber-600/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-amber-500/20\">\r\n <FileText className=\"w-5 h-5 text-amber-600\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.legalFile')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.legalFileDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {legalFileSettings.length > 0 ? (\r\n legalFileSettings.map(renderSetting)\r\n ) : (\r\n <div className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noSettings')}\r\n </div>\r\n )}\r\n </div>\r\n </div>\r\n\r\n {/* History Modal */}\r\n {historyModal && (\r\n <div className=\"fixed inset-0 z-50 overflow-y-auto\">\r\n <div className=\"flex items-center justify-center min-h-screen px-4\">\r\n <button type=\"button\" className=\"fixed inset-0 bg-black/50\" onClick={() => setHistoryModal(null)} aria-label=\"Close modal\" />\r\n <div className=\"relative rounded-[var(--radius-card)] bg-[var(--bg-card)] shadow-xl max-w-2xl w-full p-6\">\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)] mb-4\">\r\n {t('settings.historyTitle', { key: `${historyModal.category}.${historyModal.key}` })}\r\n </h3>\r\n\r\n {loadingHistory && (\r\n <div className=\"flex justify-center py-8\">\r\n <Loader2 className=\"w-8 h-8 animate-spin text-[var(--color-accent-600)]\" />\r\n </div>\r\n )}\r\n {!loadingHistory && history.length === 0 && (\r\n <p className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noHistory')}\r\n </p>\r\n )}\r\n {!loadingHistory && history.length > 0 && (\r\n <div className=\"space-y-3 max-h-96 overflow-y-auto\">\r\n {history.map((entry) => (\r\n <div key={entry.id} className=\"p-3 bg-[var(--bg-secondary)] rounded-lg\">\r\n <div className=\"flex justify-between text-sm\">\r\n <span className=\"text-[var(--text-secondary)]\">\r\n {entry.changedByUserName || t('settings.system')}\r\n </span>\r\n <span className=\"text-[var(--text-tertiary)]\">\r\n {new Date(entry.changedAt).toLocaleString()}\r\n </span>\r\n </div>\r\n <div className=\"mt-2 text-sm\">\r\n <span className=\"text-[var(--error-text)] line-through\">{entry.oldValue}</span>\r\n <span className=\"mx-2 text-[var(--text-tertiary)]\">→</span>\r\n <span className=\"text-[var(--success-text)]\">{entry.newValue}</span>\r\n </div>\r\n </div>\r\n ))}\r\n </div>\r\n )}\r\n\r\n <div className=\"mt-6 flex justify-end\">\r\n <button\r\n onClick={() => setHistoryModal(null)}\r\n className=\"px-4 py-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n >\r\n {t('common.cancel')}\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default LegalFileSettingsSection;\r\n"],"names":["TABS","Settings","Upload","FileText","SettingsPage","useTranslation","navigate","useNavigate","location","useLocation","settings","setSettings","useState","loading","setLoading","loadSettings","useCallback","result","adminApi","err","useEffect","getCurrentTab","path","t","handleTabChange","tabId","tab","jsx","Loader2","jsxs","Breadcrumb","Icon","isActive","Outlet","useSettingsSection","refreshSettings","useOutletContext","editedValues","setEditedValues","saving","setSaving","historyModal","setHistoryModal","history","setHistory","loadingHistory","setLoadingHistory","handleValueChange","category","key","value","prev","hasChanges","setting","editKey","getDisplayValue","getEditKey","newValue","next","SettingsHistoryModal","settingKey","onClose","entry","GeneralSettingsSection","handleSave","handleReset","openHistory","closeHistory","sessionSettings","generalSettings","getSourceBadgeClass","source","getSettingInput","displayValue","isEditing","e","renderSetting","changed","Clock","Fragment","Save","RotateCcw","History","SettingsIcon","FileUploadSettingsSection","fileUploadSettings","s","parseExtensions","renderSettingInput","extensions","ext","LegalFileSettingsSection","legalFileSettings","daysToYears","days","years","remainingDays","getLegallyProtected","isLegallyProtected","Shield","AlertTriangle"],"mappings":"4PAeMA,EAAoB,CACxB,CAAE,GAAI,UAAW,SAAU,wBAAyB,KAAMC,EAAAA,SAAU,MAAO,gDAAA,EAC3E,CAAE,GAAI,cAAe,SAAU,2BAA4B,KAAMC,EAAAA,OAAQ,MAAO,oDAAA,EAChF,CAAE,GAAI,aAAc,SAAU,0BAA2B,KAAMC,EAAAA,SAAU,MAAO,mDAAA,CAClF,EAEO,SAASC,GAA6B,CAC3C,KAAM,CAAE,CAAA,EAAMC,EAAAA,eAAe,OAAO,EAC9BC,EAAWC,EAAAA,YAAA,EACXC,EAAWC,EAAAA,YAAA,EACX,CAACC,EAAUC,CAAW,EAAIC,EAAAA,SAA6B,CAAA,CAAE,EACzD,CAACC,EAASC,CAAU,EAAIF,EAAAA,SAAS,EAAI,EAErCG,EAAeC,EAAAA,YAAY,SAAY,CAC3C,GAAI,CACFF,EAAW,EAAI,EACf,MAAMG,EAAS,MAAMC,WAAS,SAAS,OAAA,EACvCP,EAAYM,CAAM,CACpB,OAASE,EAAK,CACZ,QAAQ,MAAM,2BAA4BA,CAAG,CAC/C,QAAA,CACEL,EAAW,EAAK,CAClB,CACF,EAAG,CAAA,CAAE,EAELM,EAAAA,UAAU,IAAM,CACdL,EAAA,CACF,EAAG,CAACA,CAAY,CAAC,EAEjB,MAAMM,EAAgB,IAAM,CAC1B,MAAMC,EAAOd,EAAS,SAEtB,OADYR,EAAK,KAAKuB,GAAKD,EAAK,SAASC,EAAE,EAAE,CAAC,GAClC,IAAM,SACpB,EAEMC,EAAmBC,GAAkB,CACzC,MAAMC,EAAM1B,EAAK,KAAKuB,GAAKA,EAAE,KAAOE,CAAK,EACrCC,GACFpB,EAASoB,EAAI,KAAK,CAEtB,EAEA,OAAIb,EAEAc,MAAC,OAAI,UAAU,iDACb,eAACC,EAAAA,QAAA,CAAQ,UAAU,sDAAsD,CAAA,CAC3E,EAKFC,EAAAA,KAAC,MAAA,CAAI,UAAU,YACb,SAAA,CAAAF,EAAAA,IAACG,EAAAA,WAAA,CACC,MAAO,CACL,CAAE,MAAO,EAAE,cAAc,EAAG,KAAM,iBAAA,EAClC,CAAE,MAAO,EAAE,iBAAkB,UAAU,CAAA,CAAE,CAC3C,CAAA,EAIFH,MAAC,MAAA,CAAI,UAAU,oCACb,gBAAC,MAAA,CACC,SAAA,CAAAA,MAAC,KAAA,CAAG,UAAU,gDACX,SAAA,EAAE,gBAAgB,EACrB,QACC,IAAA,CAAE,UAAU,oCACV,SAAA,EAAE,mBAAmB,CAAA,CACxB,CAAA,CAAA,CACF,CAAA,CACF,EAGAA,EAAAA,IAAC,MAAA,CAAI,UAAU,wCACb,SAAAA,EAAAA,IAAC,MAAA,CAAI,UAAU,wBAAwB,aAAW,OAC/C,SAAA3B,EAAK,IAAK0B,GAAQ,CACjB,MAAMK,EAAOL,EAAI,KACXM,EAAWX,MAAoBK,EAAI,GACzC,OACEG,EAAAA,KAAC,SAAA,CAEC,QAAS,IAAML,EAAgBE,EAAI,EAAE,EACrC,UAAW;AAAA;AAAA,oBAEPM,EACE,kEACA,qHACJ;AAAA,kBAGF,SAAA,CAAAL,MAACI,GAAK,UAAW,gBAAgBC,EAAW,iCAAmC,sEAAsE,GAAI,EACxJ,EAAEN,EAAI,QAAQ,CAAA,CAAA,EAXVA,EAAI,EAAA,CAcf,CAAC,EACH,EACF,QAGCO,EAAAA,OAAA,CAAO,QAAS,CAAE,SAAAvB,EAAU,gBAAiBK,EAAa,CAAG,CAAA,EAChE,CAEJ,CC3GO,SAASmB,GAed,CACA,KAAM,CAAE,SAAAxB,EAAU,gBAAAyB,CAAA,EAAoBC,mBAAA,EAChC,CAACC,EAAcC,CAAe,EAAI1B,EAAAA,SAAiC,CAAA,CAAE,EACrE,CAAC2B,EAAQC,CAAS,EAAI5B,EAAAA,SAAwB,IAAI,EAClD,CAAC6B,EAAcC,CAAe,EAAI9B,EAAAA,SAAmD,IAAI,EACzF,CAAC+B,EAASC,CAAU,EAAIhC,EAAAA,SAA8B,CAAA,CAAE,EACxD,CAACiC,EAAgBC,CAAiB,EAAIlC,EAAAA,SAAS,EAAK,EAEpDmC,EAAoB,CAACC,EAAkBC,EAAaC,IAAkB,CAC1EZ,EAAgBa,IAAS,CACvB,GAAGA,EACH,CAAC,GAAGH,CAAQ,IAAIC,CAAG,EAAE,EAAGC,CAAA,EACxB,CACJ,EAEME,EAAcC,GAA8B,CAChD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAClD,OAAOhB,EAAaiB,CAAO,IAAM,QAAajB,EAAaiB,CAAO,IAAMD,EAAQ,KAClF,EAEME,EAAmBF,GAA8B,CACrD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAClD,OAAOhB,EAAaiB,CAAO,GAAKD,EAAQ,KAC1C,EAEMG,EAAcH,GAA8B,GAAGA,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAwDpF,MAAO,CACL,SAAA3C,EACA,aAAA2B,EACA,OAAAE,EACA,aAAAE,EACA,QAAAE,EACA,eAAAE,EACA,kBAAAE,EACA,WAAAK,EACA,gBAAAG,EACA,WAAAC,EACA,WAjEiB,MAAOH,GAA8B,CACtD,MAAMC,EAAUE,EAAWH,CAAO,EAC5BI,EAAWpB,EAAaiB,CAAO,EACrC,GAAI,EAAAG,IAAa,QAAaA,IAAaJ,EAAQ,OAEnD,GAAI,CACFb,EAAUc,CAAO,EACjB,MAAMpC,EAAAA,SAAS,SAAS,OAAOmC,EAAQ,SAAUA,EAAQ,IAAKI,CAAQ,EACtEnB,EAAgBa,GAAQ,CACtB,MAAMO,EAAO,CAAE,GAAGP,CAAA,EAClB,cAAOO,EAAKJ,CAAO,EACZI,CACT,CAAC,EACD,MAAMvB,EAAA,CACR,OAAShB,EAAK,CACZ,QAAQ,MAAM,0BAA2BA,CAAG,CAC9C,QAAA,CACEqB,EAAU,IAAI,CAChB,CACF,EA+CE,YA7CkB,MAAOa,GAA8B,CACvD,MAAMC,EAAUE,EAAWH,CAAO,EAClC,GAAI,CACFb,EAAUc,CAAO,EACjB,MAAMpC,EAAAA,SAAS,SAAS,MAAMmC,EAAQ,SAAUA,EAAQ,GAAG,EAC3Df,EAAgBa,GAAQ,CACtB,MAAMO,EAAO,CAAE,GAAGP,CAAA,EAClB,cAAOO,EAAKJ,CAAO,EACZI,CACT,CAAC,EACD,MAAMvB,EAAA,CACR,OAAShB,EAAK,CACZ,QAAQ,MAAM,2BAA4BA,CAAG,CAC/C,QAAA,CACEqB,EAAU,IAAI,CAChB,CACF,EA8BE,YA5BkB,MAAOQ,EAAkBC,IAAgB,CAC3DP,EAAgB,CAAE,SAAAM,EAAU,IAAAC,EAAK,EACjCH,EAAkB,EAAI,EACtB,GAAI,CACF,MAAM7B,EAAS,MAAMC,EAAAA,SAAS,SAAS,WAAW8B,EAAUC,CAAG,EAC/DL,EAAW3B,CAAM,CACnB,OAASE,EAAK,CACZ,QAAQ,MAAM,0BAA2BA,CAAG,CAC9C,QAAA,CACE2B,EAAkB,EAAK,CACzB,CACF,EAkBE,aAhBmB,IAAMJ,EAAgB,IAAI,CAgB7C,CAEJ,CC5GO,SAASiB,EAAqB,CAAE,SAAAX,EAAU,WAAAY,EAAY,QAAAjB,EAAS,QAAA9B,EAAS,QAAAgD,GAAoD,CACjI,KAAM,CAAE,EAAAtC,CAAA,EAAMlB,EAAAA,eAAe,OAAO,EAEpC,aACG,MAAA,CAAI,UAAU,qCACb,SAAAwB,EAAAA,KAAC,MAAA,CAAI,UAAU,qDACb,SAAA,CAAAF,EAAAA,IAAC,SAAA,CAAO,KAAK,SAAS,UAAU,4BAA4B,QAASkC,EAAS,aAAW,aAAA,CAAc,EACvGhC,EAAAA,KAAC,MAAA,CAAI,UAAU,2FACb,SAAA,CAAAF,EAAAA,IAAC,KAAA,CAAG,UAAU,wDACX,SAAAJ,EAAE,wBAAyB,CAAE,IAAK,GAAGyB,CAAQ,IAAIY,CAAU,EAAA,CAAI,EAClE,EAEC/C,SACE,MAAA,CAAI,UAAU,2BACb,SAAAc,EAAAA,IAACC,EAAAA,QAAA,CAAQ,UAAU,qDAAA,CAAsD,CAAA,CAC3E,EAED,CAACf,GAAW8B,EAAQ,SAAW,GAC9BhB,EAAAA,IAAC,IAAA,CAAE,UAAU,gDACV,SAAAJ,EAAE,oBAAoB,CAAA,CACzB,EAED,CAACV,GAAW8B,EAAQ,OAAS,SAC3B,MAAA,CAAI,UAAU,qCACZ,SAAAA,EAAQ,IAAKmB,GACZjC,EAAAA,KAAC,MAAA,CAAmB,UAAU,0CAC5B,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,+BACb,SAAA,CAAAF,EAAAA,IAAC,QAAK,UAAU,+BACb,WAAM,mBAAqBJ,EAAE,iBAAiB,EACjD,EACAI,EAAAA,IAAC,OAAA,CAAK,UAAU,8BACb,SAAA,IAAI,KAAKmC,EAAM,SAAS,EAAE,eAAA,CAAe,CAC5C,CAAA,EACF,EACAjC,EAAAA,KAAC,MAAA,CAAI,UAAU,eACb,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,wCAAyC,SAAAmC,EAAM,SAAS,EACxEnC,EAAAA,IAAC,OAAA,CAAK,UAAU,mCAAmC,SAAA,IAAC,EACpDA,EAAAA,IAAC,OAAA,CAAK,UAAU,6BAA8B,WAAM,QAAA,CAAS,CAAA,CAAA,CAC/D,CAAA,CAAA,EAbQmC,EAAM,EAchB,CACD,EACH,EAGFnC,EAAAA,IAAC,MAAA,CAAI,UAAU,wBACb,SAAAA,EAAAA,IAAC,SAAA,CACC,QAASkC,EACT,UAAU,yHAET,WAAE,eAAe,CAAA,CAAA,CACpB,CACF,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,CAEJ,CC9DO,SAASE,GAAuC,CACrD,KAAM,CAAE,CAAA,EAAM1D,EAAAA,eAAe,OAAO,EAC9B,CACJ,SAAAK,EACA,OAAA6B,EACA,aAAAE,EACA,QAAAE,EACA,eAAAE,EACA,kBAAAE,EACA,WAAAK,EACA,gBAAAG,EACA,WAAAC,EACA,WAAAQ,EACA,YAAAC,EACA,YAAAC,EACA,aAAAC,CAAA,EACEjC,EAAA,EAGEkC,EAAkB1D,EAAS,OAAO,GAAK,EAAE,WAAa,SAAS,EAC/D2D,EAAkB3D,EAAS,OAAO,GAAK,EAAE,WAAa,SAAS,EAE/D4D,EAAuBC,GACvBA,IAAW,WACN,iEAELA,IAAW,cACN,wDAEF,uDAGHC,EAAkB,CAACnB,EAA2BoB,EAAsBC,IACpErB,EAAQ,YAAc,OAEtBxB,EAAAA,KAAC,QAAA,CAAM,UAAU,mDACf,SAAA,CAAAF,EAAAA,IAAC,QAAA,CACC,KAAK,WACL,QAAS8C,IAAiB,OAC1B,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,QAAU,OAAS,OAAO,EACrG,SAAU,CAACtB,EAAQ,YAAcqB,EACjC,UAAU,cAAA,CAAA,EAEZ/C,EAAAA,IAAC,MAAA,CAAI,UAAU,ifAAA,CAAkf,CAAA,EACngB,EAGA0B,EAAQ,YAAc,OAASA,EAAQ,YAAc,UAErD1B,EAAAA,IAAC,QAAA,CACC,KAAK,SACL,MAAO8C,EACP,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,KAAK,EAChF,SAAU,CAACtB,EAAQ,YAAcqB,EACjC,UAAU,2QAAA,CAAA,EAKd/C,EAAAA,IAAC,QAAA,CACC,KAAM0B,EAAQ,YAAc,WAAa,OACzC,MAAOA,EAAQ,YAAc,WAAaoB,EAC1C,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,KAAK,EAChF,SAAU,CAACtB,EAAQ,YAAcqB,GAAarB,EAAQ,YACtD,UAAU,2QAAA,CAAA,EAKVuB,EAAiBvB,GAA8B,CACnD,MAAMC,EAAUE,EAAWH,CAAO,EAC5BqB,EAAYnC,IAAWe,EACvBuB,EAAUzB,EAAWC,CAAO,EAC5BoB,EAAelB,EAAgBF,CAAO,EAE5C,OACExB,EAAAA,KAAC,MAAA,CAEC,UAAU,6FAEV,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,sBACb,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,yCACb,SAAA,EAAE,iBAAiB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,GAAG,EACpE,EACC,CAACA,EAAQ,YACR1B,EAAAA,IAAC,QAAK,UAAU,mFACb,SAAA,EAAE,mBAAmB,CAAA,CACxB,EAEFA,EAAAA,IAAC,OAAA,CAAK,UAAW,+BAA+B2C,EAAoBjB,EAAQ,MAAM,CAAC,GAChF,SAAAA,EAAQ,MAAA,CACX,CAAA,EACF,EACCA,EAAQ,aACP1B,EAAAA,IAAC,IAAA,CAAE,UAAU,4CACV,SAAA,EAAE,yBAAyB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,WAAW,EACpF,EAEDA,EAAQ,YACPxB,OAAC,IAAA,CAAE,UAAU,mEACX,SAAA,CAAAF,EAAAA,IAACmD,EAAAA,MAAA,CAAM,UAAU,SAAA,CAAU,EAC1B,EAAE,sBAAuB,CAAE,KAAMzB,EAAQ,mBAAoB,KAAM,IAAI,KAAKA,EAAQ,UAAU,EAAE,eAAA,EAAkB,CAAA,CAAA,CACrH,CAAA,EAEJ,EAEAxB,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACZ,SAAA,CAAA2C,EAAgBnB,EAASoB,EAAcC,CAAS,EAEhDrB,EAAQ,YAAcA,EAAQ,SAAW,YACxCxB,OAAAkD,EAAAA,SAAA,CACE,SAAA,CAAApD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMqC,EAAWX,CAAO,EACjC,SAAU,CAACwB,GAAWH,EACtB,UAAU,4KACV,MAAO,EAAE,aAAa,EAErB,SAAAA,QAAa9C,EAAAA,QAAA,CAAQ,UAAU,uBAAuB,EAAKD,EAAAA,IAACqD,EAAAA,KAAA,CAAK,UAAU,SAAA,CAAU,CAAA,CAAA,EAExFrD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMsC,EAAYZ,CAAO,EAClC,SAAUA,EAAQ,QAAUA,EAAQ,cAAgBqB,EACpD,UAAU,mKACV,MAAO,EAAE,yBAAyB,EAElC,SAAA/C,EAAAA,IAACsD,EAAAA,UAAA,CAAU,UAAU,SAAA,CAAU,CAAA,CAAA,EAEjCtD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMuC,EAAYb,EAAQ,SAAUA,EAAQ,GAAG,EACxD,UAAU,mHACV,MAAO,EAAE,sBAAsB,EAE/B,SAAA1B,EAAAA,IAACuD,EAAAA,QAAA,CAAQ,UAAU,SAAA,CAAU,CAAA,CAAA,CAC/B,CAAA,CACF,CAAA,CAAA,CAEJ,CAAA,CAAA,EA5DK5B,CAAA,CA+DX,EAEA,OACEzB,EAAAA,KAAC,MAAA,CAAI,UAAU,YAEZ,SAAA,CAAAuC,EAAgB,OAAS,GACxBvC,EAAAA,KAAC,MAAA,CAAI,UAAU,iHACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,oIACb,SAAAE,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,iDACb,eAACmD,EAAAA,MAAA,CAAM,UAAU,yCAAyC,CAAA,CAC5D,SACC,MAAA,CACC,SAAA,CAAAnD,MAAC,KAAA,CAAG,UAAU,mDACX,SAAA,EAAE,2BAA2B,EAChC,QACC,IAAA,CAAE,UAAU,uCACV,SAAA,EAAE,sCAAsC,CAAA,CAC3C,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,QACC,MAAA,CAAI,UAAU,YACZ,SAAAyC,EAAgB,IAAIQ,CAAa,CAAA,CACpC,CAAA,EACF,EAIDP,EAAgB,OAAS,GACxBxC,EAAAA,KAAC,MAAA,CAAI,UAAU,iHACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,oIACb,SAAAE,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,iDACb,eAACwD,EAAAA,SAAA,CAAa,UAAU,yCAAyC,CAAA,CACnE,SACC,MAAA,CACC,SAAA,CAAAxD,MAAC,KAAA,CAAG,UAAU,mDACX,SAAA,EAAE,2BAA2B,EAChC,QACC,IAAA,CAAE,UAAU,uCACV,SAAA,EAAE,sCAAsC,CAAA,CAC3C,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,QACC,MAAA,CAAI,UAAU,YACZ,SAAA0C,EAAgB,IAAIO,CAAa,CAAA,CACpC,CAAA,EACF,EAGDR,EAAgB,SAAW,GAAKC,EAAgB,SAAW,GAC1D1C,EAAAA,IAAC,MAAA,CAAI,UAAU,iDACZ,SAAA,EAAE,qBAAqB,CAAA,CAC1B,EAGDc,GACCd,EAAAA,IAACgC,EAAA,CACC,SAAUlB,EAAa,SACvB,WAAYA,EAAa,IACzB,QAAAE,EACA,QAASE,EACT,QAASsB,CAAA,CAAA,CACX,EAEJ,CAEJ,CCjNO,SAASiB,GAA0C,CACxD,KAAM,CAAE,CAAA,EAAM/E,EAAAA,eAAe,OAAO,EAC9B,CACJ,SAAAK,EACA,OAAA6B,EACA,aAAAE,EACA,QAAAE,EACA,eAAAE,EACA,kBAAAE,EACA,WAAAK,EACA,gBAAAG,EACA,WAAAC,EACA,WAAAQ,EACA,YAAAC,EACA,YAAAC,EACA,aAAAC,CAAA,EACEjC,EAAA,EAGEmD,EAAqB3E,EAAS,OAAO4E,GAAKA,EAAE,WAAa,YAAY,EAGrEC,EAAmBrC,GAA4B,CACnD,GAAI,CACF,OAAO,KAAK,MAAMA,CAAK,CACzB,MAAQ,CACN,MAAO,CAAA,CACT,CACF,EAEMsC,EAAqB,CAACnC,EAA2BoB,EAAsBC,IACvErB,EAAQ,YAAc,MAEtBxB,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,QAAA,CACC,KAAK,SACL,MAAO8C,EACP,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,KAAK,EAChF,SAAU,CAACtB,EAAQ,YAAcqB,EACjC,UAAU,2QAAA,CAAA,EAEXrB,EAAQ,MAAQ,uBACd,OAAA,CAAK,UAAU,uCAAuC,SAAA,IAAA,CAAE,CAAA,EAE7D,EAIF1B,EAAAA,IAAC,QAAA,CACC,KAAM0B,EAAQ,YAAc,WAAa,OACzC,MAAOA,EAAQ,YAAc,WAAaoB,EAC1C,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,KAAK,EAChF,SAAU,CAACtB,EAAQ,YAAcqB,GAAarB,EAAQ,YACtD,UAAU,2QAAA,CAAA,EAKVuB,EAAiBvB,GAA8B,CACnD,MAAMC,EAAUE,EAAWH,CAAO,EAC5BqB,EAAYnC,IAAWe,EACvBuB,EAAUzB,EAAWC,CAAO,EAC5BoB,EAAelB,EAAgBF,CAAO,EAG5C,GAAIA,EAAQ,YAAc,QAAUA,EAAQ,MAAQ,oBAAqB,CACvE,MAAMoC,EAAaF,EAAgBd,CAAY,EAC/C,OACE5C,EAAAA,KAAC,MAAA,CAAkB,UAAU,2DAC3B,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,yCACb,SAAA,CAAAA,OAAC,MAAA,CACC,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,yCACb,SAAA,EAAE,iBAAiB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,GAAG,EACpE,EACC,CAACA,EAAQ,YACR1B,EAAAA,IAAC,QAAK,UAAU,mFACb,SAAA,EAAE,mBAAmB,CAAA,CACxB,EAEFA,EAAAA,IAAC,OAAA,CAAK,UAAU,6FACb,WAAQ,MAAA,CACX,CAAA,EACF,EACC0B,EAAQ,aACP1B,EAAAA,IAAC,IAAA,CAAE,UAAU,4CACV,SAAA,EAAE,yBAAyB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,WAAW,CAAA,CACpF,CAAA,EAEJ,EACCA,EAAQ,YAAcA,EAAQ,SAAW,YACxC1B,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMuC,EAAYb,EAAQ,SAAUA,EAAQ,GAAG,EACxD,UAAU,mHACV,MAAO,EAAE,sBAAsB,EAE/B,SAAA1B,EAAAA,IAACuD,EAAAA,QAAA,CAAQ,UAAU,SAAA,CAAU,CAAA,CAAA,CAC/B,EAEJ,QACC,MAAA,CAAI,UAAU,uBACZ,SAAAO,EAAW,IAAKC,GACf/D,EAAAA,IAAC,OAAA,CAEC,UAAU,uFAET,SAAA+D,CAAA,EAHIA,CAAA,CAKR,CAAA,CACH,CAAA,CAAA,EAzCQpC,CA0CV,CAEJ,CAEA,OACEzB,EAAAA,KAAC,MAAA,CAEC,UAAU,6FAEV,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,sBACb,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,yCACb,SAAA,EAAE,iBAAiB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,GAAG,EACpE,EACC,CAACA,EAAQ,YACR1B,EAAAA,IAAC,QAAK,UAAU,mFACb,SAAA,EAAE,mBAAmB,CAAA,CACxB,EAEFA,EAAAA,IAAC,OAAA,CAAK,UAAW,+BACf0B,EAAQ,SAAW,WACf,iEACAA,EAAQ,SAAW,cACnB,wDACA,sDACN,GACG,WAAQ,MAAA,CACX,CAAA,EACF,EACCA,EAAQ,aACP1B,EAAAA,IAAC,IAAA,CAAE,UAAU,4CACV,SAAA,EAAE,yBAAyB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,WAAW,EACpF,EAEDA,EAAQ,YACPxB,OAAC,IAAA,CAAE,UAAU,mEACX,SAAA,CAAAF,EAAAA,IAACmD,EAAAA,MAAA,CAAM,UAAU,SAAA,CAAU,EAC1B,EAAE,sBAAuB,CAAE,KAAMzB,EAAQ,mBAAoB,KAAM,IAAI,KAAKA,EAAQ,UAAU,EAAE,eAAA,EAAkB,CAAA,CAAA,CACrH,CAAA,EAEJ,EAEAxB,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACZ,SAAA,CAAA2D,EAAmBnC,EAASoB,EAAcC,CAAS,EAEnDrB,EAAQ,YAAcA,EAAQ,SAAW,YACxCxB,OAAAkD,EAAAA,SAAA,CACE,SAAA,CAAApD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMqC,EAAWX,CAAO,EACjC,SAAU,CAACwB,GAAWH,EACtB,UAAU,4KACV,MAAO,EAAE,aAAa,EAErB,SAAAA,QAAa9C,EAAAA,QAAA,CAAQ,UAAU,uBAAuB,EAAKD,EAAAA,IAACqD,EAAAA,KAAA,CAAK,UAAU,SAAA,CAAU,CAAA,CAAA,EAExFrD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMsC,EAAYZ,CAAO,EAClC,SAAUA,EAAQ,QAAUA,EAAQ,cAAgBqB,EACpD,UAAU,mKACV,MAAO,EAAE,yBAAyB,EAElC,SAAA/C,EAAAA,IAACsD,EAAAA,UAAA,CAAU,UAAU,SAAA,CAAU,CAAA,CAAA,EAEjCtD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMuC,EAAYb,EAAQ,SAAUA,EAAQ,GAAG,EACxD,UAAU,mHACV,MAAO,EAAE,sBAAsB,EAE/B,SAAA1B,EAAAA,IAACuD,EAAAA,QAAA,CAAQ,UAAU,SAAA,CAAU,CAAA,CAAA,CAC/B,CAAA,CACF,CAAA,CAAA,CAEJ,CAAA,CAAA,EAlEK5B,CAAA,CAqEX,EAEA,OACEzB,EAAAA,KAAC,MAAA,CAAI,UAAU,YACb,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,iHACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,oIACb,SAAAE,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,iDACb,eAACzB,EAAAA,OAAA,CAAO,UAAU,yCAAyC,CAAA,CAC7D,SACC,MAAA,CACC,SAAA,CAAAyB,MAAC,KAAA,CAAG,UAAU,mDACX,SAAA,EAAE,8BAA8B,EACnC,QACC,IAAA,CAAE,UAAU,uCACV,SAAA,EAAE,yCAAyC,CAAA,CAC9C,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,QACC,MAAA,CAAI,UAAU,YACZ,SAAA0D,EAAmB,OAAS,EAC3BA,EAAmB,IAAIT,CAAa,QAEnC,MAAA,CAAI,UAAU,gDACZ,SAAA,EAAE,qBAAqB,EAC1B,CAAA,CAEJ,CAAA,EACF,EAECnC,GACCd,EAAAA,IAACgC,EAAA,CACC,SAAUlB,EAAa,SACvB,WAAYA,EAAa,IACzB,QAAAE,EACA,QAASE,EACT,QAASsB,CAAA,CAAA,CACX,EAEJ,CAEJ,CC9NO,SAASwB,GAAyC,CACvD,KAAM,CAAE,CAAA,EAAMtF,EAAAA,eAAe,OAAO,EAC9B,CAAE,SAAAK,EAAU,gBAAAyB,CAAA,EAAoBC,mBAAA,EAChC,CAACC,EAAcC,CAAe,EAAI1B,EAAAA,SAAiC,CAAA,CAAE,EACrE,CAAC2B,EAAQC,CAAS,EAAI5B,EAAAA,SAAwB,IAAI,EAClD,CAAC6B,EAAcC,CAAe,EAAI9B,EAAAA,SAAmD,IAAI,EACzF,CAAC+B,EAASC,CAAU,EAAIhC,EAAAA,SAA8B,CAAA,CAAE,EACxD,CAACiC,EAAgBC,CAAiB,EAAIlC,EAAAA,SAAS,EAAK,EAGpDgF,EAAoBlF,EAAS,OAAO4E,GAAKA,EAAE,WAAa,WAAW,EAEnEvC,EAAoB,CAACC,EAAkBC,EAAaC,IAAkB,CAC1EZ,EAAgBa,IAAS,CACvB,GAAGA,EACH,CAAC,GAAGH,CAAQ,IAAIC,CAAG,EAAE,EAAGC,CAAA,EACxB,CACJ,EAEME,EAAcC,GAA8B,CAChD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAClD,OAAOhB,EAAaiB,CAAO,IAAM,QAAajB,EAAaiB,CAAO,IAAMD,EAAQ,KAClF,EAEME,EAAmBF,GAA8B,CACrD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAClD,OAAOhB,EAAaiB,CAAO,GAAKD,EAAQ,KAC1C,EAEMW,EAAa,MAAOX,GAA8B,CACtD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAC5CI,EAAWpB,EAAaiB,CAAO,EACrC,GAAI,EAAAG,IAAa,QAAaA,IAAaJ,EAAQ,OAEnD,GAAI,CACFb,EAAUc,CAAO,EACjB,MAAMpC,EAAAA,SAAS,SAAS,OAAOmC,EAAQ,SAAUA,EAAQ,IAAKI,CAAQ,EACtEnB,EAAgBa,GAAQ,CACtB,MAAMO,EAAO,CAAE,GAAGP,CAAA,EAClB,cAAOO,EAAKJ,CAAO,EACZI,CACT,CAAC,EACD,MAAMvB,EAAA,CACR,OAAShB,EAAK,CACZ,QAAQ,MAAM,0BAA2BA,CAAG,CAC9C,QAAA,CACEqB,EAAU,IAAI,CAChB,CACF,EAEMyB,EAAc,MAAOZ,GAA8B,CACvD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAClD,GAAI,CACFb,EAAUc,CAAO,EACjB,MAAMpC,EAAAA,SAAS,SAAS,MAAMmC,EAAQ,SAAUA,EAAQ,GAAG,EAC3Df,EAAgBa,GAAQ,CACtB,MAAMO,EAAO,CAAE,GAAGP,CAAA,EAClB,cAAOO,EAAKJ,CAAO,EACZI,CACT,CAAC,EACD,MAAMvB,EAAA,CACR,OAAShB,EAAK,CACZ,QAAQ,MAAM,2BAA4BA,CAAG,CAC/C,QAAA,CACEqB,EAAU,IAAI,CAChB,CACF,EAEM0B,EAAc,MAAOlB,EAAkBC,IAAgB,CAC3DP,EAAgB,CAAE,SAAAM,EAAU,IAAAC,EAAK,EACjCH,EAAkB,EAAI,EACtB,GAAI,CACF,MAAM7B,EAAS,MAAMC,EAAAA,SAAS,SAAS,WAAW8B,EAAUC,CAAG,EAC/DL,EAAW3B,CAAM,CACnB,OAASE,EAAK,CACZ,QAAQ,MAAM,0BAA2BA,CAAG,CAC9C,QAAA,CACE2B,EAAkB,EAAK,CACzB,CACF,EAGM+C,EAAeC,GAAyB,CAC5C,MAAMC,EAAQ,KAAK,MAAMD,EAAO,GAAG,EAC7BE,EAAgBF,EAAO,IAC7B,OAAIE,IAAkB,EACb,GAAGD,CAAK,IAAI,EAAE,gBAAgB,CAAC,GAEjC,GAAGA,CAAK,IAAI,EAAE,gBAAgB,CAAC,IAAIC,CAAa,IAAI,EAAE,eAAe,CAAC,EAC/E,EAEMC,EAAuB5C,GACpB,CAACA,EAAQ,aAAeA,EAAQ,MAAQ,iBAAmBA,EAAQ,MAAQ,cAG9EmC,EAAqB,CAACnC,EAA2BoB,EAAsBC,EAAoB3B,IAC3FM,EAAQ,YAAc,OAEtBxB,EAAAA,KAAC,QAAA,CAAM,UAAU,sCACf,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,UAAW,SAAA0B,EAAQ,IAAI,EACvC1B,EAAAA,IAAC,QAAA,CACC,KAAK,WACL,QAAS8C,IAAiB,OAC1B,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,QAAU,OAAS,OAAO,EACrG,SAAU,CAACtB,EAAQ,YAAcqB,EACjC,UAAU,eACV,aAAYrB,EAAQ,GAAA,CAAA,EAEtB1B,EAAAA,IAAC,MAAA,CAAI,UAAU,ifAAA,CAAkf,CAAA,EACngB,EAGA0B,EAAQ,YAAc,MAEtBxB,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,QAAA,CACC,KAAK,SACL,MAAO8C,EACP,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,KAAK,EAChF,SAAU,CAACtB,EAAQ,YAAcqB,EACjC,UAAU,2QAAA,CAAA,GAEVrB,EAAQ,MAAQ,iBAAmBA,EAAQ,MAAQ,qBACnDxB,EAAAA,KAAC,OAAA,CAAK,UAAU,uCAAuC,SAAA,CAAA,IACnDgE,EAAY,SAASpB,CAAY,GAAK,CAAC,EAAE,GAAA,CAAA,CAC7C,CAAA,EAEJ,EAIF9C,EAAAA,IAAC,QAAA,CACC,KAAM0B,EAAQ,YAAc,WAAa,OACzC,MAAOA,EAAQ,YAAc,WAAaoB,EAC1C,SAAWE,GAAM5B,EAAkBM,EAAQ,SAAUA,EAAQ,IAAKsB,EAAE,OAAO,KAAK,EAChF,SAAU,CAACtB,EAAQ,YAAcqB,GAAarB,EAAQ,YACtD,UAAU,2QAAA,CAAA,EAKViB,EAAuBC,GACvBA,IAAW,WAAmB,iEAC9BA,IAAW,cAAsB,wDAC9B,uDAGHK,EAAiBvB,GAA8B,CACnD,MAAMC,EAAU,GAAGD,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAC5CqB,EAAYnC,IAAWe,EACvBuB,EAAUzB,EAAWC,CAAO,EAC5BoB,EAAelB,EAAgBF,CAAO,EACtC6C,EAAqBD,EAAoB5C,CAAO,EAEtD,OACExB,EAAAA,KAAC,MAAA,CAEC,UAAW,8FACTqE,EAAqB,oCAAsC,EAC7D,GAEA,SAAA,CAAArE,EAAAA,KAAC,MAAA,CAAI,UAAU,sBACb,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,yCACb,SAAA,EAAE,iBAAiB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,GAAG,EACpE,EACC6C,GACCrE,EAAAA,KAAC,OAAA,CAAK,UAAU,8IACd,SAAA,CAAAF,EAAAA,IAACwE,EAAAA,OAAA,CAAO,UAAU,SAAA,CAAU,EAC3B,EAAE,2BAA2B,CAAA,EAChC,EAED,CAAC9C,EAAQ,YAAc,CAAC6C,GACvBvE,EAAAA,IAAC,OAAA,CAAK,UAAU,mFACb,SAAA,EAAE,mBAAmB,CAAA,CACxB,EAEFA,EAAAA,IAAC,OAAA,CAAK,UAAW,+BAA+B2C,EAAoBjB,EAAQ,MAAM,CAAC,GAChF,SAAAA,EAAQ,MAAA,CACX,CAAA,EACF,EACCA,EAAQ,aACP1B,EAAAA,IAAC,IAAA,CAAE,UAAU,4CACV,SAAA,EAAE,yBAAyB0B,EAAQ,QAAQ,IAAIA,EAAQ,GAAG,GAAIA,EAAQ,WAAW,EACpF,EAEDA,EAAQ,YACPxB,OAAC,IAAA,CAAE,UAAU,mEACX,SAAA,CAAAF,EAAAA,IAACmD,EAAAA,MAAA,CAAM,UAAU,SAAA,CAAU,EAC1B,EAAE,sBAAuB,CAAE,KAAMzB,EAAQ,mBAAoB,KAAM,IAAI,KAAKA,EAAQ,UAAU,EAAE,eAAA,EAAkB,CAAA,CAAA,CACrH,CAAA,EAEJ,EAEAxB,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACZ,SAAA,CAAA2D,EAAmBnC,EAASoB,EAAcC,EAAW3B,CAAiB,EAEtEM,EAAQ,YAAcA,EAAQ,SAAW,YACxCxB,OAAAkD,EAAAA,SAAA,CACE,SAAA,CAAApD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMqC,EAAWX,CAAO,EACjC,SAAU,CAACwB,GAAWH,EACtB,UAAU,4KACV,MAAO,EAAE,aAAa,EAErB,SAAAA,QAAa9C,EAAAA,QAAA,CAAQ,UAAU,uBAAuB,EAAKD,EAAAA,IAACqD,EAAAA,KAAA,CAAK,UAAU,SAAA,CAAU,CAAA,CAAA,EAExFrD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMsC,EAAYZ,CAAO,EAClC,SAAUA,EAAQ,QAAUA,EAAQ,cAAgBqB,EACpD,UAAU,mKACV,MAAO,EAAE,yBAAyB,EAElC,SAAA/C,EAAAA,IAACsD,EAAAA,UAAA,CAAU,UAAU,SAAA,CAAU,CAAA,CAAA,EAEjCtD,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMuC,EAAYb,EAAQ,SAAUA,EAAQ,GAAG,EACxD,UAAU,mHACV,MAAO,EAAE,sBAAsB,EAE/B,SAAA1B,EAAAA,IAACuD,EAAAA,QAAA,CAAQ,UAAU,SAAA,CAAU,CAAA,CAAA,CAC/B,CAAA,CACF,CAAA,CAAA,CAEJ,CAAA,CAAA,EApEK5B,CAAA,CAuEX,EAEA,OACEzB,EAAAA,KAAC,MAAA,CAAI,UAAU,YAEb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,gGACb,SAAAE,EAAAA,KAAC,MAAA,CAAI,UAAU,yBACb,SAAA,CAAAF,EAAAA,IAACyE,EAAAA,cAAA,CAAc,UAAU,yDAAA,CAA0D,SAClF,MAAA,CACC,SAAA,CAAAzE,MAAC,KAAA,CAAG,UAAU,yCACX,SAAA,EAAE,4BAA4B,EACjC,QACC,IAAA,CAAE,UAAU,qDACV,SAAA,EAAE,kCAAkC,CAAA,CACvC,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,EAEAE,EAAAA,KAAC,MAAA,CAAI,UAAU,iHACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,oGACb,SAAAE,EAAAA,KAAC,MAAA,CAAI,UAAU,0BACb,SAAA,CAAAF,EAAAA,IAAC,OAAI,UAAU,iCACb,eAACxB,EAAAA,SAAA,CAAS,UAAU,yBAAyB,CAAA,CAC/C,SACC,MAAA,CACC,SAAA,CAAAwB,MAAC,KAAA,CAAG,UAAU,mDACX,SAAA,EAAE,6BAA6B,EAClC,QACC,IAAA,CAAE,UAAU,uCACV,SAAA,EAAE,wCAAwC,CAAA,CAC7C,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,QACC,MAAA,CAAI,UAAU,YACZ,SAAAiE,EAAkB,OAAS,EAC1BA,EAAkB,IAAIhB,CAAa,QAElC,MAAA,CAAI,UAAU,gDACZ,SAAA,EAAE,qBAAqB,EAC1B,CAAA,CAEJ,CAAA,EACF,EAGCnC,SACE,MAAA,CAAI,UAAU,qCACb,SAAAZ,EAAAA,KAAC,MAAA,CAAI,UAAU,qDACb,SAAA,CAAAF,EAAAA,IAAC,SAAA,CAAO,KAAK,SAAS,UAAU,4BAA4B,QAAS,IAAMe,EAAgB,IAAI,EAAG,aAAW,aAAA,CAAc,EAC3Hb,EAAAA,KAAC,MAAA,CAAI,UAAU,2FACb,SAAA,CAAAF,EAAAA,IAAC,KAAA,CAAG,UAAU,wDACX,SAAA,EAAE,wBAAyB,CAAE,IAAK,GAAGc,EAAa,QAAQ,IAAIA,EAAa,GAAG,EAAA,CAAI,EACrF,EAECI,SACE,MAAA,CAAI,UAAU,2BACb,SAAAlB,EAAAA,IAACC,EAAAA,QAAA,CAAQ,UAAU,qDAAA,CAAsD,CAAA,CAC3E,EAED,CAACiB,GAAkBF,EAAQ,SAAW,GACrChB,EAAAA,IAAC,IAAA,CAAE,UAAU,gDACV,SAAA,EAAE,oBAAoB,CAAA,CACzB,EAED,CAACkB,GAAkBF,EAAQ,OAAS,SAClC,MAAA,CAAI,UAAU,qCACZ,SAAAA,EAAQ,IAAKmB,GACZjC,EAAAA,KAAC,MAAA,CAAmB,UAAU,0CAC5B,SAAA,CAAAA,EAAAA,KAAC,MAAA,CAAI,UAAU,+BACb,SAAA,CAAAF,EAAAA,IAAC,QAAK,UAAU,+BACb,WAAM,mBAAqB,EAAE,iBAAiB,EACjD,EACAA,EAAAA,IAAC,OAAA,CAAK,UAAU,8BACb,SAAA,IAAI,KAAKmC,EAAM,SAAS,EAAE,eAAA,CAAe,CAC5C,CAAA,EACF,EACAjC,EAAAA,KAAC,MAAA,CAAI,UAAU,eACb,SAAA,CAAAF,EAAAA,IAAC,OAAA,CAAK,UAAU,wCAAyC,SAAAmC,EAAM,SAAS,EACxEnC,EAAAA,IAAC,OAAA,CAAK,UAAU,mCAAmC,SAAA,IAAC,EACpDA,EAAAA,IAAC,OAAA,CAAK,UAAU,6BAA8B,WAAM,QAAA,CAAS,CAAA,CAAA,CAC/D,CAAA,CAAA,EAbQmC,EAAM,EAchB,CACD,EACH,EAGFnC,EAAAA,IAAC,MAAA,CAAI,UAAU,wBACb,SAAAA,EAAAA,IAAC,SAAA,CACC,QAAS,IAAMe,EAAgB,IAAI,EACnC,UAAU,yHAET,WAAE,eAAe,CAAA,CAAA,CACpB,CACF,CAAA,CAAA,CACF,CAAA,CAAA,CACF,CAAA,CACF,CAAA,EAEJ,CAEJ"}
|
|
1
|
+
{"version":3,"file":"index-Betxo5g5.js","sources":["../../src/pages/platform/administration/configuration/settings/SettingsPage.tsx","../../src/hooks/useSettingsSection.ts","../../src/components/platform/administration/settings/SettingsHistoryModal.tsx","../../src/pages/platform/administration/configuration/settings/GeneralSettingsSection.tsx","../../src/pages/platform/administration/configuration/settings/FileUploadSettingsSection.tsx","../../src/pages/platform/administration/configuration/settings/LegalFileSettingsSection.tsx"],"sourcesContent":["import { useState, useEffect, useCallback } from 'react';\r\nimport type { ReactElement } from 'react';\r\nimport { useTranslation } from 'react-i18next';\r\nimport { useNavigate, useLocation, Outlet } from 'react-router-dom';\r\nimport { Breadcrumb } from '@/components/ui/Breadcrumb';\r\nimport { Settings, Upload, FileText, Loader2 } from 'lucide-react';\r\nimport { adminApi, type SettingViewModel } from '@/services/api/adminApi';\r\n\r\ninterface TabConfig {\r\n id: string;\r\n labelKey: string;\r\n icon: React.ElementType;\r\n route: string;\r\n}\r\n\r\nconst TABS: TabConfig[] = [\r\n { id: 'general', labelKey: 'settings.tabs.general', icon: Settings, route: '/administration/configuration/settings/general' },\r\n { id: 'file-upload', labelKey: 'settings.tabs.fileUpload', icon: Upload, route: '/administration/configuration/settings/file-upload' },\r\n { id: 'legal-file', labelKey: 'settings.tabs.legalFile', icon: FileText, route: '/administration/configuration/settings/legal-file' }\r\n];\r\n\r\nexport function SettingsPage(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const navigate = useNavigate();\r\n const location = useLocation();\r\n const [settings, setSettings] = useState<SettingViewModel[]>([]);\r\n const [loading, setLoading] = useState(true);\r\n\r\n const loadSettings = useCallback(async () => {\r\n try {\r\n setLoading(true);\r\n const result = await adminApi.settings.getAll();\r\n setSettings(result);\r\n } catch (err) {\r\n console.error('Failed to load settings:', err);\r\n } finally {\r\n setLoading(false);\r\n }\r\n }, []);\r\n\r\n useEffect(() => {\r\n loadSettings();\r\n }, [loadSettings]);\r\n\r\n const getCurrentTab = () => {\r\n const path = location.pathname;\r\n const tab = TABS.find(t => path.includes(t.id));\r\n return tab?.id || 'general';\r\n };\r\n\r\n const handleTabChange = (tabId: string) => {\r\n const tab = TABS.find(t => t.id === tabId);\r\n if (tab) {\r\n navigate(tab.route);\r\n }\r\n };\r\n\r\n if (loading) {\r\n return (\r\n <div className=\"flex items-center justify-center min-h-[400px]\">\r\n <Loader2 className=\"w-8 h-8 animate-spin text-[var(--color-accent-600)]\" />\r\n </div>\r\n );\r\n }\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n <Breadcrumb\r\n items={[\r\n { label: t('header.title'), href: '/administration' },\r\n { label: t('settings.title', 'Settings') }\r\n ]}\r\n />\r\n\r\n {/* Header */}\r\n <div className=\"flex items-center justify-between\">\r\n <div>\r\n <h1 className=\"text-2xl font-bold text-[var(--text-primary)]\">\r\n {t('settings.title')}\r\n </h1>\r\n <p className=\"text-[var(--text-secondary)] mt-1\">\r\n {t('settings.subtitle')}\r\n </p>\r\n </div>\r\n </div>\r\n\r\n {/* Tabs */}\r\n <div className=\"border-b border-[var(--border-color)]\">\r\n <nav className=\"-mb-px flex space-x-8\" aria-label=\"Tabs\">\r\n {TABS.map((tab) => {\r\n const Icon = tab.icon;\r\n const isActive = getCurrentTab() === tab.id;\r\n return (\r\n <button\r\n key={tab.id}\r\n onClick={() => handleTabChange(tab.id)}\r\n className={`\r\n group inline-flex items-center py-4 px-1 border-b-2 font-medium text-sm transition-colors\r\n ${isActive\r\n ? 'border-[var(--color-accent-500)] text-[var(--color-accent-600)]'\r\n : 'border-transparent text-[var(--text-secondary)] hover:text-[var(--text-primary)] hover:border-[var(--border-color)]'\r\n }\r\n `}\r\n >\r\n <Icon className={`mr-2 h-5 w-5 ${isActive ? 'text-[var(--color-accent-500)]' : 'text-[var(--text-tertiary)] group-hover:text-[var(--text-secondary)]'}`} />\r\n {t(tab.labelKey)}\r\n </button>\r\n );\r\n })}\r\n </nav>\r\n </div>\r\n\r\n {/* Content */}\r\n <Outlet context={{ settings, refreshSettings: loadSettings }} />\r\n </div>\r\n );\r\n}\r\n\r\nexport default SettingsPage;\r\n","import { useState } from 'react';\r\nimport { useOutletContext } from 'react-router-dom';\r\nimport { adminApi, type SettingViewModel, type SettingHistoryDto } from '@/services/api/adminApi';\r\n\r\ninterface SettingsContext {\r\n settings: SettingViewModel[];\r\n refreshSettings: () => Promise<void>;\r\n}\r\n\r\nexport function useSettingsSection(): {\n settings: SettingViewModel[];\n editedValues: Record<string, string>;\n saving: string | null;\n historyModal: { category: string; key: string } | null;\n history: SettingHistoryDto[];\n loadingHistory: boolean;\n handleValueChange: (category: string, key: string, value: string) => void;\n hasChanges: (setting: SettingViewModel) => boolean;\n getDisplayValue: (setting: SettingViewModel) => string;\n getEditKey: (setting: SettingViewModel) => string;\n handleSave: (setting: SettingViewModel) => Promise<void>;\n handleReset: (setting: SettingViewModel) => Promise<void>;\n openHistory: (category: string, key: string) => Promise<void>;\n closeHistory: () => void;\n} {\r\n const { settings, refreshSettings } = useOutletContext<SettingsContext>();\r\n const [editedValues, setEditedValues] = useState<Record<string, string>>({});\r\n const [saving, setSaving] = useState<string | null>(null);\r\n const [historyModal, setHistoryModal] = useState<{ category: string; key: string } | null>(null);\r\n const [history, setHistory] = useState<SettingHistoryDto[]>([]);\r\n const [loadingHistory, setLoadingHistory] = useState(false);\r\n\r\n const handleValueChange = (category: string, key: string, value: string) => {\r\n setEditedValues(prev => ({\r\n ...prev,\r\n [`${category}.${key}`]: value\r\n }));\r\n };\r\n\r\n const hasChanges = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] !== undefined && editedValues[editKey] !== setting.value;\r\n };\r\n\r\n const getDisplayValue = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] ?? setting.value;\r\n };\r\n\r\n const getEditKey = (setting: SettingViewModel) => `${setting.category}.${setting.key}`;\r\n\r\n const handleSave = async (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n const newValue = editedValues[editKey];\r\n if (newValue === undefined || newValue === setting.value) return;\r\n\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.update(setting.category, setting.key, newValue);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to save setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const handleReset = async (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.reset(setting.category, setting.key);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to reset setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const openHistory = async (category: string, key: string) => {\r\n setHistoryModal({ category, key });\r\n setLoadingHistory(true);\r\n try {\r\n const result = await adminApi.settings.getHistory(category, key);\r\n setHistory(result);\r\n } catch (err) {\r\n console.error('Failed to load history:', err);\r\n } finally {\r\n setLoadingHistory(false);\r\n }\r\n };\r\n\r\n const closeHistory = () => setHistoryModal(null);\r\n\r\n return {\r\n settings,\r\n editedValues,\r\n saving,\r\n historyModal,\r\n history,\r\n loadingHistory,\r\n handleValueChange,\r\n hasChanges,\r\n getDisplayValue,\r\n getEditKey,\r\n handleSave,\r\n handleReset,\r\n openHistory,\r\n closeHistory,\r\n };\r\n}\r\n\r\nexport type { SettingsContext };\r\n","import type { ReactElement } from 'react';\r\nimport { useTranslation } from 'react-i18next';\r\nimport { Loader2 } from 'lucide-react';\r\nimport type { SettingHistoryDto } from '@/services/api/adminApi';\r\n\r\ninterface SettingsHistoryModalProps {\r\n readonly category: string;\r\n readonly settingKey: string;\r\n readonly history: SettingHistoryDto[];\r\n readonly loading: boolean;\r\n readonly onClose: () => void;\r\n}\r\n\r\nexport function SettingsHistoryModal({ category, settingKey, history, loading, onClose }: SettingsHistoryModalProps): ReactElement {\r\n const { t } = useTranslation('admin');\r\n\r\n return (\r\n <div className=\"fixed inset-0 z-50 overflow-y-auto\">\r\n <div className=\"flex items-center justify-center min-h-screen px-4\">\r\n <button type=\"button\" className=\"fixed inset-0 bg-black/50\" onClick={onClose} aria-label=\"Close modal\" />\r\n <div className=\"relative rounded-[var(--radius-card)] bg-[var(--bg-card)] shadow-xl max-w-2xl w-full p-6\">\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)] mb-4\">\r\n {t('settings.historyTitle', { key: `${category}.${settingKey}` })}\r\n </h3>\r\n\r\n {loading && (\r\n <div className=\"flex justify-center py-8\">\r\n <Loader2 className=\"w-8 h-8 animate-spin text-[var(--color-accent-600)]\" />\r\n </div>\r\n )}\r\n {!loading && history.length === 0 && (\r\n <p className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noHistory')}\r\n </p>\r\n )}\r\n {!loading && history.length > 0 && (\r\n <div className=\"space-y-3 max-h-96 overflow-y-auto\">\r\n {history.map((entry) => (\r\n <div key={entry.id} className=\"p-3 bg-[var(--bg-secondary)] rounded-lg\">\r\n <div className=\"flex justify-between text-sm\">\r\n <span className=\"text-[var(--text-secondary)]\">\r\n {entry.changedByUserName || t('settings.system')}\r\n </span>\r\n <span className=\"text-[var(--text-tertiary)]\">\r\n {new Date(entry.changedAt).toLocaleString()}\r\n </span>\r\n </div>\r\n <div className=\"mt-2 text-sm\">\r\n <span className=\"text-[var(--error-text)] line-through\">{entry.oldValue}</span>\r\n <span className=\"mx-2 text-[var(--text-tertiary)]\">→</span>\r\n <span className=\"text-[var(--success-text)]\">{entry.newValue}</span>\r\n </div>\r\n </div>\r\n ))}\r\n </div>\r\n )}\r\n\r\n <div className=\"mt-6 flex justify-end\">\r\n <button\r\n onClick={onClose}\r\n className=\"px-4 py-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n >\r\n {t('common.cancel')}\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n );\r\n}\r\n","import type { ReactElement } from 'react';\nimport { useTranslation } from 'react-i18next';\r\nimport { Save, RotateCcw, Loader2, Clock, History, Settings as SettingsIcon } from 'lucide-react';\r\nimport type { SettingViewModel } from '@/services/api/adminApi';\r\nimport { useSettingsSection } from '@/hooks/useSettingsSection';\r\nimport { SettingsHistoryModal } from '@/components/platform/administration/settings/SettingsHistoryModal';\r\n\r\nexport function GeneralSettingsSection(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const {\r\n settings,\r\n saving,\r\n historyModal,\r\n history,\r\n loadingHistory,\r\n handleValueChange,\r\n hasChanges,\r\n getDisplayValue,\r\n getEditKey,\r\n handleSave,\r\n handleReset,\r\n openHistory,\r\n closeHistory,\r\n } = useSettingsSection();\r\n\r\n // Filter settings for Session and General categories\r\n const sessionSettings = settings.filter(s => s.category === 'Session');\r\n const generalSettings = settings.filter(s => s.category === 'General');\r\n\r\n const getSourceBadgeClass = (source: string): string => {\r\n if (source === 'Database') {\r\n return 'bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]';\r\n }\r\n if (source === 'AppSettings') {\r\n return 'bg-purple-500/10 text-purple-600 dark:text-purple-400';\r\n }\r\n return 'bg-[var(--bg-secondary)] text-[var(--text-tertiary)]';\r\n };\r\n\r\n const getSettingInput = (setting: SettingViewModel, displayValue: string, isEditing: boolean): React.ReactNode => {\r\n if (setting.valueType === 'Bool') {\r\n return (\r\n <label className=\"relative inline-flex items-center cursor-pointer\">\r\n <input\r\n type=\"checkbox\"\r\n checked={displayValue === 'true'}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.checked ? 'true' : 'false')}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"sr-only peer\"\r\n />\r\n <div className=\"w-11 h-6 bg-[var(--bg-tertiary)] peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-[var(--color-accent-500)]/30 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-[var(--border-color)] after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[var(--color-accent-600)] peer-disabled:opacity-50 peer-disabled:cursor-not-allowed\"></div>\r\n </label>\r\n );\r\n }\r\n if (setting.valueType === 'Int' || setting.valueType === 'Decimal') {\r\n return (\r\n <input\r\n type=\"number\"\r\n value={displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n }\r\n return (\r\n <input\r\n type={setting.isSensitive ? 'password' : 'text'}\r\n value={setting.isSensitive ? '********' : displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing || setting.isSensitive}\r\n className=\"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n };\r\n\r\n const renderSetting = (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n const isEditing = saving === editKey;\r\n const changed = hasChanges(setting);\r\n const displayValue = getDisplayValue(setting);\r\n\r\n return (\r\n <div\r\n key={editKey}\r\n className=\"flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0\"\r\n >\r\n <div className=\"flex-1 min-w-0 pr-4\">\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {!setting.isEditable && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className={`px-2 py-0.5 text-xs rounded ${getSourceBadgeClass(setting.source)}`}>\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n {setting.modifiedAt && (\r\n <p className=\"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1\">\r\n <Clock className=\"w-3 h-3\" />\r\n {t('settings.modifiedBy', { user: setting.modifiedByUserName, date: new Date(setting.modifiedAt).toLocaleString() })}\r\n </p>\r\n )}\r\n </div>\r\n\r\n <div className=\"flex items-center gap-2\">\r\n {getSettingInput(setting, displayValue, isEditing)}\r\n\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <>\r\n <button\r\n onClick={() => handleSave(setting)}\r\n disabled={!changed || isEditing}\r\n className=\"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('common.save')}\r\n >\r\n {isEditing ? <Loader2 className=\"w-4 h-4 animate-spin\" /> : <Save className=\"w-4 h-4\" />}\r\n </button>\r\n <button\r\n onClick={() => handleReset(setting)}\r\n disabled={setting.value === setting.defaultValue || isEditing}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('settings.resetToDefault')}\r\n >\r\n <RotateCcw className=\"w-4 h-4\" />\r\n </button>\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n </>\r\n )}\r\n </div>\r\n </div>\r\n );\r\n };\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n {/* Session Settings */}\r\n {sessionSettings.length > 0 && (\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-[var(--color-accent-500)]/20\">\r\n <Clock className=\"w-5 h-5 text-[var(--color-accent-600)]\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.session')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.sessionDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {sessionSettings.map(renderSetting)}\r\n </div>\r\n </div>\r\n )}\r\n\r\n {/* General Settings */}\r\n {generalSettings.length > 0 && (\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-[var(--color-accent-500)]/20\">\r\n <SettingsIcon className=\"w-5 h-5 text-[var(--color-accent-600)]\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.general')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.generalDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {generalSettings.map(renderSetting)}\r\n </div>\r\n </div>\r\n )}\r\n\r\n {sessionSettings.length === 0 && generalSettings.length === 0 && (\r\n <div className=\"text-center py-12 text-[var(--text-secondary)]\">\r\n {t('settings.noSettings')}\r\n </div>\r\n )}\r\n\r\n {historyModal && (\r\n <SettingsHistoryModal\r\n category={historyModal.category}\r\n settingKey={historyModal.key}\r\n history={history}\r\n loading={loadingHistory}\r\n onClose={closeHistory}\r\n />\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default GeneralSettingsSection;\r\n","import type { ReactElement } from 'react';\nimport { useTranslation } from 'react-i18next';\r\nimport { Save, RotateCcw, Loader2, Upload, History, Clock } from 'lucide-react';\r\nimport type { SettingViewModel } from '@/services/api/adminApi';\r\nimport { useSettingsSection } from '@/hooks/useSettingsSection';\r\nimport { SettingsHistoryModal } from '@/components/platform/administration/settings/SettingsHistoryModal';\r\n\r\nexport function FileUploadSettingsSection(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const {\r\n settings,\r\n saving,\r\n historyModal,\r\n history,\r\n loadingHistory,\r\n handleValueChange,\r\n hasChanges,\r\n getDisplayValue,\r\n getEditKey,\r\n handleSave,\r\n handleReset,\r\n openHistory,\r\n closeHistory,\r\n } = useSettingsSection();\r\n\r\n // Filter settings for FileUpload category\r\n const fileUploadSettings = settings.filter(s => s.category === 'FileUpload');\r\n\r\n // Parse JSON extensions for display\r\n const parseExtensions = (value: string): string[] => {\r\n try {\r\n return JSON.parse(value);\r\n } catch {\r\n return [];\r\n }\r\n };\r\n\r\n const renderSettingInput = (setting: SettingViewModel, displayValue: string, isEditing: boolean) => {\r\n if (setting.valueType === 'Int') {\r\n return (\r\n <div className=\"flex items-center gap-2\">\r\n <input\r\n type=\"number\"\r\n value={displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n {setting.key === 'MaxFileSizeMB' && (\r\n <span className=\"text-sm text-[var(--text-secondary)]\">MB</span>\r\n )}\r\n </div>\r\n );\r\n }\r\n return (\r\n <input\r\n type={setting.isSensitive ? 'password' : 'text'}\r\n value={setting.isSensitive ? '********' : displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing || setting.isSensitive}\r\n className=\"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n };\r\n\r\n const renderSetting = (setting: SettingViewModel) => {\r\n const editKey = getEditKey(setting);\r\n const isEditing = saving === editKey;\r\n const changed = hasChanges(setting);\r\n const displayValue = getDisplayValue(setting);\r\n\r\n // Special rendering for JSON arrays (extensions)\r\n if (setting.valueType === 'Json' && setting.key === 'AllowedExtensions') {\r\n const extensions = parseExtensions(displayValue);\r\n return (\r\n <div key={editKey} className=\"py-4 border-b border-[var(--border-color)] last:border-0\">\r\n <div className=\"flex items-center justify-between mb-3\">\r\n <div>\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {!setting.isEditable && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)] rounded\">\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n </div>\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n )}\r\n </div>\r\n <div className=\"flex flex-wrap gap-2\">\r\n {extensions.map((ext) => (\r\n <span\r\n key={ext}\r\n className=\"px-3 py-1 bg-[var(--bg-secondary)] text-[var(--text-secondary)] rounded-full text-sm\"\r\n >\r\n {ext}\r\n </span>\r\n ))}\r\n </div>\r\n </div>\r\n );\r\n }\r\n\r\n return (\r\n <div\r\n key={editKey}\r\n className=\"flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0\"\r\n >\r\n <div className=\"flex-1 min-w-0 pr-4\">\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {!setting.isEditable && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className={`px-2 py-0.5 text-xs rounded ${\r\n setting.source === 'Database'\r\n ? 'bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]'\r\n : setting.source === 'AppSettings'\r\n ? 'bg-purple-500/10 text-purple-600 dark:text-purple-400'\r\n : 'bg-[var(--bg-secondary)] text-[var(--text-tertiary)]'\r\n }`}>\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n {setting.modifiedAt && (\r\n <p className=\"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1\">\r\n <Clock className=\"w-3 h-3\" />\r\n {t('settings.modifiedBy', { user: setting.modifiedByUserName, date: new Date(setting.modifiedAt).toLocaleString() })}\r\n </p>\r\n )}\r\n </div>\r\n\r\n <div className=\"flex items-center gap-2\">\r\n {renderSettingInput(setting, displayValue, isEditing)}\r\n\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <>\r\n <button\r\n onClick={() => handleSave(setting)}\r\n disabled={!changed || isEditing}\r\n className=\"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('common.save')}\r\n >\r\n {isEditing ? <Loader2 className=\"w-4 h-4 animate-spin\" /> : <Save className=\"w-4 h-4\" />}\r\n </button>\r\n <button\r\n onClick={() => handleReset(setting)}\r\n disabled={setting.value === setting.defaultValue || isEditing}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('settings.resetToDefault')}\r\n >\r\n <RotateCcw className=\"w-4 h-4\" />\r\n </button>\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n </>\r\n )}\r\n </div>\r\n </div>\r\n );\r\n };\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-[var(--color-accent-500)]/10 to-[var(--color-accent-600)]/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-[var(--color-accent-500)]/20\">\r\n <Upload className=\"w-5 h-5 text-[var(--color-accent-600)]\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.fileUpload')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.fileUploadDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {fileUploadSettings.length > 0 ? (\r\n fileUploadSettings.map(renderSetting)\r\n ) : (\r\n <div className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noSettings')}\r\n </div>\r\n )}\r\n </div>\r\n </div>\r\n\r\n {historyModal && (\r\n <SettingsHistoryModal\r\n category={historyModal.category}\r\n settingKey={historyModal.key}\r\n history={history}\r\n loading={loadingHistory}\r\n onClose={closeHistory}\r\n />\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default FileUploadSettingsSection;\r\n","import { useState } from 'react';\r\nimport type { ReactElement } from 'react';\r\nimport { useTranslation } from 'react-i18next';\r\nimport { useOutletContext } from 'react-router-dom';\r\nimport { Save, RotateCcw, Loader2, FileText, History, Clock, AlertTriangle, Shield } from 'lucide-react';\r\nimport { adminApi, type SettingViewModel, type SettingHistoryDto } from '@/services/api/adminApi';\r\n\r\ninterface SettingsContext {\r\n settings: SettingViewModel[];\r\n refreshSettings: () => Promise<void>;\r\n}\r\n\r\nexport function LegalFileSettingsSection(): ReactElement {\r\n const { t } = useTranslation('admin');\r\n const { settings, refreshSettings } = useOutletContext<SettingsContext>();\r\n const [editedValues, setEditedValues] = useState<Record<string, string>>({});\r\n const [saving, setSaving] = useState<string | null>(null);\r\n const [historyModal, setHistoryModal] = useState<{ category: string; key: string } | null>(null);\r\n const [history, setHistory] = useState<SettingHistoryDto[]>([]);\r\n const [loadingHistory, setLoadingHistory] = useState(false);\r\n\r\n // Filter settings for LegalFile category\r\n const legalFileSettings = settings.filter(s => s.category === 'LegalFile');\r\n\r\n const handleValueChange = (category: string, key: string, value: string) => {\r\n setEditedValues(prev => ({\r\n ...prev,\r\n [`${category}.${key}`]: value\r\n }));\r\n };\r\n\r\n const hasChanges = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] !== undefined && editedValues[editKey] !== setting.value;\r\n };\r\n\r\n const getDisplayValue = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n return editedValues[editKey] ?? setting.value;\r\n };\r\n\r\n const handleSave = async (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n const newValue = editedValues[editKey];\r\n if (newValue === undefined || newValue === setting.value) return;\r\n\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.update(setting.category, setting.key, newValue);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to save setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const handleReset = async (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n try {\r\n setSaving(editKey);\r\n await adminApi.settings.reset(setting.category, setting.key);\r\n setEditedValues(prev => {\r\n const next = { ...prev };\r\n delete next[editKey];\r\n return next;\r\n });\r\n await refreshSettings();\r\n } catch (err) {\r\n console.error('Failed to reset setting:', err);\r\n } finally {\r\n setSaving(null);\r\n }\r\n };\r\n\r\n const openHistory = async (category: string, key: string) => {\r\n setHistoryModal({ category, key });\r\n setLoadingHistory(true);\r\n try {\r\n const result = await adminApi.settings.getHistory(category, key);\r\n setHistory(result);\r\n } catch (err) {\r\n console.error('Failed to load history:', err);\r\n } finally {\r\n setLoadingHistory(false);\r\n }\r\n };\r\n\r\n // Convert days to years for display\r\n const daysToYears = (days: number): string => {\r\n const years = Math.floor(days / 365);\r\n const remainingDays = days % 365;\r\n if (remainingDays === 0) {\r\n return `${years} ${t('settings.years')}`;\r\n }\r\n return `${years} ${t('settings.years')} ${remainingDays} ${t('settings.days')}`;\r\n };\r\n\r\n const getLegallyProtected = (setting: SettingViewModel): boolean => {\r\n return !setting.isEditable && (setting.key === 'RetentionDays' || setting.key === 'EnableWorm');\r\n };\r\n\r\n const renderSettingInput = (setting: SettingViewModel, displayValue: string, isEditing: boolean, handleValueChange: (category: string, key: string, value: string) => void) => {\r\n if (setting.valueType === 'Bool') {\r\n return (\r\n <label className=\"relative inline-flex cursor-pointer\">\r\n <span className=\"sr-only\">{setting.key}</span>\r\n <input\r\n type=\"checkbox\"\r\n checked={displayValue === 'true'}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.checked ? 'true' : 'false')}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"sr-only peer\"\r\n aria-label={setting.key}\r\n />\r\n <div className=\"w-11 h-6 bg-[var(--bg-tertiary)] peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-[var(--color-accent-500)]/30 rounded-full peer peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:left-[2px] after:bg-white after:border-[var(--border-color)] after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[var(--color-accent-600)] peer-disabled:opacity-50 peer-disabled:cursor-not-allowed\"></div>\r\n </label>\r\n );\r\n }\r\n if (setting.valueType === 'Int') {\r\n return (\r\n <div className=\"flex items-center gap-2\">\r\n <input\r\n type=\"number\"\r\n value={displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing}\r\n className=\"w-24 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n {(setting.key === 'RetentionDays' || setting.key === 'ArchiveAfterDays') && (\r\n <span className=\"text-sm text-[var(--text-secondary)]\">\r\n ({daysToYears(parseInt(displayValue) || 0)})\r\n </span>\r\n )}\r\n </div>\r\n );\r\n }\r\n return (\r\n <input\r\n type={setting.isSensitive ? 'password' : 'text'}\r\n value={setting.isSensitive ? '********' : displayValue}\r\n onChange={(e) => handleValueChange(setting.category, setting.key, e.target.value)}\r\n disabled={!setting.isEditable || isEditing || setting.isSensitive}\r\n className=\"w-48 px-3 py-2 border border-[var(--border-color)] rounded-[var(--radius-button)] bg-[var(--bg-card)] text-[var(--text-primary)] disabled:opacity-50 disabled:cursor-not-allowed focus:ring-2 focus:ring-[var(--color-accent-500)] focus:border-[var(--color-accent-500)]\"\r\n />\r\n );\r\n };\r\n\r\n const getSourceBadgeClass = (source: string): string => {\r\n if (source === 'Database') return 'bg-[var(--color-accent-500)]/10 text-[var(--color-accent-600)]';\r\n if (source === 'AppSettings') return 'bg-purple-500/10 text-purple-600 dark:text-purple-400';\r\n return 'bg-[var(--bg-secondary)] text-[var(--text-tertiary)]';\r\n };\r\n\r\n const renderSetting = (setting: SettingViewModel) => {\r\n const editKey = `${setting.category}.${setting.key}`;\r\n const isEditing = saving === editKey;\r\n const changed = hasChanges(setting);\r\n const displayValue = getDisplayValue(setting);\r\n const isLegallyProtected = getLegallyProtected(setting);\r\n\r\n return (\r\n <div\r\n key={editKey}\r\n className={`flex items-center justify-between py-4 border-b border-[var(--border-color)] last:border-0 ${\r\n isLegallyProtected ? 'bg-[var(--warning-bg)] -mx-6 px-6' : ''\r\n }`}\r\n >\r\n <div className=\"flex-1 min-w-0 pr-4\">\r\n <div className=\"flex items-center gap-2\">\r\n <span className=\"font-medium text-[var(--text-primary)]\">\r\n {t(`settings.keys.${setting.category}.${setting.key}`, setting.key)}\r\n </span>\r\n {isLegallyProtected && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--warning-bg)] text-[var(--warning-text)] border border-[var(--warning-border)] rounded flex items-center gap-1\">\r\n <Shield className=\"w-3 h-3\" />\r\n {t('settings.legallyProtected')}\r\n </span>\r\n )}\r\n {!setting.isEditable && !isLegallyProtected && (\r\n <span className=\"px-2 py-0.5 text-xs bg-[var(--bg-secondary)] text-[var(--text-tertiary)] rounded\">\r\n {t('settings.readonly')}\r\n </span>\r\n )}\r\n <span className={`px-2 py-0.5 text-xs rounded ${getSourceBadgeClass(setting.source)}`}>\r\n {setting.source}\r\n </span>\r\n </div>\r\n {setting.description && (\r\n <p className=\"text-sm text-[var(--text-secondary)] mt-1\">\r\n {t(`settings.descriptions.${setting.category}.${setting.key}`, setting.description)}\r\n </p>\r\n )}\r\n {setting.modifiedAt && (\r\n <p className=\"text-xs text-[var(--text-tertiary)] mt-1 flex items-center gap-1\">\r\n <Clock className=\"w-3 h-3\" />\r\n {t('settings.modifiedBy', { user: setting.modifiedByUserName, date: new Date(setting.modifiedAt).toLocaleString() })}\r\n </p>\r\n )}\r\n </div>\r\n\r\n <div className=\"flex items-center gap-2\">\r\n {renderSettingInput(setting, displayValue, isEditing, handleValueChange)}\r\n\r\n {setting.isEditable && setting.source === 'Database' && (\r\n <>\r\n <button\r\n onClick={() => handleSave(setting)}\r\n disabled={!changed || isEditing}\r\n className=\"p-2 text-[var(--color-accent-600)] hover:bg-[var(--color-accent-500)]/10 rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('common.save')}\r\n >\r\n {isEditing ? <Loader2 className=\"w-4 h-4 animate-spin\" /> : <Save className=\"w-4 h-4\" />}\r\n </button>\r\n <button\r\n onClick={() => handleReset(setting)}\r\n disabled={setting.value === setting.defaultValue || isEditing}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] disabled:opacity-50 disabled:cursor-not-allowed transition-colors\"\r\n title={t('settings.resetToDefault')}\r\n >\r\n <RotateCcw className=\"w-4 h-4\" />\r\n </button>\r\n <button\r\n onClick={() => openHistory(setting.category, setting.key)}\r\n className=\"p-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n title={t('settings.viewHistory')}\r\n >\r\n <History className=\"w-4 h-4\" />\r\n </button>\r\n </>\r\n )}\r\n </div>\r\n </div>\r\n );\r\n };\r\n\r\n return (\r\n <div className=\"space-y-6\">\r\n {/* Legal Warning Banner */}\r\n <div className=\"bg-[var(--warning-bg)] border border-[var(--warning-border)] rounded-[var(--radius-card)] p-4\">\r\n <div className=\"flex items-start gap-3\">\r\n <AlertTriangle className=\"w-5 h-5 text-[var(--warning-text)] flex-shrink-0 mt-0.5\" />\r\n <div>\r\n <h4 className=\"font-medium text-[var(--warning-text)]\">\r\n {t('settings.legalWarningTitle')}\r\n </h4>\r\n <p className=\"text-sm text-[var(--warning-text)] opacity-90 mt-1\">\r\n {t('settings.legalWarningDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div className=\"rounded-[var(--radius-card)] border border-[var(--border-color)] bg-[var(--bg-card)] overflow-hidden shadow-sm\">\r\n <div className=\"px-6 py-4 bg-gradient-to-r from-amber-500/10 to-amber-600/5 border-b border-[var(--border-color)]\">\r\n <div className=\"flex items-center gap-3\">\r\n <div className=\"p-2 rounded-lg bg-amber-500/20\">\r\n <FileText className=\"w-5 h-5 text-amber-600\" />\r\n </div>\r\n <div>\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)]\">\r\n {t('settings.sections.legalFile')}\r\n </h3>\r\n <p className=\"text-sm text-[var(--text-secondary)]\">\r\n {t('settings.sections.legalFileDescription')}\r\n </p>\r\n </div>\r\n </div>\r\n </div>\r\n <div className=\"px-6 py-2\">\r\n {legalFileSettings.length > 0 ? (\r\n legalFileSettings.map(renderSetting)\r\n ) : (\r\n <div className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noSettings')}\r\n </div>\r\n )}\r\n </div>\r\n </div>\r\n\r\n {/* History Modal */}\r\n {historyModal && (\r\n <div className=\"fixed inset-0 z-50 overflow-y-auto\">\r\n <div className=\"flex items-center justify-center min-h-screen px-4\">\r\n <button type=\"button\" className=\"fixed inset-0 bg-black/50\" onClick={() => setHistoryModal(null)} aria-label=\"Close modal\" />\r\n <div className=\"relative rounded-[var(--radius-card)] bg-[var(--bg-card)] shadow-xl max-w-2xl w-full p-6\">\r\n <h3 className=\"text-lg font-semibold text-[var(--text-primary)] mb-4\">\r\n {t('settings.historyTitle', { key: `${historyModal.category}.${historyModal.key}` })}\r\n </h3>\r\n\r\n {loadingHistory && (\r\n <div className=\"flex justify-center py-8\">\r\n <Loader2 className=\"w-8 h-8 animate-spin text-[var(--color-accent-600)]\" />\r\n </div>\r\n )}\r\n {!loadingHistory && history.length === 0 && (\r\n <p className=\"text-center py-8 text-[var(--text-secondary)]\">\r\n {t('settings.noHistory')}\r\n </p>\r\n )}\r\n {!loadingHistory && history.length > 0 && (\r\n <div className=\"space-y-3 max-h-96 overflow-y-auto\">\r\n {history.map((entry) => (\r\n <div key={entry.id} className=\"p-3 bg-[var(--bg-secondary)] rounded-lg\">\r\n <div className=\"flex justify-between text-sm\">\r\n <span className=\"text-[var(--text-secondary)]\">\r\n {entry.changedByUserName || t('settings.system')}\r\n </span>\r\n <span className=\"text-[var(--text-tertiary)]\">\r\n {new Date(entry.changedAt).toLocaleString()}\r\n </span>\r\n </div>\r\n <div className=\"mt-2 text-sm\">\r\n <span className=\"text-[var(--error-text)] line-through\">{entry.oldValue}</span>\r\n <span className=\"mx-2 text-[var(--text-tertiary)]\">→</span>\r\n <span className=\"text-[var(--success-text)]\">{entry.newValue}</span>\r\n </div>\r\n </div>\r\n ))}\r\n </div>\r\n )}\r\n\r\n <div className=\"mt-6 flex justify-end\">\r\n <button\r\n onClick={() => setHistoryModal(null)}\r\n className=\"px-4 py-2 text-[var(--text-secondary)] hover:bg-[var(--bg-secondary)] rounded-[var(--radius-button)] transition-colors\"\r\n >\r\n {t('common.cancel')}\r\n </button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n )}\r\n </div>\r\n );\r\n}\r\n\r\nexport default LegalFileSettingsSection;\r\n"],"names":["TABS","Settings","Upload","FileText","SettingsPage","useTranslation","navigate","useNavigate","location","useLocation","settings","setSettings","useState","loading","setLoading","loadSettings","useCallback","result","adminApi","err","useEffect","getCurrentTab","path","t","handleTabChange","tabId","tab","jsx","Loader2","jsxs","Breadcrumb","Icon","isActive","Outlet","useSettingsSection","refreshSettings","useOutletContext","editedValues","setEditedValues","saving","setSaving","historyModal","setHistoryModal","history","setHistory","loadingHistory","setLoadingHistory","handleValueChange","category","key","value","prev","hasChanges","setting","editKey","getDisplayValue","getEditKey","newValue","next","SettingsHistoryModal","settingKey","onClose","entry","GeneralSettingsSection","handleSave","handleReset","openHistory","closeHistory","sessionSettings","generalSettings","getSourceBadgeClass","source","getSettingInput","displayValue","isEditing","e","renderSetting","changed","Clock","Fragment","Save","RotateCcw","History","SettingsIcon","FileUploadSettingsSection","fileUploadSettings","s","parseExtensions","renderSettingInput","extensions","ext","LegalFileSettingsSection","legalFileSettings","daysToYears","days","years","remainingDays","getLegallyProtected","isLegallyProtected","Shield","AlertTriangle"],"mappings":"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"}
|
|
@@ -2,7 +2,7 @@ import { jsx as a, jsxs as r, Fragment as F } from "react/jsx-runtime";
|
|
|
2
2
|
import { useState as A, useEffect as G, useCallback as R } from "react";
|
|
3
3
|
import { useTranslation as Y } from "react-i18next";
|
|
4
4
|
import { useSearchParams as W } from "react-router-dom";
|
|
5
|
-
import { L as Z, M as _, O as z, Q, i as X } from "./index-
|
|
5
|
+
import { L as Z, M as _, O as z, Q, i as X } from "./index-lpIzhufD.js";
|
|
6
6
|
import { Loader2 as V, AlertTriangle as T, ShieldCheck as q, Key as H, Zap as J, Users as B, Building as ee, Calendar as ae, Server as te, Monitor as re, Layers as se, Shield as ne, CheckCircle as I, Infinity as ie, Clock as le, X as j, Unlock as ce, Lock as oe } from "lucide-react";
|
|
7
7
|
const de = {
|
|
8
8
|
blue: "bg-[var(--info-bg)] text-[var(--info-text)] border-[var(--info-border)]",
|
|
@@ -528,4 +528,4 @@ function E({ icon: e, label: t, max: s, current: n, margin: i = 0, t: l }) {
|
|
|
528
528
|
export {
|
|
529
529
|
Ce as LicenseManagementPage
|
|
530
530
|
};
|
|
531
|
-
//# sourceMappingURL=index-
|
|
531
|
+
//# sourceMappingURL=index-BmaJz475.js.map
|