@dashadmin/dash-admin 1.3.24 → 1.3.26

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (511) hide show
  1. package/dist/AppAsyncWrapper.js +30 -1
  2. package/dist/AppWrapper.js +35 -1
  3. package/dist/DASHAdmin.js +312 -1
  4. package/dist/RoutingWrapper.js +76 -1
  5. package/dist/classes/DASHStorageClass.js +15 -1
  6. package/dist/components/Input/RichTextField.js +26 -1
  7. package/dist/components/Redirect.js +56 -1
  8. package/dist/components/avatar/Avatar.js +72 -1
  9. package/dist/components/avatar/SingleImageUploader.js +46 -1
  10. package/dist/components/currency/Format.js +71 -1
  11. package/dist/components/custom/Redirect.js +20 -1
  12. package/dist/components/error/Error.js +161 -1
  13. package/dist/components/geocoding/GeocodingField.js +133 -1
  14. package/dist/components/i18n/BridgedLocalesMenuButton.js +126 -1
  15. package/dist/components/i18n/LangSwitcher.js +196 -1
  16. package/dist/components/loader/GlobalLoader.js +71 -1
  17. package/dist/components/logs/LogFile.js +99 -1
  18. package/dist/components/logs/LogFileById.js +75 -1
  19. package/dist/components/logs/LogViewer.js +33 -1
  20. package/dist/components/menu/AppMenu.js +11 -1
  21. package/dist/components/menu/CustomMenuItemLink.js +77 -1
  22. package/dist/components/menu/DarkToggleMode.js +47 -1
  23. package/dist/components/menu/LanguageToggleMode.js +99 -1
  24. package/dist/components/misc/CustomImageInput.js +54 -1
  25. package/dist/components/misc/DASHGlobalErrorHandler.js +159 -1
  26. package/dist/components/misc/DarkModeSwitcher.js +40 -1
  27. package/dist/components/misc/LanguageSwitcher.js +193 -1
  28. package/dist/components/misc/MUIHtmlToolTip.js +50 -1
  29. package/dist/components/misc/MuiSimpleJsonTable.js +178 -1
  30. package/dist/components/misc/MultiLevelTable.js +124 -1
  31. package/dist/components/misc/NoResults.js +17 -1
  32. package/dist/components/misc/PDFViewer.js +86 -1
  33. package/dist/components/navigation/Breadcrumbs.js +178 -1
  34. package/dist/components/navigation/BreadcrumbsManager.js +194 -1
  35. package/dist/components/navigation/index.js +8 -1
  36. package/dist/components/panel/Error.js +161 -1
  37. package/dist/components/permission/AvailablePermissionsContext.js +157 -1
  38. package/dist/components/permission/PermissionsSelector.js +719 -1
  39. package/dist/components/permission/PermissionsSelectorList.js +400 -1
  40. package/dist/components/racustombuttons/CustomRAButton.js +94 -1
  41. package/dist/components/racustombuttons/QuickButton.js +109 -1
  42. package/dist/components/racustombuttons/QuickEditButton.js +114 -1
  43. package/dist/components/racustombuttons/QuickIconButton.js +119 -1
  44. package/dist/components/scrollbar/Scrollbar.js +52 -1
  45. package/dist/components/selects/MultiSelect.js +42 -1
  46. package/dist/components/selects/RASearchableSelect.js +207 -1
  47. package/dist/components/selects/RASearchableSelectChips.js +225 -1
  48. package/dist/components/selects/SearchableSelect.js +128 -1
  49. package/dist/components/subscription/BillingCycleSelectInput.js +50 -1
  50. package/dist/components/subscription/PlanAddonsSettings.js +396 -1
  51. package/dist/components/subscription/PlanAuditLog.js +345 -1
  52. package/dist/components/subscription/PlanFeaturesSettings.js +151 -1
  53. package/dist/components/subscription/PlanLimitsSettings.js +168 -1
  54. package/dist/components/subscription/PlanMetadataSettings.js +219 -1
  55. package/dist/components/subscription/PlanPricesSettings.js +222 -1
  56. package/dist/components/subscription/index.js +12 -1
  57. package/dist/components/svgelements/SvgLoading.js +330 -1
  58. package/dist/components/svgelements/SvgLogo.js +70 -1
  59. package/dist/components/svgelements/SvgLogoMin.js +76 -1
  60. package/dist/components/svgelements/SvgShipping.js +1001 -1
  61. package/dist/components/tenant/TenantAttributes.js +107 -1
  62. package/dist/components/tenant/TenantAttributesContext.js +150 -1
  63. package/dist/components/tenant/TenantSelector.js +160 -1
  64. package/dist/components/tenant/TenantSettings.js +107 -1
  65. package/dist/components/tenant/TenantSettingsContext.js +152 -1
  66. package/dist/components/tenant/TenantTheme.js +112 -1
  67. package/dist/components/user/AvatarComponent.js +270 -1
  68. package/dist/components/user/TenantAvatarComponent.js +83 -1
  69. package/dist/components/user/UserPreferences.js +422 -1
  70. package/dist/config/ACTIONS.js +25 -1
  71. package/dist/config/ConstantsService.js +25 -1
  72. package/dist/config/DASHAdminSystemConstants.js +137 -1
  73. package/dist/config/PARSERS.js +9 -1
  74. package/dist/contexts/DashQueryClientContext.js +78 -1
  75. package/dist/contexts/DashResourceContext.js +21 -1
  76. package/dist/contexts/I18nBridgeContext.js +123 -1
  77. package/dist/contexts/I18nBridgeSetter.js +29 -1
  78. package/dist/contexts/I18nReduxSync.js +16 -1
  79. package/dist/contexts/SubscriptionPlanFormatsProvider.js +175 -1
  80. package/dist/contexts/SystemRequestsCache.js +194 -1
  81. package/dist/contexts/auth/AuthContext.js +396 -1
  82. package/dist/contexts/auth/AuthContextLocalStorage.js +118 -1
  83. package/dist/contexts/auth/DASHAuthenticationService.js +679 -1
  84. package/dist/contexts/auth/WindowContext.js +10 -1
  85. package/dist/contexts/auth/index.js +2 -1
  86. package/dist/contexts/com/FCMContext.js +378 -1
  87. package/dist/contexts/com/LaravelEchoContext.js +45 -1
  88. package/dist/contexts/com/LaravelEchoMgr.js +121 -1
  89. package/dist/contexts/com/components/DefaultNotificationComponent.js +31 -1
  90. package/dist/contexts/com/components/NotificationAttributesTable.js +28 -1
  91. package/dist/contexts/com/components/NotificationRenderer.js +36 -1
  92. package/dist/contexts/com/components/NotificationsWidget.js +142 -1
  93. package/dist/contexts/com/components/notificationFormats.js +15 -1
  94. package/dist/contexts/com/useLaravelEcho.js +340 -1
  95. package/dist/contexts/dictionary/DictionaryContext.js +43 -1
  96. package/dist/default-theme/DASHAppProviders.js +86 -1
  97. package/dist/default-theme/DashThemeContext.js +146 -1
  98. package/dist/default-theme/DashThemeHelperProvider.js +54 -1
  99. package/dist/default-theme/DomainAppLayout.js +126 -1
  100. package/dist/default-theme/DomainHeader.js +114 -1
  101. package/dist/default-theme/DomainTheme.js +60 -1
  102. package/dist/default-theme/FullLayoutMarkup.js +55 -1
  103. package/dist/default-theme/StaticLayout.js +36 -1
  104. package/dist/default-theme/index.js +19 -1
  105. package/dist/default-theme/menu/AppMaterialMenu.js +364 -1
  106. package/dist/default-theme/menu/AppMenuComponents/SubmenuPortal.js +33 -1
  107. package/dist/default-theme/menu/AppMenuComponents/collapsed/CollapsedSidebarItem.js +239 -1
  108. package/dist/default-theme/menu/AppMenuComponents/collapsed/CollapsedSidebarItems.js +22 -1
  109. package/dist/default-theme/menu/AppMenuComponents/expanded/CollapsableSidebarMenu.js +209 -1
  110. package/dist/default-theme/menu/AppMenuComponents/expanded/SidebarItem.js +71 -1
  111. package/dist/default-theme/menu/AppMenuComponents/getItem.js +7 -1
  112. package/dist/default-theme/menu/AppMenuComponents/submenuConstants.js +78 -1
  113. package/dist/default-theme/menu/AppSidebarMaterial.js +232 -1
  114. package/dist/default-theme/menu/SidebarActions.js +75 -1
  115. package/dist/default-theme/updateDomCssVariables.js +134 -1
  116. package/dist/helpers/checkRole.js +14 -1
  117. package/dist/helpers/parseAxiosError.js +28 -2
  118. package/dist/hooks/audio/useAudio.js +24 -1
  119. package/dist/hooks/axios.js +52 -1
  120. package/dist/hooks/data/useDashboardStats.js +88 -1
  121. package/dist/hooks/data/usePackageByClientsStats.js +51 -1
  122. package/dist/hooks/data/usePackageByCommunesStats.js +51 -1
  123. package/dist/hooks/data/usePackageStats.js +88 -1
  124. package/dist/hooks/data/usePackageWithdrawDetails.js +69 -1
  125. package/dist/hooks/data/useStats.js +94 -1
  126. package/dist/hooks/isCurrentPath.js +18 -1
  127. package/dist/hooks/notifications/WSMessagesManager.js +128 -1
  128. package/dist/hooks/notifications/WSPusherManager.js +70 -1
  129. package/dist/hooks/useBreadcrumbs.js +208 -1
  130. package/dist/hooks/useClickOutside.js +18 -1
  131. package/dist/hooks/useFormPersistance.js +129 -1
  132. package/dist/hooks/useGlobalErrorMediator.js +37 -1
  133. package/dist/hooks/useGlobalLoaderMgr.js +20 -1
  134. package/dist/hooks/useHash.js +24 -1
  135. package/dist/hooks/useLocalStorage.js +68 -1
  136. package/dist/hooks/usePolyglotTranslation.js +57 -1
  137. package/dist/hooks/usePrevious.js +12 -1
  138. package/dist/hooks/usePrintSelected.js +30 -1
  139. package/dist/hooks/useVirtualHash.js +28 -1
  140. package/dist/hooks/window/useWindowSize.js +23 -1
  141. package/dist/index.js +120 -97
  142. package/dist/interfaces.js +6 -1
  143. package/dist/layout/AppLayout.js +36 -1
  144. package/dist/layout/ApplicationLayout.js +110 -1
  145. package/dist/layout/MotionWrapper.js +182 -1
  146. package/dist/layout/MotionWrapper.original.js +182 -1
  147. package/dist/layout/ResourceMenu.js +87 -1
  148. package/dist/layout/ResoureLayout.js +47 -1
  149. package/dist/layout/TransitionWrapper.js +83 -2
  150. package/dist/pages/ChangePassword.js +198 -1
  151. package/dist/pages/DASHLanding.js +9 -1
  152. package/dist/pages/DASHLightWeightLogin.js +495 -1
  153. package/dist/pages/DASHSimpleLogin.js +231 -1
  154. package/dist/pages/Login.js +113 -1
  155. package/dist/pages/Profile.js +347 -1
  156. package/dist/pages/RecoverPassword.js +175 -1
  157. package/dist/pages/SingleImageUploader.js +40 -1
  158. package/dist/pages/VerifyAccount.js +129 -1
  159. package/dist/providers/authProvider.js +186 -1
  160. package/dist/providers/dataProvider.js +468 -1
  161. package/dist/providers/i18n/en.js +514 -1
  162. package/dist/providers/i18n/es.js +520 -1
  163. package/dist/providers/i18n/languages.js +8 -1
  164. package/dist/providers/i18nProvider.js +23 -1
  165. package/dist/react-admin-dash/RADashComponent.js +210 -1
  166. package/dist/react-admin-dash/Resource.js +62 -1
  167. package/dist/resources/Brand/BrandResource.js +106 -1
  168. package/dist/resources/DASHResourceLoader.js +104 -1
  169. package/dist/resources/Log/LogResource.js +59 -1
  170. package/dist/resources/Tenant/ImpersonateTenantResource.js +113 -1
  171. package/dist/resources/Trash/TrashTemplate.js +195 -1
  172. package/dist/resources.js +259 -1
  173. package/dist/schemas/log.js +45 -1
  174. package/dist/schemas/notification.js +39 -1
  175. package/dist/schemas/permissions.js +44 -1
  176. package/dist/schemas/roles.js +48 -1
  177. package/dist/schemas/rolesDataGrid.js +48 -1
  178. package/dist/schemas/subscription/components/SubscriptionActions.js +172 -1
  179. package/dist/schemas/subscription/components/SubscriptionPaymentHistory.js +101 -1
  180. package/dist/schemas/subscription/components/SubscriptionPlanFeatures.js +115 -1
  181. package/dist/schemas/subscription/components/SubscriptionPlanPreview.js +92 -1
  182. package/dist/schemas/subscription/components/SubscriptionPlanStats.js +181 -1
  183. package/dist/schemas/subscription/components/SubscriptionStatusIndicator.js +70 -1
  184. package/dist/schemas/subscription/subscriptionPlanSchema.js +184 -1
  185. package/dist/schemas/subscription/subscriptionSchema.js +147 -1
  186. package/dist/schemas/subscriptionPlan.js +397 -1
  187. package/dist/schemas/tenant.js +122 -1
  188. package/dist/schemas/tenantUser.js +94 -1
  189. package/dist/schemas/tenant_superadmin.js +118 -1
  190. package/dist/schemas/user.js +26 -1
  191. package/dist/systemResources.js +1039 -1
  192. package/dist/systemResources.old.js +668 -1
  193. package/dist/templates/ResourceTemplate.js +196 -1
  194. package/dist/templates/ResourceTemplateCreate.js +144 -1
  195. package/dist/templates/ResourceTemplateEdit.js +147 -1
  196. package/dist/templates/ResourceTemplateFull.js +518 -1
  197. package/dist/templates/ResourceTemplateList.js +295 -1
  198. package/dist/templates/ResourceTemplateShow.js +45 -1
  199. package/dist/templates/TrashTemplate.js +175 -1
  200. package/dist/tenantResources.js +206 -1
  201. package/dist/test_portal_bubbles.js +37 -1
  202. package/dist/theme/AppHeader.js +60 -1
  203. package/dist/theme/AppLayoutSetting.js +34 -1
  204. package/dist/theme/AppLogo.js +7 -1
  205. package/dist/theme/AppSidebar.js +30 -1
  206. package/dist/theme/AppSidebarContent.js +84 -1
  207. package/dist/theme/AppSidebarLogo.js +42 -1
  208. package/dist/theme/components/PageTitle.js +61 -1
  209. package/dist/utils/cache/CacheInvalidatorContext.js +100 -1
  210. package/dist/utils/cache/CacheInvalidatorListenerComponent.js +10 -1
  211. package/dist/utils/cache/useCacheInvalidatorListener.js +17 -1
  212. package/dist/utils/convertFileToBase64.js +10 -1
  213. package/dist/utils/convertToFile.js +12 -1
  214. package/dist/utils/cookies.js +45 -1
  215. package/dist/utils/dashDefaultQueryClient.js +15 -1
  216. package/dist/utils/dataUrlToBlob.js +23 -1
  217. package/dist/utils/deepReplace.js +17 -1
  218. package/dist/utils/getProfileMenu.js +31 -1
  219. package/dist/utils/getTenantSettings.js +9 -1
  220. package/dist/utils/getType.js +10 -1
  221. package/dist/utils/getUserSettings.js +9 -1
  222. package/dist/utils/hasCode.js +14 -1
  223. package/dist/utils/index.js +19 -1
  224. package/dist/utils/injectTenantStyles.js +19 -1
  225. package/dist/utils/isComponent.js +23 -1
  226. package/dist/utils/navEvents.js +75 -1
  227. package/dist/utils/prototypes.js +5 -1
  228. package/dist/utils/regexp.js +7 -1
  229. package/dist/utils/resolveObjectPath.js +5 -1
  230. package/dist/utils/setNativeValue.js +17 -1
  231. package/dist/utils/slugify.js +7 -1
  232. package/dist/utils/validators/emailValidator.js +8 -1
  233. package/dist/utils/validators/requiredValidator.js +7 -1
  234. package/dist/utils/validators/rutValidator.js +25 -1
  235. package/dist/utils/validators.js +57 -1
  236. package/package.json +153 -69
  237. package/src/AppAsyncWrapper.tsx +42 -0
  238. package/src/AppWrapper.tsx +45 -0
  239. package/src/DASHAdmin.tsx +501 -0
  240. package/src/RoutingWrapper.tsx +139 -0
  241. package/src/classes/DASHStorageClass.tsx +8 -0
  242. package/src/components/Input/RichTextField.tsx +27 -0
  243. package/src/components/Redirect.tsx +46 -0
  244. package/src/components/avatar/Avatar.tsx +67 -0
  245. package/src/components/avatar/SingleImageUploader.tsx +53 -0
  246. package/src/components/currency/Format.tsx +77 -0
  247. package/src/components/custom/Redirect.tsx +23 -0
  248. package/src/components/error/Error.tsx +168 -0
  249. package/src/components/geocoding/GeocodingField.tsx +167 -0
  250. package/src/components/i18n/BridgedLocalesMenuButton.tsx +159 -0
  251. package/src/components/i18n/LangSwitcher.tsx +204 -0
  252. package/src/components/loader/GlobalLoader.tsx +56 -0
  253. package/src/components/logs/LogFile.tsx +88 -0
  254. package/src/components/logs/LogFileById.tsx +68 -0
  255. package/src/components/logs/LogViewer.tsx +43 -0
  256. package/src/components/menu/AppMenu.tsx +22 -0
  257. package/src/components/menu/CustomMenuItemLink.tsx +61 -0
  258. package/src/components/menu/DarkToggleMode.tsx +65 -0
  259. package/src/components/menu/LanguageToggleMode.tsx +120 -0
  260. package/src/components/misc/CustomImageInput.tsx +62 -0
  261. package/src/components/misc/DASHGlobalErrorHandler.tsx +194 -0
  262. package/src/components/misc/DarkModeSwitcher.tsx +112 -0
  263. package/src/components/misc/LanguageSwitcher.tsx +199 -0
  264. package/src/components/misc/MUIHtmlToolTip.tsx +15 -0
  265. package/src/components/misc/MuiSimpleJsonTable.tsx +279 -0
  266. package/src/components/misc/MultiLevelTable.tsx +229 -0
  267. package/src/components/misc/NoResults.tsx +25 -0
  268. package/src/components/misc/PDFViewer.tsx +85 -0
  269. package/src/components/navigation/Breadcrumbs.tsx +220 -0
  270. package/src/components/navigation/BreadcrumbsManager.tsx +310 -0
  271. package/src/components/navigation/index.tsx +5 -0
  272. package/src/components/panel/Error.tsx +168 -0
  273. package/src/components/permission/AvailablePermissionsContext.tsx +173 -0
  274. package/src/components/permission/PermissionsSelector.tsx +819 -0
  275. package/src/components/permission/PermissionsSelectorList.tsx +468 -0
  276. package/src/components/permission/tsx +0 -0
  277. package/src/components/racustombuttons/CustomRAButton.tsx +87 -0
  278. package/src/components/racustombuttons/QuickButton.tsx +117 -0
  279. package/src/components/racustombuttons/QuickEditButton.tsx +115 -0
  280. package/src/components/racustombuttons/QuickIconButton.tsx +122 -0
  281. package/src/components/scrollbar/Scrollbar.tsx +31 -0
  282. package/src/components/selects/MultiSelect.tsx +53 -0
  283. package/src/components/selects/RASearchableSelect.tsx +279 -0
  284. package/src/components/selects/RASearchableSelectChips.tsx +304 -0
  285. package/src/components/selects/SearchableSelect.tsx +135 -0
  286. package/src/components/subscription/BillingCycleSelectInput.tsx +68 -0
  287. package/src/components/subscription/PlanAddonsSettings.tsx +522 -0
  288. package/src/components/subscription/PlanAuditLog.tsx +439 -0
  289. package/src/components/subscription/PlanFeaturesSettings.tsx +223 -0
  290. package/src/components/subscription/PlanLimitsSettings.tsx +230 -0
  291. package/src/components/subscription/PlanMetadataSettings.tsx +284 -0
  292. package/src/components/subscription/PlanPricesSettings.tsx +295 -0
  293. package/src/components/subscription/index.ts +7 -0
  294. package/src/components/svgelements/SvgLoading.tsx +229 -0
  295. package/src/components/svgelements/SvgLogo.tsx +53 -0
  296. package/src/components/svgelements/SvgLogoMin.tsx +64 -0
  297. package/src/components/svgelements/SvgShipping.tsx +673 -0
  298. package/src/components/tenant/AvailablePermissionsContext.tsx +0 -0
  299. package/src/components/tenant/TenantAttributes.tsx +153 -0
  300. package/src/components/tenant/TenantAttributesContext.tsx +163 -0
  301. package/src/components/tenant/TenantSelector.tsx +139 -0
  302. package/src/components/tenant/TenantSettings.tsx +172 -0
  303. package/src/components/tenant/TenantSettingsContext.tsx +165 -0
  304. package/src/components/tenant/TenantTheme.tsx +172 -0
  305. package/src/components/user/AvatarComponent.tsx +301 -0
  306. package/src/components/user/TenantAvatarComponent.tsx +99 -0
  307. package/src/components/user/UserPreferences.tsx +642 -0
  308. package/src/config/ACTIONS.tsx +26 -0
  309. package/src/config/ConstantsService.tsx +44 -0
  310. package/src/config/DASHAdminSystemConstants.tsx +182 -0
  311. package/src/config/PARSERS.tsx +9 -0
  312. package/src/contexts/DashQueryClientContext.tsx +113 -0
  313. package/src/contexts/DashResourceContext.tsx +34 -0
  314. package/src/contexts/I18nBridgeContext.tsx +156 -0
  315. package/src/contexts/I18nBridgeSetter.tsx +39 -0
  316. package/src/contexts/I18nReduxSync.tsx +27 -0
  317. package/src/contexts/SubscriptionPlanFormatsProvider.tsx +217 -0
  318. package/src/contexts/SystemRequestsCache.tsx +215 -0
  319. package/src/contexts/auth/AuthContext.tsx +553 -0
  320. package/src/contexts/auth/AuthContextLocalStorage.tsx +140 -0
  321. package/src/contexts/auth/DASHAuthenticationService.tsx +917 -0
  322. package/src/contexts/auth/WindowContext.tsx +15 -0
  323. package/src/contexts/auth/index.tsx +3 -0
  324. package/src/contexts/com/FCMContext.tsx +415 -0
  325. package/src/contexts/com/LaravelEchoContext.tsx +41 -0
  326. package/src/contexts/com/LaravelEchoMgr.tsx +155 -0
  327. package/src/contexts/com/components/DefaultNotificationComponent.tsx +20 -0
  328. package/src/contexts/com/components/NotificationAttributesTable.tsx +54 -0
  329. package/src/contexts/com/components/NotificationRenderer.tsx +42 -0
  330. package/src/contexts/com/components/NotificationsWidget.tsx +182 -0
  331. package/src/contexts/com/components/notificationFormats.tsx +50 -0
  332. package/src/contexts/com/useLaravelEcho.tsx +432 -0
  333. package/src/contexts/dictionary/DictionaryContext.tsx +94 -0
  334. package/src/default-theme/DASHAppProviders.tsx +117 -0
  335. package/src/default-theme/DashThemeContext.tsx +218 -0
  336. package/src/default-theme/DashThemeHelperProvider.tsx +98 -0
  337. package/src/default-theme/DomainAppLayout.tsx +166 -0
  338. package/src/default-theme/DomainHeader.tsx +162 -0
  339. package/src/default-theme/DomainTheme.tsx +59 -0
  340. package/src/default-theme/FullLayoutMarkup.tsx +250 -0
  341. package/src/default-theme/StaticLayout.tsx +47 -0
  342. package/src/default-theme/index.tsx +11 -0
  343. package/src/default-theme/menu/AppMaterialMenu.tsx +545 -0
  344. package/src/default-theme/menu/AppMenuComponents/SubmenuPortal.tsx +56 -0
  345. package/src/default-theme/menu/AppMenuComponents/collapsed/CollapsedSidebarItem.tsx +254 -0
  346. package/src/default-theme/menu/AppMenuComponents/collapsed/CollapsedSidebarItems.tsx +31 -0
  347. package/src/default-theme/menu/AppMenuComponents/expanded/CollapsableSidebarMenu.tsx +276 -0
  348. package/src/default-theme/menu/AppMenuComponents/expanded/SidebarItem.tsx +91 -0
  349. package/src/default-theme/menu/AppMenuComponents/getItem.tsx +17 -0
  350. package/src/default-theme/menu/AppMenuComponents/interfaces.tsx +41 -0
  351. package/src/default-theme/menu/AppMenuComponents/submenuConstants.ts +85 -0
  352. package/src/default-theme/menu/AppSidebarMaterial.tsx +339 -0
  353. package/src/default-theme/menu/SidebarActions.tsx +84 -0
  354. package/src/default-theme/updateDomCssVariables.tsx +176 -0
  355. package/src/helpers/checkRole.tsx +66 -0
  356. package/src/helpers/parseAxiosError.tsx +41 -0
  357. package/src/hooks/audio/useAudio.tsx +39 -0
  358. package/src/hooks/axios.tsx +62 -0
  359. package/src/hooks/data/useDashboardStats.tsx +65 -0
  360. package/src/hooks/data/usePackageByClientsStats.tsx +44 -0
  361. package/src/hooks/data/usePackageByCommunesStats.tsx +44 -0
  362. package/src/hooks/data/usePackageStats.tsx +69 -0
  363. package/src/hooks/data/usePackageWithdrawDetails.tsx +61 -0
  364. package/src/hooks/data/useStats.tsx +86 -0
  365. package/src/hooks/isCurrentPath.tsx +22 -0
  366. package/src/hooks/notifications/WSMessagesManager.tsx +152 -0
  367. package/src/hooks/notifications/WSPusherManager.tsx +92 -0
  368. package/src/hooks/useBreadcrumbs.tsx +329 -0
  369. package/src/hooks/useClickOutside.tsx +17 -0
  370. package/src/hooks/useFormPersistance.tsx +150 -0
  371. package/src/hooks/useGlobalErrorMediator.tsx +43 -0
  372. package/src/hooks/useGlobalLoaderMgr.tsx +21 -0
  373. package/src/hooks/useHash.tsx +27 -0
  374. package/src/hooks/useLocalStorage.tsx +196 -0
  375. package/src/hooks/usePolyglotTranslation.tsx +62 -0
  376. package/src/hooks/usePrevious.tsx +9 -0
  377. package/src/hooks/usePrintSelected.tsx +37 -0
  378. package/src/hooks/useVirtualHash.tsx +33 -0
  379. package/src/hooks/window/useWindowSize.tsx +36 -0
  380. package/src/index.tsx +110 -0
  381. package/src/interfaces/IAppResourceConfig.tsx +20 -0
  382. package/src/interfaces/Log.tsx +16 -0
  383. package/src/interfaces/Tenant.tsx +25 -0
  384. package/src/interfaces/communication/IActions.tsx +5 -0
  385. package/src/interfaces/communication/INotification.tsx +21 -0
  386. package/src/interfaces/communication/IPayload.tsx +3 -0
  387. package/src/interfaces/communication/IRequestAction.tsx +10 -0
  388. package/src/interfaces/communication/IRequestPayload.tsx +7 -0
  389. package/src/interfaces/communication/IResponseAction.tsx +7 -0
  390. package/src/interfaces/misc/IAny.tsx +3 -0
  391. package/src/interfaces/misc/IFlashMessage.tsx +10 -0
  392. package/src/interfaces/navigation/IAppMenu.ts +14 -0
  393. package/src/interfaces/navigation/ICollapsableSidebarMenu.ts +12 -0
  394. package/src/interfaces/navigation/IMenuItem.ts +16 -0
  395. package/src/interfaces/notifications/IModel.tsx +6 -0
  396. package/src/interfaces/notifications/IModelField.tsx +8 -0
  397. package/src/interfaces/notifications/IUserNotification.tsx +12 -0
  398. package/src/interfaces/notifications/IUserNotificationData.tsx +13 -0
  399. package/src/interfaces/notifications/IUserNotificationPayload.tsx +8 -0
  400. package/src/interfaces/user/IGetAuth.tsx +92 -0
  401. package/src/interfaces/user/ITenantSettings.tsx +3 -0
  402. package/src/interfaces/user/IUser.tsx +60 -0
  403. package/src/interfaces/user/IUserSettings.tsx +43 -0
  404. package/src/interfaces.tsx +6 -0
  405. package/src/layout/AppLayout.tsx +112 -0
  406. package/src/layout/ApplicationLayout.tsx +197 -0
  407. package/src/layout/MotionWrapper.original.tsx +236 -0
  408. package/src/layout/MotionWrapper.tsx +236 -0
  409. package/src/layout/ResourceMenu.tsx +164 -0
  410. package/src/layout/ResoureLayout.tsx +105 -0
  411. package/src/layout/TransitionWrapper.tsx +125 -0
  412. package/src/pages/ChangePassword.tsx +179 -0
  413. package/src/pages/DASHLanding.tsx +22 -0
  414. package/src/pages/DASHLightWeightLogin.tsx +610 -0
  415. package/src/pages/DASHSimpleLogin.tsx +260 -0
  416. package/src/pages/Login.tsx +111 -0
  417. package/src/pages/Profile.tsx +394 -0
  418. package/src/pages/RecoverPassword.tsx +166 -0
  419. package/src/pages/SingleImageUploader.tsx +55 -0
  420. package/src/pages/VerifyAccount.tsx +128 -0
  421. package/src/providers/authProvider.tsx +210 -0
  422. package/src/providers/dataProvider.tsx +672 -0
  423. package/src/providers/i18n/en.ts +533 -0
  424. package/src/providers/i18n/es.ts +539 -0
  425. package/src/providers/i18n/languages.tsx +5 -0
  426. package/src/providers/i18nProvider.tsx +23 -0
  427. package/src/react-admin-dash/RADashComponent.tsx +240 -0
  428. package/src/react-admin-dash/Resource.tsx +77 -0
  429. package/src/resources/Brand/BrandResource.tsx +130 -0
  430. package/src/resources/DASHResourceLoader.ts +180 -0
  431. package/src/resources/Log/LogResource.tsx +73 -0
  432. package/src/resources/Tenant/ImpersonateTenantResource.tsx +113 -0
  433. package/src/resources/Trash/TrashTemplate.tsx +180 -0
  434. package/src/resources.tsx +280 -0
  435. package/src/schemas/log.tsx +45 -0
  436. package/src/schemas/notification.tsx +42 -0
  437. package/src/schemas/permissions.tsx +44 -0
  438. package/src/schemas/roles.tsx +48 -0
  439. package/src/schemas/rolesDataGrid.tsx +48 -0
  440. package/src/schemas/subscription/components/SubscriptionActions.tsx +153 -0
  441. package/src/schemas/subscription/components/SubscriptionPaymentHistory.tsx +120 -0
  442. package/src/schemas/subscription/components/SubscriptionPlanFeatures.tsx +155 -0
  443. package/src/schemas/subscription/components/SubscriptionPlanPreview.tsx +128 -0
  444. package/src/schemas/subscription/components/SubscriptionPlanStats.tsx +268 -0
  445. package/src/schemas/subscription/components/SubscriptionStatusIndicator.tsx +72 -0
  446. package/src/schemas/subscription/subscriptionPlanSchema.tsx +185 -0
  447. package/src/schemas/subscription/subscriptionSchema.tsx +152 -0
  448. package/src/schemas/subscriptionPlan.ts +422 -0
  449. package/src/schemas/tenant.tsx +129 -0
  450. package/src/schemas/tenantUser.tsx +96 -0
  451. package/src/schemas/tenant_superadmin.tsx +119 -0
  452. package/src/schemas/user.tsx +26 -0
  453. package/src/systemResources.old.tsx +928 -0
  454. package/src/systemResources.tsx +1135 -0
  455. package/src/templates/ResourceTemplate.tsx +240 -0
  456. package/src/templates/ResourceTemplateCreate.tsx +167 -0
  457. package/src/templates/ResourceTemplateEdit.tsx +176 -0
  458. package/src/templates/ResourceTemplateFull.tsx +581 -0
  459. package/src/templates/ResourceTemplateList.tsx +340 -0
  460. package/src/templates/ResourceTemplateShow.tsx +88 -0
  461. package/src/templates/TrashTemplate.tsx +179 -0
  462. package/src/tenantResources.tsx +223 -0
  463. package/src/test_portal_bubbles.tsx +40 -0
  464. package/src/theme/AppHeader.tsx +52 -0
  465. package/src/theme/AppLayoutSetting.tsx +33 -0
  466. package/src/theme/AppLogo.tsx +13 -0
  467. package/src/theme/AppSidebar.tsx +12 -0
  468. package/src/theme/AppSidebarContent.tsx +105 -0
  469. package/src/theme/AppSidebarLogo.tsx +48 -0
  470. package/src/theme/components/PageTitle.tsx +37 -0
  471. package/src/utils/cache/CacheInvalidatorContext.tsx +125 -0
  472. package/src/utils/cache/CacheInvalidatorListenerComponent.tsx +10 -0
  473. package/src/utils/cache/useCacheInvalidatorListener.tsx +27 -0
  474. package/src/utils/convertFileToBase64.tsx +9 -0
  475. package/src/utils/convertToFile.tsx +14 -0
  476. package/src/utils/cookies.tsx +33 -0
  477. package/src/utils/dashDefaultQueryClient.tsx +34 -0
  478. package/src/utils/dataUrlToBlob.tsx +30 -0
  479. package/src/utils/deepReplace.tsx +19 -0
  480. package/src/utils/getProfileMenu.tsx +41 -0
  481. package/src/utils/getTenantSettings.tsx +9 -0
  482. package/src/utils/getType.tsx +8 -0
  483. package/src/utils/getUserSettings.tsx +8 -0
  484. package/src/utils/hasCode.tsx +14 -0
  485. package/src/utils/index.ts +19 -0
  486. package/src/utils/injectTenantStyles.ts +28 -0
  487. package/src/utils/isComponent.tsx +45 -0
  488. package/src/utils/navEvents.tsx +91 -0
  489. package/src/utils/prototypes.tsx +5 -0
  490. package/src/utils/regexp.tsx +4 -0
  491. package/src/utils/resolveObjectPath.tsx +4 -0
  492. package/src/utils/setNativeValue.tsx +24 -0
  493. package/src/utils/slugify.tsx +11 -0
  494. package/src/utils/validators/emailValidator.tsx +6 -0
  495. package/src/utils/validators/requiredValidator.tsx +5 -0
  496. package/src/utils/validators/rutValidator.tsx +31 -0
  497. package/src/utils/validators.tsx +96 -0
  498. package/dist/ChangePassword-DmWAQAX_.js +0 -152
  499. package/dist/Login-t41HMhVT.js +0 -101
  500. package/dist/Profile-By-gdPUd.js +0 -294
  501. package/dist/RecoverPassword-DzPdbkg2.js +0 -149
  502. package/dist/VerifyAccount-Bofaxe5w.js +0 -99
  503. package/dist/index-gfebuoyf.js +0 -21614
  504. /package/dist/{components/permission/tsx → declarations.d.js} +0 -0
  505. /package/{dist → src}/DASHAdmin.tsx.old +0 -0
  506. /package/{dist → src}/declarations.d.ts +0 -0
  507. /package/{dist → src}/default-theme/DashThemeContext.tsx.fast +0 -0
  508. /package/{dist → src}/default-theme/DashThemeContext.tsx.old +0 -0
  509. /package/{dist → src}/templates/ResourceTemplate.tsx.memoized_experiment +0 -0
  510. /package/{dist → src}/templates/ResourceTemplate.tsx.original +0 -0
  511. /package/{dist → src}/templates/ResourceTemplateOld.tsx.bkup +0 -0
@@ -0,0 +1,642 @@
1
+ import React, { useState, useEffect, useCallback, useMemo, useRef } from "react";
2
+ import {
3
+ CircularProgress,
4
+ Box,
5
+ Typography,
6
+ Switch,
7
+ FormControlLabel,
8
+ Card,
9
+ CardContent,
10
+ Alert,
11
+ Button,
12
+ Snackbar,
13
+ Tabs,
14
+ Tab,
15
+ TextField,
16
+ Slider,
17
+ InputAdornment,
18
+ } from "@mui/material";
19
+ import {
20
+ Save as SaveIcon,
21
+ Notifications as NotificationsIcon,
22
+ Settings as SettingsIcon,
23
+ } from "@mui/icons-material";
24
+ import { useAxios } from 'dash-axios-hook';
25
+ import { AuthPersistenceService } from 'dash-auth';
26
+
27
+ // ============================================================================
28
+ // TYPES AND INTERFACES
29
+ // ============================================================================
30
+
31
+ interface NotificationConfig {
32
+ id: string;
33
+ name: string;
34
+ className: string;
35
+ hasEmail: boolean;
36
+ hasPush: boolean;
37
+ hasSocket: boolean;
38
+ hasDatabase: boolean;
39
+ }
40
+
41
+ interface NotificationPreference {
42
+ id: string;
43
+ name: string;
44
+ email: boolean;
45
+ push: boolean;
46
+ }
47
+
48
+ interface PreferenceFormat {
49
+ id: string;
50
+ group: string;
51
+ tab: string;
52
+ attribute: string;
53
+ label: string;
54
+ visible: boolean;
55
+ required: boolean;
56
+ custom?: boolean;
57
+ type: 'boolean' | 'integer' | 'string' | 'textarea' | 'select' | 'custom';
58
+ component?: string;
59
+ editable: boolean;
60
+ rules?: string;
61
+ default_value: any;
62
+ description?: string;
63
+ min?: number;
64
+ max?: number;
65
+ maxLength?: number;
66
+ rows?: number;
67
+ options?: { value: any; label: string }[];
68
+ }
69
+
70
+ interface GroupConfig {
71
+ label: string;
72
+ icon: string;
73
+ order: number;
74
+ description?: string;
75
+ }
76
+
77
+ interface PreferencesResponse {
78
+ preferences: Record<string, any>;
79
+ preference_formats: PreferenceFormat[];
80
+ groups: Record<string, GroupConfig>;
81
+ }
82
+
83
+ // ============================================================================
84
+ // HELPER FUNCTIONS
85
+ // ============================================================================
86
+
87
+ const getGroupIcon = (icon: string) => {
88
+ switch (icon) {
89
+ case 'notifications':
90
+ return <NotificationsIcon />;
91
+ case 'settings':
92
+ default:
93
+ return <SettingsIcon />;
94
+ }
95
+ };
96
+
97
+ // ============================================================================
98
+ // MAIN COMPONENT
99
+ // ============================================================================
100
+
101
+ /**
102
+ * UserPreferences Component
103
+ *
104
+ * This component renders user preferences organized into groups (tabs).
105
+ * It supports multiple preference types:
106
+ * - boolean: Toggle switch
107
+ * - integer: Number input with slider
108
+ * - string: Text input
109
+ * - textarea: Multi-line text input
110
+ * - custom: Custom component (e.g., NotificationPreferences)
111
+ *
112
+ * Preferences are fetched from /api/system/user/preferences
113
+ * and saved via PUT to the same endpoint.
114
+ */
115
+ const UserPreferences: React.FC = () => {
116
+ // ========================================================================
117
+ // STATE
118
+ // ========================================================================
119
+ const [loading, setLoading] = useState(true);
120
+ const [saving, setSaving] = useState(false);
121
+ const [isDirty, setIsDirty] = useState(false);
122
+ const [activeTab, setActiveTab] = useState(0);
123
+
124
+ // Notification preferences (custom component)
125
+ const [notificationPreferences, setNotificationPreferences] = useState<NotificationPreference[]>([]);
126
+
127
+ // Generic preferences (non-notification)
128
+ const [genericPreferences, setGenericPreferences] = useState<Record<string, any>>({});
129
+
130
+ // Schema from backend
131
+ const [preferenceFormats, setPreferenceFormats] = useState<PreferenceFormat[]>([]);
132
+ const [groups, setGroups] = useState<Record<string, GroupConfig>>({});
133
+
134
+ const [snackbar, setSnackbar] = useState<{ open: boolean; message: string; severity: 'success' | 'error' }>({
135
+ open: false,
136
+ message: '',
137
+ severity: 'success',
138
+ });
139
+
140
+ const axios = useAxios();
141
+
142
+ // Get notification configs from systemValues (for custom notification component)
143
+ // Memoize to prevent infinite re-renders
144
+ const availableNotifications: NotificationConfig[] = useMemo(() => {
145
+ const systemValues = AuthPersistenceService.getSystemValues();
146
+ return systemValues?.user_notifications || [];
147
+ }, []);
148
+
149
+ // Use ref to access availableNotifications in callbacks without causing re-renders
150
+ const availableNotificationsRef = useRef(availableNotifications);
151
+ availableNotificationsRef.current = availableNotifications;
152
+
153
+ // ========================================================================
154
+ // COMPUTED VALUES
155
+ // ========================================================================
156
+
157
+ // Get sorted groups for tabs
158
+ const sortedGroups = Object.entries(groups)
159
+ .sort(([, a], [, b]) => (a.order || 0) - (b.order || 0))
160
+ .map(([key, config]) => ({ key, ...config }));
161
+
162
+ // Get preferences for a specific group
163
+ const getPreferencesForGroup = (groupKey: string): PreferenceFormat[] => {
164
+ return preferenceFormats.filter(pf => pf.group === groupKey && pf.visible);
165
+ };
166
+
167
+ // ========================================================================
168
+ // CALLBACKS
169
+ // ========================================================================
170
+
171
+ // Build notification preferences from user data
172
+ const buildNotificationPreferencesFromUser = useCallback((
173
+ userPrefs: any,
174
+ notifications: NotificationConfig[]
175
+ ): NotificationPreference[] => {
176
+ const notificationPrefs = userPrefs?.notifications || [];
177
+
178
+ return notifications.map((notif) => {
179
+ const existing = Array.isArray(notificationPrefs)
180
+ ? notificationPrefs.find((p: any) => p.id === notif.id)
181
+ : null;
182
+
183
+ return {
184
+ id: notif.id,
185
+ name: notif.name,
186
+ email: existing?.email !== undefined ? existing.email : notif.hasEmail,
187
+ push: existing?.push !== undefined ? existing.push : notif.hasPush,
188
+ };
189
+ });
190
+ }, []);
191
+
192
+ // Fetch preferences from API
193
+ const fetchPreferences = useCallback(async () => {
194
+ try {
195
+ console.log('🔄 UserPreferences - Fetching preferences from API...');
196
+ const response = await axios.get('/system/user/preferences');
197
+ const data: PreferencesResponse = response.data;
198
+
199
+ console.log('✅ UserPreferences - Fetched:', data);
200
+
201
+ // Set schema
202
+ setPreferenceFormats(data.preference_formats || []);
203
+ setGroups(data.groups || {});
204
+
205
+ // Set user preferences
206
+ const userPrefs = data.preferences || {};
207
+
208
+ // Build notification preferences (custom component)
209
+ // Use ref to get current availableNotifications without dependency
210
+ const builtNotifPrefs = buildNotificationPreferencesFromUser(userPrefs, availableNotificationsRef.current);
211
+ setNotificationPreferences(builtNotifPrefs);
212
+
213
+ // Build generic preferences
214
+ const genericPrefs: Record<string, any> = {};
215
+ (data.preference_formats || []).forEach(format => {
216
+ if (format.type !== 'custom') {
217
+ genericPrefs[format.id] = userPrefs[format.id] ?? format.default_value;
218
+ }
219
+ });
220
+ setGenericPreferences(genericPrefs);
221
+
222
+ setIsDirty(false);
223
+
224
+ } catch (error) {
225
+ console.error('❌ UserPreferences - Fetch error:', error);
226
+ setSnackbar({
227
+ open: true,
228
+ message: 'Failed to load preferences',
229
+ severity: 'error',
230
+ });
231
+ }
232
+ }, [buildNotificationPreferencesFromUser]);
233
+
234
+ // Initialize on mount only
235
+ useEffect(() => {
236
+ const init = async () => {
237
+ await fetchPreferences();
238
+ setLoading(false);
239
+ };
240
+ init();
241
+ // eslint-disable-next-line react-hooks/exhaustive-deps
242
+ }, []);
243
+
244
+ // ========================================================================
245
+ // HANDLERS
246
+ // ========================================================================
247
+
248
+ // Handle tab change
249
+ const handleTabChange = (_event: React.SyntheticEvent, newValue: number) => {
250
+ setActiveTab(newValue);
251
+ };
252
+
253
+ // Handle notification toggle
254
+ const handleNotificationToggle = (notificationId: string, type: 'email' | 'push', checked: boolean) => {
255
+ setNotificationPreferences(prev => prev.map(pref =>
256
+ pref.id === notificationId ? { ...pref, [type]: checked } : pref
257
+ ));
258
+ setIsDirty(true);
259
+ };
260
+
261
+ // Handle generic preference change
262
+ const handleGenericPreferenceChange = (preferenceId: string, value: any) => {
263
+ setGenericPreferences(prev => ({ ...prev, [preferenceId]: value }));
264
+ setIsDirty(true);
265
+ };
266
+
267
+ // Handle save
268
+ const handleSave = async () => {
269
+ setSaving(true);
270
+ try {
271
+ const preferencesToSave: Record<string, any> = {
272
+ notifications: notificationPreferences,
273
+ ...genericPreferences,
274
+ };
275
+
276
+ const response = await axios.put('/system/user/preferences', {
277
+ preferences: preferencesToSave,
278
+ });
279
+
280
+ console.log('✅ UserPreferences - Saved:', response.data);
281
+
282
+ setSnackbar({
283
+ open: true,
284
+ message: 'Preferences saved successfully!',
285
+ severity: 'success',
286
+ });
287
+ setIsDirty(false);
288
+
289
+ // Refresh to ensure UI is in sync
290
+ await fetchPreferences();
291
+
292
+ } catch (error: any) {
293
+ console.error('❌ UserPreferences - Save error:', error);
294
+ setSnackbar({
295
+ open: true,
296
+ message: error.response?.data?.message || 'Failed to save preferences',
297
+ severity: 'error',
298
+ });
299
+ } finally {
300
+ setSaving(false);
301
+ }
302
+ };
303
+
304
+ // Close snackbar
305
+ const handleCloseSnackbar = () => {
306
+ setSnackbar(prev => ({ ...prev, open: false }));
307
+ };
308
+
309
+ // ========================================================================
310
+ // RENDER HELPERS
311
+ // ========================================================================
312
+
313
+ // Get notification config
314
+ const getNotificationConfig = (id: string): NotificationConfig | undefined => {
315
+ return availableNotifications.find(n => n.id === id);
316
+ };
317
+
318
+ // Format notification name
319
+ const formatNotificationName = (name: string): string => {
320
+ return name.replace(/Notification$/, '').replace(/([A-Z])/g, ' $1').trim();
321
+ };
322
+
323
+ // Render a single generic preference field
324
+ const renderPreferenceField = (format: PreferenceFormat) => {
325
+ const value = genericPreferences[format.id];
326
+
327
+ switch (format.type) {
328
+ case 'boolean':
329
+ return (
330
+ <Card key={format.id} sx={{ mb: 2 }}>
331
+ <CardContent sx={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
332
+ <Box>
333
+ <Typography variant="subtitle1">{format.label}</Typography>
334
+ {format.description && (
335
+ <Typography variant="caption" color="text.secondary">
336
+ {format.description}
337
+ </Typography>
338
+ )}
339
+ </Box>
340
+ <Switch
341
+ checked={!!value}
342
+ onChange={(e) => handleGenericPreferenceChange(format.id, e.target.checked)}
343
+ disabled={saving || !format.editable}
344
+ />
345
+ </CardContent>
346
+ </Card>
347
+ );
348
+
349
+ case 'integer':
350
+ return (
351
+ <Card key={format.id} sx={{ mb: 2 }}>
352
+ <CardContent>
353
+ <Typography variant="subtitle1" gutterBottom>{format.label}</Typography>
354
+ {format.description && (
355
+ <Typography variant="caption" color="text.secondary" display="block" mb={2}>
356
+ {format.description}
357
+ </Typography>
358
+ )}
359
+ <Box sx={{ display: 'flex', alignItems: 'center', gap: 2 }}>
360
+ <Slider
361
+ value={value ?? format.default_value ?? 0}
362
+ onChange={(_, newValue) => handleGenericPreferenceChange(format.id, newValue)}
363
+ min={format.min ?? 0}
364
+ max={format.max ?? 100}
365
+ disabled={saving || !format.editable}
366
+ sx={{ flexGrow: 1 }}
367
+ valueLabelDisplay="auto"
368
+ />
369
+ <TextField
370
+ type="number"
371
+ value={value ?? format.default_value ?? 0}
372
+ onChange={(e) => handleGenericPreferenceChange(format.id, parseInt(e.target.value, 10))}
373
+ disabled={saving || !format.editable}
374
+ inputProps={{
375
+ min: format.min ?? 0,
376
+ max: format.max ?? 100,
377
+ }}
378
+ sx={{ width: 80 }}
379
+ size="small"
380
+ />
381
+ </Box>
382
+ </CardContent>
383
+ </Card>
384
+ );
385
+
386
+ case 'string':
387
+ return (
388
+ <Card key={format.id} sx={{ mb: 2 }}>
389
+ <CardContent>
390
+ <Typography variant="subtitle1" gutterBottom>{format.label}</Typography>
391
+ {format.description && (
392
+ <Typography variant="caption" color="text.secondary" display="block" mb={2}>
393
+ {format.description}
394
+ </Typography>
395
+ )}
396
+ <TextField
397
+ fullWidth
398
+ value={value ?? ''}
399
+ onChange={(e) => handleGenericPreferenceChange(format.id, e.target.value)}
400
+ disabled={saving || !format.editable}
401
+ inputProps={{ maxLength: format.maxLength }}
402
+ InputProps={format.maxLength ? {
403
+ endAdornment: (
404
+ <InputAdornment position="end">
405
+ {(value?.length || 0)}/{format.maxLength}
406
+ </InputAdornment>
407
+ ),
408
+ } : undefined}
409
+ />
410
+ </CardContent>
411
+ </Card>
412
+ );
413
+
414
+ case 'textarea':
415
+ return (
416
+ <Card key={format.id} sx={{ mb: 2 }}>
417
+ <CardContent>
418
+ <Typography variant="subtitle1" gutterBottom>{format.label}</Typography>
419
+ {format.description && (
420
+ <Typography variant="caption" color="text.secondary" display="block" mb={2}>
421
+ {format.description}
422
+ </Typography>
423
+ )}
424
+ <TextField
425
+ fullWidth
426
+ multiline
427
+ rows={format.rows ?? 3}
428
+ value={value ?? ''}
429
+ onChange={(e) => handleGenericPreferenceChange(format.id, e.target.value)}
430
+ disabled={saving || !format.editable}
431
+ inputProps={{ maxLength: format.maxLength }}
432
+ helperText={format.maxLength ? `${(value?.length || 0)}/${format.maxLength} characters` : undefined}
433
+ />
434
+ </CardContent>
435
+ </Card>
436
+ );
437
+
438
+ default:
439
+ return null;
440
+ }
441
+ };
442
+
443
+ // Render notification preferences (custom component)
444
+ const renderNotificationPreferences = () => {
445
+ if (availableNotifications.length === 0) {
446
+ return (
447
+ <Alert severity="info">
448
+ No notification types available.
449
+ </Alert>
450
+ );
451
+ }
452
+
453
+ return (
454
+ <>
455
+ {notificationPreferences.map((pref) => {
456
+ const config = getNotificationConfig(pref.id);
457
+
458
+ return (
459
+ <Card key={pref.id} sx={{ mb: 2 }}>
460
+ <CardContent sx={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center' }}>
461
+ <Box>
462
+ <Typography variant="subtitle1">
463
+ {formatNotificationName(pref.name)}
464
+ </Typography>
465
+ <Typography variant="caption" color="text.secondary">
466
+ {pref.name}
467
+ </Typography>
468
+ </Box>
469
+
470
+ <Box sx={{ display: 'flex', gap: 2 }}>
471
+ {config?.hasEmail && (
472
+ <FormControlLabel
473
+ control={
474
+ <Switch
475
+ checked={pref.email}
476
+ onChange={(e) => handleNotificationToggle(pref.id, 'email', e.target.checked)}
477
+ color="primary"
478
+ disabled={saving}
479
+ />
480
+ }
481
+ label="Email"
482
+ />
483
+ )}
484
+
485
+ {config?.hasPush && (
486
+ <FormControlLabel
487
+ control={
488
+ <Switch
489
+ checked={pref.push}
490
+ onChange={(e) => handleNotificationToggle(pref.id, 'push', e.target.checked)}
491
+ color="primary"
492
+ disabled={saving}
493
+ />
494
+ }
495
+ label="Push"
496
+ />
497
+ )}
498
+
499
+ {!config?.hasEmail && !config?.hasPush && (
500
+ <Typography variant="caption" color="text.secondary">
501
+ (No configurable channels)
502
+ </Typography>
503
+ )}
504
+ </Box>
505
+ </CardContent>
506
+ </Card>
507
+ );
508
+ })}
509
+ </>
510
+ );
511
+ };
512
+
513
+ // Render group content
514
+ const renderGroupContent = (groupKey: string) => {
515
+ const groupPreferences = getPreferencesForGroup(groupKey);
516
+ const groupConfig = groups[groupKey];
517
+
518
+ return (
519
+ <Box sx={{ pt: 2 }}>
520
+ {groupConfig?.description && (
521
+ <Typography variant="body2" color="text.secondary" sx={{ mb: 3 }}>
522
+ {groupConfig.description}
523
+ </Typography>
524
+ )}
525
+
526
+ {groupPreferences.map(format => {
527
+ if (format.type === 'custom' && format.component === 'NotificationPreferences') {
528
+ return (
529
+ <Box key={format.id}>
530
+ {renderNotificationPreferences()}
531
+ </Box>
532
+ );
533
+ }
534
+ return renderPreferenceField(format);
535
+ })}
536
+
537
+ {groupPreferences.length === 0 && (
538
+ <Alert severity="info">
539
+ No preferences available in this group.
540
+ </Alert>
541
+ )}
542
+ </Box>
543
+ );
544
+ };
545
+
546
+ // ========================================================================
547
+ // RENDER
548
+ // ========================================================================
549
+
550
+ if (loading) {
551
+ return (
552
+ <Box sx={{ display: 'flex', justifyContent: 'center', p: 4 }}>
553
+ <CircularProgress />
554
+ </Box>
555
+ );
556
+ }
557
+
558
+ return (
559
+ <Box sx={{ p: 2 }}>
560
+ {/* Header */}
561
+ <Box sx={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center', mb: 2 }}>
562
+ <Typography variant="h5">User Preferences</Typography>
563
+
564
+ </Box>
565
+
566
+ {isDirty && (
567
+ <Alert severity="info" sx={{ mb: 2 }}>
568
+ You have unsaved changes. Click "Save Preferences" to apply them.
569
+ </Alert>
570
+ )}
571
+
572
+ {/* Tabs for groups */}
573
+ {sortedGroups.length > 1 && (
574
+ <Tabs
575
+ value={activeTab}
576
+ onChange={handleTabChange}
577
+ sx={{ borderBottom: 1, borderColor: 'divider', mb: 2 }}
578
+ >
579
+ {sortedGroups.map((group, index) => (
580
+ <Tab
581
+ key={group.key}
582
+ label={group.label}
583
+ icon={getGroupIcon(group.icon)}
584
+ iconPosition="start"
585
+ id={`preference-tab-${index}`}
586
+ aria-controls={`preference-tabpanel-${index}`}
587
+ />
588
+ ))}
589
+ </Tabs>
590
+ )}
591
+
592
+ {/* Tab content */}
593
+ {sortedGroups.map((group, index) => (
594
+ <Box
595
+ key={group.key}
596
+ role="tabpanel"
597
+ hidden={activeTab !== index}
598
+ id={`preference-tabpanel-${index}`}
599
+ aria-labelledby={`preference-tab-${index}`}
600
+ >
601
+ {activeTab === index && renderGroupContent(group.key)}
602
+ </Box>
603
+ ))}
604
+
605
+ {/* Single group - no tabs */}
606
+ {sortedGroups.length === 1 && renderGroupContent(sortedGroups[0].key)}
607
+
608
+ {/* No groups */}
609
+ {sortedGroups.length === 0 && (
610
+ <Alert severity="warning">
611
+ No preference groups configured.
612
+ </Alert>
613
+ )}
614
+
615
+ {/* Snackbar */}
616
+ <Snackbar
617
+ open={snackbar.open}
618
+ autoHideDuration={4000}
619
+ onClose={handleCloseSnackbar}
620
+ anchorOrigin={{ vertical: 'bottom', horizontal: 'center' }}
621
+ >
622
+ <Alert onClose={handleCloseSnackbar} severity={snackbar.severity} sx={{ width: '100%' }}>
623
+ {snackbar.message}
624
+ </Alert>
625
+ </Snackbar>
626
+
627
+ <Button
628
+ variant="contained"
629
+ fullWidth
630
+ color="primary"
631
+ startIcon={saving ? <CircularProgress size={16} color="inherit" /> : <SaveIcon />}
632
+ onClick={handleSave}
633
+ disabled={saving || !isDirty}
634
+ >
635
+ {saving ? 'Saving...' : 'Save Preferences'}
636
+ </Button>
637
+
638
+ </Box>
639
+ );
640
+ };
641
+
642
+ export default UserPreferences;
@@ -0,0 +1,26 @@
1
+ import { IActions } from '../interfaces/communication/IActions';
2
+ import PARSERS from './PARSERS';
3
+
4
+ const ACTIONS: IActions = {
5
+ GET_NOTIFICATIONS: {
6
+ ACTION: 'GET_NOTIFICATIONS',
7
+ SUCCESS: 'GET_NOTIFICATIONS_SUCCESS',
8
+ FAILURE: 'GET_NOTIFICATIONS_FAILURE',
9
+ PATH: '/api/notification',
10
+ METHOD: 'POST',
11
+ AUTH: true,
12
+ PARSER: PARSERS.GET_NOTIFICATIONS,
13
+ },
14
+
15
+ READ_NOTIFICATION: {
16
+ ACTION: 'READ_NOTIFICATION',
17
+ SUCCESS: 'READ_NOTIFICATION_SUCCESS',
18
+ FAILURE: 'READ_NOTIFICATION_FAILURE',
19
+ PATH: '/api/notification/read/:id',
20
+ METHOD: 'GET',
21
+ AUTH: true,
22
+ // PARSER: PARSERS.GET_NOTIFICATIONS
23
+ },
24
+ };
25
+
26
+ export default ACTIONS;