@jmruthers/pace-core 0.2.7 → 0.5.1

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 (541) hide show
  1. package/dist/{DataTable-EEUDXPE5.js → DataTable-GX3XERFJ.js} +8 -4
  2. package/dist/{DataTable-C1AEm9Cx.d.ts → DataTable-ltTFXHS3.d.ts} +3 -1
  3. package/dist/{chunk-VYG4AXYW.js → chunk-5EL3KHOQ.js} +2 -2
  4. package/dist/{chunk-ETEJVKYK.js → chunk-6CR3MRZN.js} +1426 -62
  5. package/dist/chunk-6CR3MRZN.js.map +1 -0
  6. package/dist/chunk-AUE24LVR.js +268 -0
  7. package/dist/chunk-AUE24LVR.js.map +1 -0
  8. package/dist/chunk-COBPIXXQ.js +379 -0
  9. package/dist/chunk-COBPIXXQ.js.map +1 -0
  10. package/dist/{chunk-EWKPTNPO.js → chunk-GSNM5D6H.js} +388 -86
  11. package/dist/chunk-GSNM5D6H.js.map +1 -0
  12. package/dist/{chunk-2V3Y6YBC.js → chunk-OEGRKULD.js} +1 -42
  13. package/dist/chunk-OEGRKULD.js.map +1 -0
  14. package/dist/chunk-OYRY44Q2.js +62 -0
  15. package/dist/chunk-OYRY44Q2.js.map +1 -0
  16. package/dist/{chunk-RRUYHORU.js → chunk-T3XIA4AJ.js} +297 -433
  17. package/dist/chunk-T3XIA4AJ.js.map +1 -0
  18. package/dist/{chunk-HEMJ4SUJ.js → chunk-TGDCLPP2.js} +11 -7
  19. package/dist/{chunk-HEMJ4SUJ.js.map → chunk-TGDCLPP2.js.map} +1 -1
  20. package/dist/{chunk-HNDFPXUU.js → chunk-U6JDHVC2.js} +6 -4
  21. package/dist/{chunk-HNDFPXUU.js.map → chunk-U6JDHVC2.js.map} +1 -1
  22. package/dist/{chunk-TIVL4UQ7.js → chunk-XJK2J4N6.js} +6 -4
  23. package/dist/{chunk-TIVL4UQ7.js.map → chunk-XJK2J4N6.js.map} +1 -1
  24. package/dist/components.d.ts +2 -2
  25. package/dist/components.js +21 -20
  26. package/dist/components.js.map +1 -1
  27. package/dist/hooks.d.ts +1 -1
  28. package/dist/hooks.js +7 -7
  29. package/dist/index.d.ts +2 -2
  30. package/dist/index.js +26 -25
  31. package/dist/index.js.map +1 -1
  32. package/dist/providers.js +8 -7
  33. package/dist/rbac/index.d.ts +806 -806
  34. package/dist/rbac/index.js +937 -1179
  35. package/dist/rbac/index.js.map +1 -1
  36. package/dist/{types-DiRQsGJs.d.ts → types-BRDU7N6w.d.ts} +12 -1
  37. package/dist/utils.d.ts +2 -2
  38. package/dist/utils.js +6 -6
  39. package/docs/api/classes/ErrorBoundary.md +1 -1
  40. package/docs/api/classes/PublicErrorBoundary.md +1 -1
  41. package/docs/api/interfaces/AggregateConfig.md +4 -4
  42. package/docs/api/interfaces/ButtonProps.md +1 -1
  43. package/docs/api/interfaces/CardProps.md +1 -1
  44. package/docs/api/interfaces/ColorPalette.md +1 -1
  45. package/docs/api/interfaces/ColorShade.md +1 -1
  46. package/docs/api/interfaces/DataTableAction.md +21 -8
  47. package/docs/api/interfaces/DataTableColumn.md +1 -1
  48. package/docs/api/interfaces/DataTableProps.md +46 -33
  49. package/docs/api/interfaces/DataTableToolbarButton.md +7 -7
  50. package/docs/api/interfaces/EmptyStateConfig.md +5 -5
  51. package/docs/api/interfaces/EventContextType.md +1 -1
  52. package/docs/api/interfaces/EventLogoProps.md +1 -1
  53. package/docs/api/interfaces/EventProviderProps.md +1 -1
  54. package/docs/api/interfaces/FileSizeLimits.md +1 -1
  55. package/docs/api/interfaces/FileUploadProps.md +1 -1
  56. package/docs/api/interfaces/FooterProps.md +1 -1
  57. package/docs/api/interfaces/InactivityWarningModalProps.md +1 -1
  58. package/docs/api/interfaces/InputProps.md +1 -1
  59. package/docs/api/interfaces/LabelProps.md +1 -1
  60. package/docs/api/interfaces/LoginFormProps.md +1 -1
  61. package/docs/api/interfaces/NavigationItem.md +1 -1
  62. package/docs/api/interfaces/NavigationMenuProps.md +1 -1
  63. package/docs/api/interfaces/Organisation.md +1 -1
  64. package/docs/api/interfaces/OrganisationContextType.md +1 -1
  65. package/docs/api/interfaces/OrganisationMembership.md +2 -2
  66. package/docs/api/interfaces/OrganisationProviderProps.md +1 -1
  67. package/docs/api/interfaces/OrganisationSecurityError.md +1 -1
  68. package/docs/api/interfaces/PaceAppLayoutProps.md +1 -1
  69. package/docs/api/interfaces/PaceLoginPageProps.md +1 -1
  70. package/docs/api/interfaces/PaletteData.md +1 -1
  71. package/docs/api/interfaces/PublicErrorBoundaryProps.md +1 -1
  72. package/docs/api/interfaces/PublicErrorBoundaryState.md +1 -1
  73. package/docs/api/interfaces/PublicLoadingSpinnerProps.md +1 -1
  74. package/docs/api/interfaces/PublicPageFooterProps.md +1 -1
  75. package/docs/api/interfaces/PublicPageHeaderProps.md +1 -1
  76. package/docs/api/interfaces/PublicPageLayoutProps.md +1 -1
  77. package/docs/api/interfaces/StorageConfig.md +1 -1
  78. package/docs/api/interfaces/StorageFileInfo.md +1 -1
  79. package/docs/api/interfaces/StorageFileMetadata.md +1 -1
  80. package/docs/api/interfaces/StorageListOptions.md +1 -1
  81. package/docs/api/interfaces/StorageListResult.md +1 -1
  82. package/docs/api/interfaces/StorageUploadOptions.md +1 -1
  83. package/docs/api/interfaces/StorageUploadResult.md +1 -1
  84. package/docs/api/interfaces/StorageUrlOptions.md +1 -1
  85. package/docs/api/interfaces/StyleImport.md +1 -1
  86. package/docs/api/interfaces/ToastActionElement.md +1 -1
  87. package/docs/api/interfaces/ToastProps.md +1 -1
  88. package/docs/api/interfaces/UnifiedAuthContextType.md +1 -1
  89. package/docs/api/interfaces/UnifiedAuthProviderProps.md +1 -1
  90. package/docs/api/interfaces/UseInactivityTrackerOptions.md +1 -1
  91. package/docs/api/interfaces/UseInactivityTrackerReturn.md +1 -1
  92. package/docs/api/interfaces/UsePublicEventLogoOptions.md +1 -1
  93. package/docs/api/interfaces/UsePublicEventLogoReturn.md +1 -1
  94. package/docs/api/interfaces/UsePublicEventOptions.md +1 -1
  95. package/docs/api/interfaces/UsePublicEventReturn.md +1 -1
  96. package/docs/api/interfaces/UsePublicRouteParamsReturn.md +1 -1
  97. package/docs/api/interfaces/UserEventAccess.md +1 -1
  98. package/docs/api/interfaces/UserMenuProps.md +1 -1
  99. package/docs/api/interfaces/UserProfile.md +1 -1
  100. package/docs/api/modules.md +3 -3
  101. package/package.json +5 -2
  102. package/src/__tests__/REBUILD_PLAN.md +223 -0
  103. package/src/__tests__/TESTING_GUIDELINES.md +341 -0
  104. package/src/__tests__/fixtures/mocks.ts +93 -0
  105. package/src/__tests__/helpers/component-test-utils.tsx +145 -0
  106. package/src/__tests__/helpers/test-utils.tsx +117 -0
  107. package/src/__tests__/integration/UserProfile.test.tsx +128 -0
  108. package/src/__tests__/setup.ts +37 -225
  109. package/src/__tests__/templates/component.test.template.tsx +97 -75
  110. package/src/__tests__/templates/hook.test.template.ts +173 -0
  111. package/src/__tests__/types/test.types.ts +106 -0
  112. package/src/components/Alert/Alert.test.tsx +496 -0
  113. package/src/components/Avatar/Avatar.test.tsx +484 -0
  114. package/src/components/Button/Button.test.tsx +662 -0
  115. package/src/components/Card/Card.test.tsx +593 -0
  116. package/src/components/Checkbox/Checkbox.test.tsx +461 -0
  117. package/src/components/DataTable/DataTable.tsx +9 -1
  118. package/src/components/DataTable/components/AccessDeniedPage.tsx +168 -0
  119. package/src/components/DataTable/components/ActionButtons.tsx +18 -1
  120. package/src/components/DataTable/components/DataTableCore.tsx +97 -11
  121. package/src/components/DataTable/components/DataTableToolbar.tsx +22 -10
  122. package/src/components/DataTable/components/UnifiedTableBody.tsx +33 -4
  123. package/src/components/DataTable/examples/HierarchicalActionsExample.tsx +1 -0
  124. package/src/components/DataTable/examples/HierarchicalExample.tsx +3 -0
  125. package/src/components/DataTable/examples/InitialPageSizeExample.tsx +3 -0
  126. package/src/components/DataTable/examples/PerformanceExample.tsx +3 -0
  127. package/src/components/DataTable/types.ts +39 -1
  128. package/src/components/Dialog/Dialog.test.tsx +1139 -0
  129. package/src/components/ErrorBoundary/ErrorBoundary.test.tsx +752 -0
  130. package/src/components/FileUpload/FileUpload.test.tsx +665 -0
  131. package/src/hooks/useCounter.test.ts +135 -0
  132. package/src/rbac/index.ts +3 -3
  133. package/dist/chunk-2V3Y6YBC.js.map +0 -1
  134. package/dist/chunk-BEZRLNK3.js +0 -1744
  135. package/dist/chunk-BEZRLNK3.js.map +0 -1
  136. package/dist/chunk-ETEJVKYK.js.map +0 -1
  137. package/dist/chunk-EWKPTNPO.js.map +0 -1
  138. package/dist/chunk-OHXGNT3K.js +0 -21
  139. package/dist/chunk-OHXGNT3K.js.map +0 -1
  140. package/dist/chunk-RRUYHORU.js.map +0 -1
  141. package/src/__tests__/README.md +0 -404
  142. package/src/__tests__/debug-provider.unit.test.tsx +0 -67
  143. package/src/__tests__/e2e/workflows.test.tsx +0 -373
  144. package/src/__tests__/hybridPermissions.unit.test.tsx +0 -474
  145. package/src/__tests__/index.integration.test.ts +0 -491
  146. package/src/__tests__/mocks/MockAuthProvider-standalone.tsx +0 -47
  147. package/src/__tests__/mocks/MockAuthProvider.tsx +0 -63
  148. package/src/__tests__/mocks/enhancedSupabaseMock.ts +0 -252
  149. package/src/__tests__/mocks/index.test.ts +0 -23
  150. package/src/__tests__/mocks/index.ts +0 -16
  151. package/src/__tests__/mocks/mockAuth.ts +0 -155
  152. package/src/__tests__/mocks/mockSupabase.ts +0 -83
  153. package/src/__tests__/mocks/mockSupabaseClient.ts +0 -63
  154. package/src/__tests__/mocks/providers.tsx +0 -22
  155. package/src/__tests__/patterns/__tests__/testPatterns.test.ts +0 -394
  156. package/src/__tests__/patterns/testPatterns.ts +0 -124
  157. package/src/__tests__/performance/componentPerformance.performance.test.ts +0 -27
  158. package/src/__tests__/performance/index.ts +0 -24
  159. package/src/__tests__/performance/performanceValidation.performance.test.ts +0 -15
  160. package/src/__tests__/security/security.unit.test.tsx +0 -7
  161. package/src/__tests__/security/securityValidation.security.test.tsx +0 -153
  162. package/src/__tests__/setupTests.d.ts +0 -1
  163. package/src/__tests__/shared/componentTestUtils.tsx +0 -475
  164. package/src/__tests__/shared/errorHandlingTestUtils.ts +0 -107
  165. package/src/__tests__/shared/index.ts +0 -81
  166. package/src/__tests__/shared/integrationTestUtils.tsx +0 -375
  167. package/src/__tests__/shared/performanceTestUtils.tsx +0 -476
  168. package/src/__tests__/shared/testUtils.optimized.tsx +0 -685
  169. package/src/__tests__/simple.test.tsx +0 -20
  170. package/src/__tests__/test-utils/dataFactories.ts +0 -60
  171. package/src/__tests__/test-utils/index.ts +0 -6
  172. package/src/__tests__/typeSafety.unit.test.ts +0 -65
  173. package/src/__tests__/unifiedAuth.unit.test.tsx +0 -151
  174. package/src/__tests__/utils/accessibilityHelpers.ts +0 -254
  175. package/src/__tests__/utils/assertions.ts +0 -50
  176. package/src/__tests__/utils/deterministicHelpers.ts +0 -31
  177. package/src/__tests__/utils/edgeCaseConfig.test.ts +0 -75
  178. package/src/__tests__/utils/edgeCaseConfig.ts +0 -98
  179. package/src/__tests__/utils/mockHelpers.ts +0 -149
  180. package/src/__tests__/utils/mockLoader.ts +0 -101
  181. package/src/__tests__/utils/performanceHelpers.ts +0 -55
  182. package/src/__tests__/utils/performanceTestHelpers.ts +0 -68
  183. package/src/__tests__/utils/testDataFactories.ts +0 -28
  184. package/src/__tests__/utils/testIsolation.ts +0 -67
  185. package/src/__tests__/utils/visualTestHelpers.ts +0 -20
  186. package/src/__tests__/visual/__snapshots__/componentSnapshots.visual.test.tsx.snap +0 -68
  187. package/src/__tests__/visual/__snapshots__/componentVisuals.visual.test.tsx.snap +0 -14
  188. package/src/__tests__/visual/__snapshots__/visualRegression.test.tsx.snap +0 -217
  189. package/src/__tests__/visual/__snapshots__/visualRegression.visual.test.tsx.snap +0 -24
  190. package/src/__tests__/visual/componentSnapshots.visual.test.tsx +0 -33
  191. package/src/__tests__/visual/componentVisuals.visual.test.tsx +0 -12
  192. package/src/__tests__/visual/visualRegression.visual.test.tsx +0 -20
  193. package/src/components/Alert/__tests__/Alert.unit.test.tsx +0 -381
  194. package/src/components/Avatar/__tests__/Avatar.unit.test.tsx +0 -232
  195. package/src/components/Button/__tests__/Button.accessibility.test.tsx +0 -131
  196. package/src/components/Button/__tests__/Button.comprehensive.test.tsx +0 -721
  197. package/src/components/Button/__tests__/Button.unit.test.tsx +0 -189
  198. package/src/components/Button/__tests__/EventSelector.integration.test.tsx +0 -285
  199. package/src/components/Card/__tests__/Card.accessibility.test.tsx +0 -394
  200. package/src/components/Card/__tests__/Card.comprehensive.test.tsx +0 -599
  201. package/src/components/Card/__tests__/Card.integration.test.tsx +0 -673
  202. package/src/components/Card/__tests__/Card.performance.test.tsx +0 -546
  203. package/src/components/Card/__tests__/Card.unit.test.tsx +0 -330
  204. package/src/components/Card/__tests__/Card.visual.test.tsx +0 -599
  205. package/src/components/Card/__tests__/README.md +0 -211
  206. package/src/components/Checkbox/__tests__/Checkbox.unit.test.tsx +0 -520
  207. package/src/components/DataTable/__tests__/DataTable.errorHandling.test.tsx +0 -251
  208. package/src/components/DataTable/__tests__/DataTable.hierarchical.test.tsx +0 -680
  209. package/src/components/DataTable/__tests__/DataTable.infinite-loop.test.tsx +0 -323
  210. package/src/components/DataTable/__tests__/DataTable.integration.test.tsx +0 -716
  211. package/src/components/DataTable/__tests__/DataTable.performance.test.tsx +0 -589
  212. package/src/components/DataTable/__tests__/DataTable.permissions.test.tsx +0 -316
  213. package/src/components/DataTable/__tests__/DataTable.regressionFixes.test.tsx +0 -546
  214. package/src/components/DataTable/__tests__/DataTable.selection.controlled.test.tsx +0 -386
  215. package/src/components/DataTable/__tests__/DataTable.selection.test.tsx +0 -338
  216. package/src/components/DataTable/__tests__/DataTable.sorting.test.tsx +0 -321
  217. package/src/components/DataTable/__tests__/DataTable.userWorkflows.test.tsx +0 -320
  218. package/src/components/DataTable/__tests__/DataTable.workflowValidation.test.tsx +0 -583
  219. package/src/components/DataTable/__tests__/DataTable.workflows.test.tsx +0 -711
  220. package/src/components/DataTable/__tests__/performance-regression.test.tsx +0 -777
  221. package/src/components/DataTable/__tests__/performance.test.tsx +0 -365
  222. package/src/components/DataTable/components/__tests__/ActionButtons.unit.test.tsx +0 -150
  223. package/src/components/DataTable/components/__tests__/BulkOperationsDropdown.test.tsx +0 -224
  224. package/src/components/DataTable/components/__tests__/ColumnVisibilityDropdown.unit.test.tsx +0 -244
  225. package/src/components/DataTable/components/__tests__/DataTable.accessibility.test.tsx +0 -629
  226. package/src/components/DataTable/components/__tests__/DataTable.integration.test.tsx +0 -470
  227. package/src/components/DataTable/components/__tests__/DataTable.performance.test.tsx +0 -160
  228. package/src/components/DataTable/components/__tests__/DataTable.real.test.tsx +0 -251
  229. package/src/components/DataTable/components/__tests__/DataTable.security.test.tsx +0 -171
  230. package/src/components/DataTable/components/__tests__/DataTable.unit.test.tsx +0 -290
  231. package/src/components/DataTable/components/__tests__/DataTableBody.unit.test.tsx +0 -147
  232. package/src/components/DataTable/components/__tests__/DataTableErrorBoundary.unit.test.tsx +0 -182
  233. package/src/components/DataTable/components/__tests__/DataTableModals.unit.test.tsx +0 -123
  234. package/src/components/DataTable/components/__tests__/EditableRow.unit.test.tsx +0 -660
  235. package/src/components/DataTable/components/__tests__/EmptyState.unit.test.tsx +0 -256
  236. package/src/components/DataTable/components/__tests__/ExpandButton.test.tsx +0 -498
  237. package/src/components/DataTable/components/__tests__/FilterRow.unit.test.tsx +0 -112
  238. package/src/components/DataTable/components/__tests__/FilteringToggle.unit.test.tsx +0 -133
  239. package/src/components/DataTable/components/__tests__/GroupHeader.unit.test.tsx +0 -172
  240. package/src/components/DataTable/components/__tests__/GroupingDropdown.unit.test.tsx +0 -222
  241. package/src/components/DataTable/components/__tests__/ImportModal.unit.test.tsx +0 -780
  242. package/src/components/DataTable/components/__tests__/LoadingState.unit.test.tsx +0 -65
  243. package/src/components/DataTable/components/__tests__/PaginationControls.unit.test.tsx +0 -634
  244. package/src/components/DataTable/components/__tests__/StateComponents.unit.test.tsx +0 -48
  245. package/src/components/DataTable/components/__tests__/UnifiedTableBody.hierarchical.test.tsx +0 -541
  246. package/src/components/DataTable/components/__tests__/ViewRowModal.unit.test.tsx +0 -228
  247. package/src/components/DataTable/components/__tests__/VirtualizedDataTable.unit.test.tsx +0 -568
  248. package/src/components/DataTable/core/__tests__/ActionManager.unit.test.ts +0 -405
  249. package/src/components/DataTable/core/__tests__/ArchitectureIntegration.unit.test.tsx +0 -445
  250. package/src/components/DataTable/core/__tests__/ColumnFactory.unit.test.ts +0 -288
  251. package/src/components/DataTable/core/__tests__/ColumnManager.unit.test.ts +0 -623
  252. package/src/components/DataTable/core/__tests__/DataManager.unit.test.ts +0 -431
  253. package/src/components/DataTable/core/__tests__/DataTableContext.unit.test.tsx +0 -433
  254. package/src/components/DataTable/core/__tests__/LocalDataAdapter.unit.test.ts +0 -422
  255. package/src/components/DataTable/core/__tests__/PluginRegistry.unit.test.tsx +0 -207
  256. package/src/components/DataTable/core/__tests__/StateManager.unit.test.ts +0 -278
  257. package/src/components/DataTable/examples/__tests__/PerformanceExample.unit.test.tsx +0 -281
  258. package/src/components/DataTable/hooks/__tests__/useColumnOrderPersistence.unit.test.ts +0 -407
  259. package/src/components/DataTable/hooks/__tests__/useColumnReordering.unit.test.ts +0 -679
  260. package/src/components/DataTable/utils/__tests__/debugTools.unit.test.ts +0 -267
  261. package/src/components/DataTable/utils/__tests__/errorHandling.unit.test.ts +0 -467
  262. package/src/components/DataTable/utils/__tests__/exportUtils.unit.test.ts +0 -380
  263. package/src/components/DataTable/utils/__tests__/flexibleImport.unit.test.ts +0 -233
  264. package/src/components/DataTable/utils/__tests__/performanceUtils.unit.test.ts +0 -414
  265. package/src/components/Dialog/__tests__/Dialog.accessibility.test.tsx +0 -521
  266. package/src/components/Dialog/__tests__/Dialog.auto-size.example.tsx +0 -157
  267. package/src/components/Dialog/__tests__/Dialog.enhanced.test.tsx +0 -538
  268. package/src/components/Dialog/__tests__/Dialog.unit.test.tsx +0 -1373
  269. package/src/components/Dialog/examples/__tests__/SmartDialogExample.unit.test.tsx +0 -151
  270. package/src/components/Dialog/utils/__tests__/safeHtml.unit.test.ts +0 -611
  271. package/src/components/ErrorBoundary/__tests__/ErrorBoundary.accessibility.test.tsx +0 -517
  272. package/src/components/ErrorBoundary/__tests__/ErrorBoundary.integration.test.tsx +0 -572
  273. package/src/components/ErrorBoundary/__tests__/ErrorBoundary.unit.test.tsx +0 -579
  274. package/src/components/EventSelector/__tests__/EventSelector.test.tsx +0 -528
  275. package/src/components/FileUpload/__tests__/FileUpload.integration.test.tsx +0 -992
  276. package/src/components/FileUpload/__tests__/FileUpload.real.test.tsx +0 -927
  277. package/src/components/FileUpload/__tests__/FileUpload.test.tsx +0 -855
  278. package/src/components/FileUpload/__tests__/FileUpload.unit.test.tsx +0 -1311
  279. package/src/components/FileUpload/__tests__/FileUpload.unmocked.test.tsx +0 -937
  280. package/src/components/Footer/__tests__/Footer.accessibility.test.tsx +0 -359
  281. package/src/components/Footer/__tests__/Footer.integration.test.tsx +0 -353
  282. package/src/components/Footer/__tests__/Footer.performance.test.tsx +0 -309
  283. package/src/components/Footer/__tests__/Footer.unit.test.tsx +0 -309
  284. package/src/components/Footer/__tests__/Footer.visual.test.tsx +0 -335
  285. package/src/components/Form/__tests__/Form.accessibility.test.tsx +0 -820
  286. package/src/components/Form/__tests__/Form.unit.test.tsx +0 -305
  287. package/src/components/Form/__tests__/FormErrorSummary.unit.test.tsx +0 -285
  288. package/src/components/Form/__tests__/FormFieldset.unit.test.tsx +0 -241
  289. package/src/components/Header/__tests__/Header.accessibility.test.tsx +0 -382
  290. package/src/components/Header/__tests__/Header.comprehensive.test.tsx +0 -509
  291. package/src/components/Header/__tests__/Header.unit.test.tsx +0 -335
  292. package/src/components/InactivityWarningModal/InactivityWarningModal.test.tsx +0 -196
  293. package/src/components/InactivityWarningModal/__tests__/InactivityWarningModal.unit.test.tsx +0 -224
  294. package/src/components/Input/__tests__/Input.accessibility.test.tsx +0 -632
  295. package/src/components/Input/__tests__/Input.unit.test.tsx +0 -1121
  296. package/src/components/Label/__tests__/Label.accessibility.test.tsx +0 -239
  297. package/src/components/Label/__tests__/Label.unit.test.tsx +0 -331
  298. package/src/components/LoadingSpinner/__tests__/LoadingSpinner.accessibility.test.tsx +0 -116
  299. package/src/components/LoadingSpinner/__tests__/LoadingSpinner.unit.test.tsx +0 -144
  300. package/src/components/LoginForm/__tests__/LoginForm.accessibility.test.tsx +0 -201
  301. package/src/components/LoginForm/__tests__/LoginForm.unit.test.tsx +0 -119
  302. package/src/components/NavigationMenu/__tests__/NavigationMenu.accessibility.test.tsx +0 -378
  303. package/src/components/NavigationMenu/__tests__/NavigationMenu.enhanced.test.tsx +0 -768
  304. package/src/components/NavigationMenu/__tests__/NavigationMenu.integration.test.tsx +0 -576
  305. package/src/components/NavigationMenu/__tests__/NavigationMenu.performance.test.tsx +0 -585
  306. package/src/components/NavigationMenu/__tests__/NavigationMenu.real.component.test.tsx +0 -783
  307. package/src/components/NavigationMenu/__tests__/NavigationMenu.security.enhanced.test.tsx +0 -810
  308. package/src/components/NavigationMenu/__tests__/NavigationMenu.security.test.tsx +0 -494
  309. package/src/components/NavigationMenu/__tests__/NavigationMenu.unit.test.tsx +0 -331
  310. package/src/components/NavigationMenu/__tests__/NavigationMenu.userWorkflows.test.tsx +0 -347
  311. package/src/components/NavigationMenu/__tests__/NavigationMenu.workflows.test.tsx +0 -584
  312. package/src/components/OrganisationSelector/__tests__/OrganisationSelector.unit.test.tsx +0 -664
  313. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.accessibility.test.tsx +0 -288
  314. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.integration.test.tsx +0 -893
  315. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.performance.test.tsx +0 -629
  316. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.security.test.tsx +0 -782
  317. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.unit.test.tsx +0 -904
  318. package/src/components/PaceLoginPage/__tests__/PaceLoginPage.accessibility.test.tsx +0 -463
  319. package/src/components/PaceLoginPage/__tests__/PaceLoginPage.integration.test.tsx +0 -586
  320. package/src/components/PaceLoginPage/__tests__/PaceLoginPage.unit.test.tsx +0 -533
  321. package/src/components/PasswordReset/__tests__/PasswordChangeForm.accessibility.test.tsx +0 -408
  322. package/src/components/PasswordReset/__tests__/PasswordChangeForm.unit.test.tsx +0 -561
  323. package/src/components/PasswordReset/__tests__/PasswordReset.integration.test.tsx +0 -304
  324. package/src/components/PasswordReset/__tests__/PasswordResetForm.accessibility.test.tsx +0 -20
  325. package/src/components/PasswordReset/__tests__/PasswordResetForm.unit.test.tsx +0 -523
  326. package/src/components/PasswordReset/__tests__/__mocks__/UnifiedAuthProvider.ts +0 -29
  327. package/src/components/Print/__tests__/Print.comprehensive.test.tsx +0 -331
  328. package/src/components/PrintButton/__tests__/PrintButton.unit.test.tsx +0 -429
  329. package/src/components/PrintButton/__tests__/PrintButtonGroup.unit.test.tsx +0 -277
  330. package/src/components/PrintButton/__tests__/PrintToolbar.unit.test.tsx +0 -264
  331. package/src/components/PrintCard/__tests__/PrintCard.unit.test.tsx +0 -233
  332. package/src/components/PrintCard/__tests__/PrintCardContent.test.tsx +0 -284
  333. package/src/components/PrintCard/__tests__/PrintCardGrid.unit.test.tsx +0 -214
  334. package/src/components/PrintCard/__tests__/PrintCardImage.unit.test.tsx +0 -264
  335. package/src/components/PrintDataTable/__tests__/PrintDataTable.unit.test.tsx +0 -361
  336. package/src/components/PrintDataTable/__tests__/PrintTableGroup.unit.test.tsx +0 -314
  337. package/src/components/PrintDataTable/__tests__/PrintTableRow.unit.test.tsx +0 -362
  338. package/src/components/PrintFooter/__tests__/PrintFooter.unit.test.tsx +0 -500
  339. package/src/components/PrintFooter/__tests__/PrintFooterContent.unit.test.tsx +0 -321
  340. package/src/components/PrintFooter/__tests__/PrintFooterInfo.unit.test.tsx +0 -335
  341. package/src/components/PrintFooter/__tests__/PrintPageNumber.unit.test.tsx +0 -340
  342. package/src/components/PrintGrid/__tests__/PrintGrid.unit.test.tsx +0 -340
  343. package/src/components/PrintGrid/__tests__/PrintGridBreakpoint.unit.test.tsx +0 -261
  344. package/src/components/PrintGrid/__tests__/PrintGridContainer.unit.test.tsx +0 -338
  345. package/src/components/PrintGrid/__tests__/PrintGridItem.unit.test.tsx +0 -338
  346. package/src/components/PrintHeader/__tests__/PrintCoverHeader.unit.test.tsx +0 -309
  347. package/src/components/PrintHeader/__tests__/PrintHeader.unit.test.tsx +0 -202
  348. package/src/components/PrintLayout/__tests__/PrintLayout.unit.test.tsx +0 -238
  349. package/src/components/PrintPageBreak/__tests__/PrintPageBreak.unit.test.tsx +0 -263
  350. package/src/components/PrintPageBreak/__tests__/PrintPageBreakGroup.unit.test.tsx +0 -239
  351. package/src/components/PrintPageBreak/__tests__/PrintPageBreakIndicator.unit.test.tsx +0 -235
  352. package/src/components/PrintSection/__tests__/PrintColumn.unit.test.tsx +0 -385
  353. package/src/components/PrintSection/__tests__/PrintDivider.unit.test.tsx +0 -373
  354. package/src/components/PrintSection/__tests__/PrintSection.unit.test.tsx +0 -390
  355. package/src/components/PrintSection/__tests__/PrintSectionContent.unit.test.tsx +0 -321
  356. package/src/components/PrintSection/__tests__/PrintSectionHeader.unit.test.tsx +0 -334
  357. package/src/components/PrintText/__tests__/PrintText.unit.test.tsx +0 -351
  358. package/src/components/Progress/__tests__/Progress.accessibility.test.tsx +0 -240
  359. package/src/components/Progress/__tests__/Progress.unit.test.tsx +0 -242
  360. package/src/components/PublicLayout/__tests__/EventLogo.test.tsx +0 -761
  361. package/src/components/PublicLayout/__tests__/PublicErrorBoundary.simplified.test.tsx +0 -228
  362. package/src/components/PublicLayout/__tests__/PublicErrorBoundary.test.tsx +0 -228
  363. package/src/components/PublicLayout/__tests__/PublicLoadingSpinner.test.tsx +0 -459
  364. package/src/components/PublicLayout/__tests__/PublicPageFooter.test.tsx +0 -362
  365. package/src/components/PublicLayout/__tests__/PublicPageHeader.test.tsx +0 -522
  366. package/src/components/PublicLayout/__tests__/PublicPageLayout.test.tsx +0 -599
  367. package/src/components/PublicLayout/__tests__/PublicPageProvider.test.tsx +0 -513
  368. package/src/components/RBAC/__tests__/PagePermissionGuard.unit.test.tsx +0 -683
  369. package/src/components/RBAC/__tests__/RBAC.integration.test.tsx +0 -573
  370. package/src/components/RBAC/__tests__/RBACGuard.unit.test.tsx +0 -467
  371. package/src/components/RBAC/__tests__/RBACProvider.accessibility.test.tsx +0 -475
  372. package/src/components/RBAC/__tests__/RBACProvider.advanced.test.tsx +0 -569
  373. package/src/components/RBAC/__tests__/RBACProvider.integration.test.tsx +0 -352
  374. package/src/components/RBAC/__tests__/RBACProvider.unit.test.tsx +0 -128
  375. package/src/components/RBAC/__tests__/RoleBasedContent.unit.test.tsx +0 -657
  376. package/src/components/Select/__tests__/SearchableSelect.unit.test.tsx +0 -437
  377. package/src/components/Select/__tests__/Select.accessibility.test.tsx +0 -1202
  378. package/src/components/Select/__tests__/Select.actual.test.tsx +0 -774
  379. package/src/components/Select/__tests__/Select.comprehensive.test.tsx +0 -837
  380. package/src/components/Select/__tests__/Select.enhanced.test.tsx +0 -1101
  381. package/src/components/Select/__tests__/Select.integration.test.tsx +0 -772
  382. package/src/components/Select/__tests__/Select.performance.test.tsx +0 -695
  383. package/src/components/Select/__tests__/Select.real-world.test.tsx +0 -1046
  384. package/src/components/Select/__tests__/Select.search-algorithms.test.tsx +0 -968
  385. package/src/components/Select/__tests__/Select.unit.test.tsx +0 -647
  386. package/src/components/Select/__tests__/Select.utils.test.tsx +0 -890
  387. package/src/components/Table/__tests__/Table.accessibility.test.tsx +0 -233
  388. package/src/components/Table/__tests__/Table.unit.test.tsx +0 -235
  389. package/src/components/Toast/__tests__/Toast.accessibility.test.tsx +0 -238
  390. package/src/components/Toast/__tests__/Toast.integration.test.tsx +0 -699
  391. package/src/components/Toast/__tests__/Toast.unit.test.tsx +0 -750
  392. package/src/components/Tooltip/__tests__/Tooltip.accessibility.test.tsx +0 -121
  393. package/src/components/Tooltip/__tests__/Tooltip.unit.test.tsx +0 -185
  394. package/src/components/UserMenu/__tests__/UserMenu.accessibility.test.tsx +0 -139
  395. package/src/components/UserMenu/__tests__/UserMenu.integration.test.tsx +0 -188
  396. package/src/components/UserMenu/__tests__/UserMenu.unit.test.tsx +0 -458
  397. package/src/components/__tests__/EdgeCaseTesting.enhanced.test.tsx +0 -524
  398. package/src/components/__tests__/ErrorTesting.enhanced.test.tsx +0 -455
  399. package/src/components/__tests__/SuperAdminGuard.test.tsx +0 -456
  400. package/src/components/__tests__/SuperAdminGuard.unit.test.tsx +0 -456
  401. package/src/components/examples/__tests__/PermissionExample.unit.test.tsx +0 -360
  402. package/src/hooks/__tests__/hooks.integration.test.tsx +0 -575
  403. package/src/hooks/__tests__/useApiFetch.unit.test.ts +0 -115
  404. package/src/hooks/__tests__/useComponentPerformance.unit.test.tsx +0 -133
  405. package/src/hooks/__tests__/useDebounce.unit.test.ts +0 -82
  406. package/src/hooks/__tests__/useFocusTrap.unit.test.tsx +0 -293
  407. package/src/hooks/__tests__/useInactivityTracker.unit.test.ts +0 -385
  408. package/src/hooks/__tests__/useOrganisationPermissions.unit.test.tsx +0 -286
  409. package/src/hooks/__tests__/useOrganisationSecurity.unit.test.tsx +0 -838
  410. package/src/hooks/__tests__/usePermissionCache.unit.test.ts +0 -627
  411. package/src/hooks/__tests__/useRBAC.unit.test.ts +0 -911
  412. package/src/hooks/__tests__/useSecureDataAccess.unit.test.tsx +0 -537
  413. package/src/hooks/__tests__/useToast.unit.test.tsx +0 -62
  414. package/src/hooks/__tests__/useZodForm.unit.test.tsx +0 -37
  415. package/src/hooks/public/__tests__/usePublicEvent.test.tsx +0 -397
  416. package/src/hooks/public/__tests__/usePublicEventLogo.test.tsx +0 -690
  417. package/src/hooks/public/__tests__/usePublicRouteParams.test.tsx +0 -449
  418. package/src/providers/__tests__/EventProvider.unit.test.tsx +0 -768
  419. package/src/providers/__tests__/OrganisationProvider.basic.test.tsx +0 -116
  420. package/src/providers/__tests__/OrganisationProvider.unit.test.tsx +0 -1312
  421. package/src/providers/__tests__/UnifiedAuthProvider.inactivity.test.tsx +0 -601
  422. package/src/providers/__tests__/UnifiedAuthProvider.unit.test.tsx +0 -683
  423. package/src/providers/__tests__/index.unit.test.ts +0 -78
  424. package/src/rbac/__tests__/PagePermissionGuard.test.tsx +0 -673
  425. package/src/rbac/__tests__/README.md +0 -170
  426. package/src/rbac/__tests__/RoleBasedRouter.test.tsx +0 -709
  427. package/src/rbac/__tests__/TestContext.tsx +0 -72
  428. package/src/rbac/__tests__/__mocks__/cache.ts +0 -144
  429. package/src/rbac/__tests__/__mocks__/supabase.ts +0 -152
  430. package/src/rbac/__tests__/adapters-hooks-comprehensive.test.tsx +0 -782
  431. package/src/rbac/__tests__/adapters-hooks.test.tsx +0 -561
  432. package/src/rbac/__tests__/adapters.comprehensive.test.tsx +0 -963
  433. package/src/rbac/__tests__/adapters.test.tsx +0 -444
  434. package/src/rbac/__tests__/api.test.ts +0 -620
  435. package/src/rbac/__tests__/audit-observability-comprehensive.test.ts +0 -792
  436. package/src/rbac/__tests__/audit-observability.test.ts +0 -549
  437. package/src/rbac/__tests__/audit.test.ts +0 -616
  438. package/src/rbac/__tests__/build-contract-compliance-simple.test.ts +0 -230
  439. package/src/rbac/__tests__/cache-invalidation-comprehensive.test.ts +0 -889
  440. package/src/rbac/__tests__/cache-invalidation.test.ts +0 -457
  441. package/src/rbac/__tests__/cache.test.ts +0 -458
  442. package/src/rbac/__tests__/components-navigation-guard.enhanced.test.tsx +0 -859
  443. package/src/rbac/__tests__/components-navigation-guard.test.tsx +0 -895
  444. package/src/rbac/__tests__/components-navigation-provider.test.tsx +0 -692
  445. package/src/rbac/__tests__/components-page-permission-guard.test.tsx +0 -673
  446. package/src/rbac/__tests__/components-page-permission-provider.test.tsx +0 -614
  447. package/src/rbac/__tests__/components-permission-enforcer.enhanced.fixed.test.tsx +0 -836
  448. package/src/rbac/__tests__/components-permission-enforcer.enhanced.test.tsx +0 -837
  449. package/src/rbac/__tests__/components-permission-enforcer.test.tsx +0 -825
  450. package/src/rbac/__tests__/components-role-based-router.test.tsx +0 -709
  451. package/src/rbac/__tests__/components-secure-data-provider.test.tsx +0 -607
  452. package/src/rbac/__tests__/config.test.ts +0 -583
  453. package/src/rbac/__tests__/core-logic-unit.test.ts +0 -190
  454. package/src/rbac/__tests__/core-permission-logic-comprehensive.test.ts +0 -1467
  455. package/src/rbac/__tests__/core-permission-logic-fixed.test.ts +0 -151
  456. package/src/rbac/__tests__/core-permission-logic-simple.test.ts +0 -968
  457. package/src/rbac/__tests__/core-permission-logic.test.ts +0 -966
  458. package/src/rbac/__tests__/edge-cases-comprehensive.test.ts +0 -988
  459. package/src/rbac/__tests__/edge-cases.test.ts +0 -654
  460. package/src/rbac/__tests__/engine.test.ts +0 -361
  461. package/src/rbac/__tests__/engine.unit.test.ts +0 -361
  462. package/src/rbac/__tests__/hooks.enhanced.test.tsx +0 -979
  463. package/src/rbac/__tests__/hooks.fixed.test.tsx +0 -475
  464. package/src/rbac/__tests__/hooks.test.tsx +0 -385
  465. package/src/rbac/__tests__/index.test.ts +0 -269
  466. package/src/rbac/__tests__/integration.enhanced.test.tsx +0 -824
  467. package/src/rbac/__tests__/page-permission-guard-super-admin.test.tsx +0 -261
  468. package/src/rbac/__tests__/performance.enhanced.test.tsx +0 -724
  469. package/src/rbac/__tests__/permissions.test.ts +0 -383
  470. package/src/rbac/__tests__/requires-event.test.ts +0 -330
  471. package/src/rbac/__tests__/scope-isolation-comprehensive.test.ts +0 -1349
  472. package/src/rbac/__tests__/scope-isolation.test.ts +0 -755
  473. package/src/rbac/__tests__/secure-client-rls-comprehensive.test.ts +0 -592
  474. package/src/rbac/__tests__/secure-client-rls.test.ts +0 -377
  475. package/src/rbac/__tests__/security.test.ts +0 -296
  476. package/src/rbac/__tests__/setup.ts +0 -228
  477. package/src/rbac/__tests__/test-utils-enhanced.tsx +0 -400
  478. package/src/rbac/__tests__/types.test.ts +0 -685
  479. package/src/rbac/components/__tests__/EnhancedNavigationMenu.test.tsx +0 -631
  480. package/src/rbac/components/__tests__/NavigationProvider.test.tsx +0 -667
  481. package/src/rbac/components/__tests__/PagePermissionProvider.test.tsx +0 -647
  482. package/src/rbac/components/__tests__/SecureDataProvider.test.tsx +0 -496
  483. package/src/rbac/testing/__tests__/index.test.tsx +0 -342
  484. package/src/rbac/utils/__tests__/eventContext.test.ts +0 -428
  485. package/src/rbac/utils/__tests__/eventContext.unit.test.ts +0 -428
  486. package/src/styles/__tests__/styles.unit.test.ts +0 -164
  487. package/src/test-dom-cleanup.test.tsx +0 -38
  488. package/src/theming/__tests__/README.md +0 -335
  489. package/src/theming/__tests__/runtime.accessibility.test.ts +0 -474
  490. package/src/theming/__tests__/runtime.error.test.ts +0 -616
  491. package/src/theming/__tests__/runtime.integration.test.ts +0 -376
  492. package/src/theming/__tests__/runtime.performance.test.ts +0 -411
  493. package/src/theming/__tests__/runtime.unit.test.ts +0 -470
  494. package/src/types/__tests__/database.unit.test.ts +0 -489
  495. package/src/types/__tests__/guards.unit.test.ts +0 -146
  496. package/src/types/__tests__/index.unit.test.ts +0 -77
  497. package/src/types/__tests__/organisation.unit.test.ts +0 -713
  498. package/src/types/__tests__/rbac.unit.test.ts +0 -621
  499. package/src/types/__tests__/security.unit.test.ts +0 -347
  500. package/src/types/__tests__/supabase.unit.test.ts +0 -658
  501. package/src/types/__tests__/theme.unit.test.ts +0 -218
  502. package/src/types/__tests__/unified.unit.test.ts +0 -537
  503. package/src/types/__tests__/validation.unit.test.ts +0 -616
  504. package/src/utils/__tests__/appConfig.unit.test.ts +0 -55
  505. package/src/utils/__tests__/appNameResolver.unit.test.ts +0 -137
  506. package/src/utils/__tests__/audit.unit.test.ts +0 -69
  507. package/src/utils/__tests__/auth-utils.unit.test.ts +0 -70
  508. package/src/utils/__tests__/bundleAnalysis.unit.test.ts +0 -317
  509. package/src/utils/__tests__/cn.unit.test.ts +0 -34
  510. package/src/utils/__tests__/deviceFingerprint.unit.test.ts +0 -480
  511. package/src/utils/__tests__/dynamicUtils.unit.test.ts +0 -322
  512. package/src/utils/__tests__/formatDate.unit.test.ts +0 -109
  513. package/src/utils/__tests__/formatting.unit.test.ts +0 -66
  514. package/src/utils/__tests__/index.unit.test.ts +0 -251
  515. package/src/utils/__tests__/lazyLoad.unit.test.tsx +0 -304
  516. package/src/utils/__tests__/organisationContext.unit.test.ts +0 -192
  517. package/src/utils/__tests__/performanceBudgets.unit.test.ts +0 -259
  518. package/src/utils/__tests__/permissionTypes.unit.test.ts +0 -250
  519. package/src/utils/__tests__/permissionUtils.unit.test.ts +0 -362
  520. package/src/utils/__tests__/sanitization.unit.test.ts +0 -346
  521. package/src/utils/__tests__/schemaUtils.unit.test.ts +0 -441
  522. package/src/utils/__tests__/secureDataAccess.unit.test.ts +0 -334
  523. package/src/utils/__tests__/secureErrors.unit.test.ts +0 -377
  524. package/src/utils/__tests__/secureStorage.unit.test.ts +0 -293
  525. package/src/utils/__tests__/security.unit.test.ts +0 -127
  526. package/src/utils/__tests__/securityMonitor.unit.test.ts +0 -280
  527. package/src/utils/__tests__/sessionTracking.unit.test.ts +0 -370
  528. package/src/utils/__tests__/validation.unit.test.ts +0 -84
  529. package/src/utils/__tests__/validationUtils.unit.test.ts +0 -571
  530. package/src/utils/print/__tests__/PrintDataProcessor.unit.test.ts +0 -219
  531. package/src/utils/print/__tests__/usePrintOptimization.unit.test.tsx +0 -353
  532. package/src/utils/storage/__tests__/config.unit.test.ts +0 -206
  533. package/src/utils/storage/__tests__/helpers.unit.test.ts +0 -648
  534. package/src/utils/storage/__tests__/index.unit.test.ts +0 -167
  535. package/src/utils/storage/__tests__/types.unit.test.ts +0 -441
  536. package/src/validation/__tests__/common.unit.test.ts +0 -101
  537. package/src/validation/__tests__/csrf.unit.test.ts +0 -302
  538. package/src/validation/__tests__/passwordSchema.unit.test.ts +0 -98
  539. package/src/validation/__tests__/sqlInjectionProtection.unit.test.ts +0 -466
  540. /package/dist/{DataTable-EEUDXPE5.js.map → DataTable-GX3XERFJ.js.map} +0 -0
  541. /package/dist/{chunk-VYG4AXYW.js.map → chunk-5EL3KHOQ.js.map} +0 -0
@@ -1,1744 +0,0 @@
1
- import {
2
- Button,
3
- Dialog,
4
- DialogContent,
5
- DialogDescription,
6
- DialogHeader,
7
- DialogTitle,
8
- init_Button,
9
- init_Dialog
10
- } from "./chunk-ETEJVKYK.js";
11
- import {
12
- AccessLevel,
13
- init_unified
14
- } from "./chunk-YDJW5XTN.js";
15
- import {
16
- DebugLogger,
17
- init_debugLogger,
18
- init_organisationContext,
19
- setOrganisationContext
20
- } from "./chunk-2V3Y6YBC.js";
21
- import {
22
- __esm,
23
- __export
24
- } from "./chunk-PLDDJCW6.js";
25
-
26
- // src/providers/AuthProvider.tsx
27
- import { createContext, useContext, useState, useEffect, useCallback, useMemo } from "react";
28
- import { AuthError } from "@supabase/supabase-js";
29
- import { jsx } from "react/jsx-runtime";
30
- function AuthProvider({ children, supabaseClient }) {
31
- const [user, setUser] = useState(null);
32
- const [session, setSession] = useState(null);
33
- const [authLoading, setAuthLoading] = useState(true);
34
- const [authError, setAuthError] = useState(null);
35
- useEffect(() => {
36
- const handleError = (event) => {
37
- if (event.error?.message?.includes("AuthSessionMissingError") || event.error?.message?.includes("Auth session missing")) {
38
- console.warn("[AuthProvider] Suppressing AuthSessionMissingError during logout");
39
- event.preventDefault();
40
- return false;
41
- }
42
- };
43
- const handleUnhandledRejection = (event) => {
44
- if (event.reason?.message?.includes("AuthSessionMissingError") || event.reason?.message?.includes("Auth session missing")) {
45
- console.warn("[AuthProvider] Suppressing unhandled AuthSessionMissingError");
46
- event.preventDefault();
47
- return false;
48
- }
49
- };
50
- window.addEventListener("error", handleError);
51
- window.addEventListener("unhandledrejection", handleUnhandledRejection);
52
- return () => {
53
- window.removeEventListener("error", handleError);
54
- window.removeEventListener("unhandledrejection", handleUnhandledRejection);
55
- };
56
- }, []);
57
- useEffect(() => {
58
- if (!supabaseClient) return;
59
- const loadInitialUser = async () => {
60
- try {
61
- const response = await supabaseClient.auth.getUser();
62
- const { data: { user: initialUser }, error } = response || { data: { user: null }, error: null };
63
- if (error) {
64
- console.warn("Failed to get initial user:", error);
65
- setAuthLoading(false);
66
- return;
67
- }
68
- if (initialUser) {
69
- setUser(initialUser);
70
- }
71
- setAuthLoading(false);
72
- } catch (error) {
73
- console.error("Error loading initial user:", error);
74
- setAuthLoading(false);
75
- }
76
- };
77
- const timeoutId = setTimeout(() => {
78
- if (authLoading && !user && !session) {
79
- loadInitialUser();
80
- }
81
- }, 100);
82
- return () => clearTimeout(timeoutId);
83
- }, [supabaseClient, authLoading, user, session]);
84
- useEffect(() => {
85
- if (!supabaseClient) {
86
- setAuthLoading(false);
87
- return;
88
- }
89
- const timeoutId = setTimeout(() => {
90
- if (authLoading) {
91
- console.warn("AuthProvider: Auth loading timeout reached");
92
- setAuthLoading(false);
93
- }
94
- }, 2e3);
95
- try {
96
- DebugLogger.log("AuthProvider", "Setting up auth state change listener...");
97
- const authStateChange = supabaseClient.auth.onAuthStateChange(
98
- (event, session2) => {
99
- try {
100
- DebugLogger.log("AuthProvider", "Auth state changed:", event, session2?.user?.email);
101
- clearTimeout(timeoutId);
102
- if (event === "SIGNED_OUT") {
103
- DebugLogger.log("AuthProvider", "User signed out, clearing all state");
104
- setSession(null);
105
- setUser(null);
106
- setAuthLoading(false);
107
- setAuthError(null);
108
- } else {
109
- setSession(session2);
110
- setUser(session2?.user ?? null);
111
- setAuthLoading(false);
112
- if (session2) {
113
- setAuthError(null);
114
- }
115
- }
116
- } catch (error) {
117
- console.warn("[AuthProvider] Error in auth state change handler:", error);
118
- setAuthLoading(false);
119
- }
120
- }
121
- );
122
- const subscription = authStateChange?.data?.subscription || authStateChange;
123
- return () => {
124
- clearTimeout(timeoutId);
125
- if (subscription && typeof subscription.unsubscribe === "function") {
126
- DebugLogger.log("AuthProvider", "Cleaning up auth state listener");
127
- subscription.unsubscribe();
128
- }
129
- };
130
- } catch (error) {
131
- console.error("AuthProvider: Error setting up auth state change listener:", error);
132
- if (error instanceof Error && !error.message.includes("No API key found") && !error.message.includes("AuthSessionMissingError") && !error.message.includes("Auth session missing")) {
133
- setAuthError(error);
134
- }
135
- setAuthLoading(false);
136
- clearTimeout(timeoutId);
137
- return () => {
138
- };
139
- }
140
- }, [supabaseClient]);
141
- const signIn = useCallback(async (email, password) => {
142
- if (!supabaseClient) {
143
- const error = new Error("No Supabase client provided");
144
- setAuthError(error);
145
- return { error };
146
- }
147
- setAuthLoading(true);
148
- setAuthError(null);
149
- try {
150
- const { error } = await supabaseClient.auth.signInWithPassword({ email, password: password || "" });
151
- if (error) {
152
- setAuthError(error);
153
- }
154
- return { error };
155
- } catch (err) {
156
- const authError2 = err;
157
- setAuthError(authError2);
158
- return { error: authError2 };
159
- } finally {
160
- setAuthLoading(false);
161
- }
162
- }, [supabaseClient]);
163
- const signUp = useCallback(async (email, password) => {
164
- if (!supabaseClient) {
165
- const error = new Error("No Supabase client provided");
166
- setAuthError(error);
167
- return { error };
168
- }
169
- setAuthError(null);
170
- try {
171
- const { error } = await supabaseClient.auth.signUp({ email, password });
172
- if (error) {
173
- setAuthError(error);
174
- }
175
- return { error };
176
- } catch (err) {
177
- const authError2 = err;
178
- setAuthError(authError2);
179
- return { error: authError2 };
180
- }
181
- }, [supabaseClient]);
182
- const signOut = useCallback(async () => {
183
- DebugLogger.log("AuthProvider", "signOut called");
184
- setAuthLoading(false);
185
- if (!supabaseClient) {
186
- DebugLogger.log("AuthProvider", "No supabase client, clearing state manually");
187
- setUser(null);
188
- setSession(null);
189
- setAuthError(null);
190
- return { error: null };
191
- }
192
- DebugLogger.log("AuthProvider", "Pre-clearing state before signOut call");
193
- setUser(null);
194
- setSession(null);
195
- setAuthError(null);
196
- try {
197
- DebugLogger.log("AuthProvider", "Calling supabase signOut");
198
- const signOutPromise = supabaseClient.auth.signOut();
199
- const timeoutPromise = new Promise(
200
- (_, reject) => setTimeout(() => reject(new Error("SignOut timeout")), 3e3)
201
- );
202
- const { error } = await Promise.race([signOutPromise, timeoutPromise]);
203
- if (error && !error.message?.includes("SignOut timeout")) {
204
- console.error("[AuthProvider] signOut error:", error);
205
- }
206
- DebugLogger.log("AuthProvider", "signOut process completed");
207
- return { error: null };
208
- } catch (err) {
209
- console.warn("[AuthProvider] signOut exception (ignored):", err);
210
- return { error: null };
211
- }
212
- }, [supabaseClient]);
213
- const resetPassword = useCallback(async (email) => {
214
- if (!supabaseClient) {
215
- const error = new Error("No Supabase client provided");
216
- setAuthError(error);
217
- return { error };
218
- }
219
- setAuthError(null);
220
- try {
221
- const { error } = await supabaseClient.auth.resetPasswordForEmail(email);
222
- if (error) {
223
- setAuthError(error);
224
- }
225
- return { error };
226
- } catch (err) {
227
- const authError2 = err;
228
- setAuthError(authError2);
229
- return { error: authError2 };
230
- }
231
- }, [supabaseClient]);
232
- const updatePassword = useCallback(async (password) => {
233
- if (!supabaseClient) return { error: new AuthError("Supabase client not available.", 500) };
234
- const { error } = await supabaseClient.auth.updateUser({ password });
235
- if (error) {
236
- setAuthError(error);
237
- }
238
- return { error };
239
- }, [supabaseClient]);
240
- const refreshSession = useCallback(async () => {
241
- if (!supabaseClient) {
242
- const error = new Error("No Supabase client provided");
243
- setAuthError(error);
244
- return { error };
245
- }
246
- setAuthError(null);
247
- try {
248
- const { error } = await supabaseClient.auth.refreshSession();
249
- if (error) {
250
- setAuthError(error);
251
- }
252
- return { error };
253
- } catch (err) {
254
- const authError2 = err;
255
- setAuthError(authError2);
256
- return { error: authError2 };
257
- }
258
- }, [supabaseClient]);
259
- const isAuthenticated = !!user;
260
- const contextValue = useMemo(() => ({
261
- user,
262
- session,
263
- isAuthenticated,
264
- authLoading,
265
- authError,
266
- error: authError,
267
- // Alias for backward compatibility
268
- supabase: supabaseClient || null,
269
- signIn,
270
- signUp,
271
- signOut,
272
- resetPassword,
273
- updatePassword,
274
- refreshSession
275
- }), [
276
- user,
277
- session,
278
- isAuthenticated,
279
- authLoading,
280
- authError,
281
- supabaseClient,
282
- signIn,
283
- signUp,
284
- signOut,
285
- resetPassword,
286
- updatePassword,
287
- refreshSession
288
- ]);
289
- return /* @__PURE__ */ jsx(AuthContext.Provider, { value: contextValue, children });
290
- }
291
- var AuthContext, useAuth;
292
- var init_AuthProvider = __esm({
293
- "src/providers/AuthProvider.tsx"() {
294
- "use strict";
295
- init_debugLogger();
296
- AuthContext = createContext(void 0);
297
- useAuth = () => {
298
- const context = useContext(AuthContext);
299
- if (!context) {
300
- throw new Error("useAuth must be used within an AuthProvider");
301
- }
302
- return context;
303
- };
304
- }
305
- });
306
-
307
- // src/providers/RBACProvider.tsx
308
- import { createContext as createContext2, useContext as useContext2, useState as useState2, useEffect as useEffect2, useCallback as useCallback2, useMemo as useMemo2 } from "react";
309
- import { jsx as jsx2 } from "react/jsx-runtime";
310
- function RBACProvider({
311
- children,
312
- supabaseClient,
313
- user,
314
- session,
315
- appName,
316
- enableRBAC = false,
317
- persistState = true,
318
- enablePersistence,
319
- requireOrganisationContext: _requireOrganisationContext = true
320
- }) {
321
- const shouldPersist = enablePersistence !== void 0 ? enablePersistence : persistState;
322
- const [permissions, setPermissions] = useState2({});
323
- const [roles, setRoles] = useState2([]);
324
- const [accessLevel, setAccessLevel] = useState2("viewer" /* VIEWER */);
325
- const [rbacLoading, setRbacLoading] = useState2(false);
326
- const [rbacError, setRbacError] = useState2(null);
327
- const [selectedEventId, setSelectedEventId] = useState2(null);
328
- const [appConfig, setAppConfig] = useState2(null);
329
- const [userEventAccess, setUserEventAccess] = useState2([]);
330
- const [eventAccessLoading, setEventAccessLoading] = useState2(false);
331
- const [selectedOrganisationId, _setSelectedOrganisationId] = useState2(null);
332
- useEffect2(() => {
333
- if (!supabaseClient) return;
334
- const loadAppConfig = async () => {
335
- try {
336
- const { getCurrentAppName } = await import("./appNameResolver-7GHF5ED2.js");
337
- const resolvedAppName = getCurrentAppName() || appName;
338
- const { data: appData, error: appError } = await supabaseClient.from("rbac_apps").select("id").eq("name", resolvedAppName).eq("is_active", true).single();
339
- if (appError || !appData) {
340
- console.warn("App not found or inactive:", resolvedAppName);
341
- setAppConfig({
342
- supports_direct_access: false,
343
- requires_event: true
344
- });
345
- return;
346
- }
347
- const response = await supabaseClient.rpc("get_app_config", {
348
- p_app_id: appData.id
349
- });
350
- const { data } = response || {};
351
- if (data && data.length > 0) {
352
- setAppConfig({
353
- supports_direct_access: data[0].supports_direct_access,
354
- requires_event: data[0].requires_event
355
- });
356
- } else {
357
- setAppConfig({
358
- supports_direct_access: false,
359
- requires_event: true
360
- });
361
- }
362
- } catch (error) {
363
- console.warn("Clearing corrupted localStorage data");
364
- if (typeof localStorage !== "undefined") {
365
- localStorage.removeItem(STORAGE_KEYS.SELECTED_EVENT);
366
- }
367
- console.warn("Failed to load app configuration:", error);
368
- setAppConfig({
369
- supports_direct_access: false,
370
- requires_event: true
371
- });
372
- }
373
- };
374
- loadAppConfig();
375
- }, [supabaseClient, appName]);
376
- useEffect2(() => {
377
- const isSuperAdmin = user?.user_metadata?.globalRole === "super_admin";
378
- if (isSuperAdmin) {
379
- setRoles(["super_admin"]);
380
- } else {
381
- setRoles([]);
382
- }
383
- }, [user]);
384
- useEffect2(() => {
385
- if (!shouldPersist) return;
386
- try {
387
- const persistedEvent = localStorage.getItem(STORAGE_KEYS.SELECTED_EVENT);
388
- if (persistedEvent) {
389
- setSelectedEventId(JSON.parse(persistedEvent));
390
- }
391
- } catch (error) {
392
- console.warn("Clearing corrupted localStorage data");
393
- localStorage.removeItem(STORAGE_KEYS.SELECTED_EVENT);
394
- }
395
- }, [shouldPersist]);
396
- useEffect2(() => {
397
- if (!shouldPersist) return;
398
- try {
399
- if (selectedEventId) {
400
- localStorage.setItem(
401
- STORAGE_KEYS.SELECTED_EVENT,
402
- JSON.stringify(selectedEventId)
403
- );
404
- } else {
405
- localStorage.removeItem(STORAGE_KEYS.SELECTED_EVENT);
406
- }
407
- } catch (error) {
408
- console.warn("Clearing corrupted localStorage data");
409
- localStorage.removeItem(STORAGE_KEYS.SELECTED_EVENT);
410
- console.warn("Failed to persist auth state:", error);
411
- }
412
- }, [selectedEventId, shouldPersist]);
413
- const refreshPermissions = useCallback2(async (eventId) => {
414
- if (!supabaseClient || !user || !appConfig || !session) {
415
- DebugLogger.log("RBACProvider", "refreshPermissions: Missing required dependencies, clearing permissions");
416
- setPermissions({});
417
- setRoles([]);
418
- setAccessLevel("viewer" /* VIEWER */);
419
- return;
420
- }
421
- const isSuperAdmin = user?.user_metadata?.globalRole === "super_admin";
422
- if (isSuperAdmin) {
423
- setPermissions({
424
- "admin:create": true,
425
- "admin:read": true,
426
- "admin:update": true,
427
- "admin:delete": true,
428
- "users:create": true,
429
- "users:read": true,
430
- "users:update": true,
431
- "users:delete": true,
432
- "events:create": true,
433
- "events:read": true,
434
- "events:update": true,
435
- "events:delete": true
436
- });
437
- setRoles(["super_admin"]);
438
- setAccessLevel("admin" /* ADMIN */);
439
- return;
440
- }
441
- const shouldLoadDirectPermissions = !eventId && !appConfig.requires_event;
442
- const shouldLoadEventPermissions = eventId;
443
- const shouldClearPermissions = !eventId && appConfig.requires_event;
444
- if (shouldClearPermissions) {
445
- setPermissions({});
446
- setRoles([]);
447
- setAccessLevel("viewer" /* VIEWER */);
448
- return;
449
- }
450
- if (!shouldLoadDirectPermissions && !shouldLoadEventPermissions) {
451
- return;
452
- }
453
- setRbacLoading(true);
454
- setRbacError(null);
455
- try {
456
- const { getCurrentAppName } = await import("./appNameResolver-7GHF5ED2.js");
457
- const resolvedAppName = getCurrentAppName() || appName;
458
- const { data: appData, error: appError } = await supabaseClient.from("rbac_apps").select("id").eq("name", resolvedAppName).eq("is_active", true).single();
459
- if (appError || !appData) {
460
- console.warn("App not found or inactive:", resolvedAppName);
461
- setRbacLoading(false);
462
- return;
463
- }
464
- const { data, error } = await supabaseClient.rpc("get_rbac_permissions", {
465
- p_user_id: user.id,
466
- p_app_id: appData.id,
467
- p_event_id: eventId || null,
468
- p_organisation_id: selectedOrganisationId || null
469
- });
470
- if (error) {
471
- throw error;
472
- }
473
- const { permissions: permissions2, roles: roles2, access_level } = transformRBACPermissions(data, appName);
474
- setPermissions(permissions2);
475
- setRoles(roles2);
476
- setAccessLevel(access_level);
477
- } catch (err) {
478
- setRbacError(err);
479
- } finally {
480
- setRbacLoading(false);
481
- }
482
- }, [supabaseClient, user, session, appName, appConfig, selectedOrganisationId]);
483
- const loadUserEventAccess = useCallback2(async () => {
484
- if (!supabaseClient || !user || !session) {
485
- DebugLogger.log("RBACProvider", "loadUserEventAccess: Missing required dependencies, clearing event access");
486
- setUserEventAccess([]);
487
- return;
488
- }
489
- setEventAccessLoading(true);
490
- try {
491
- const { getCurrentAppName } = await import("./appNameResolver-7GHF5ED2.js");
492
- const resolvedAppName = getCurrentAppName() || appName;
493
- const { data: appData, error: appError } = await supabaseClient.from("rbac_apps").select("id").eq("name", resolvedAppName).eq("is_active", true).single();
494
- if (appError || !appData) {
495
- console.warn("App not found or inactive:", resolvedAppName);
496
- setEventAccessLoading(false);
497
- return;
498
- }
499
- const { data, error } = await supabaseClient.from("rbac_event_app_roles").select(`
500
- event_id,
501
- role,
502
- granted_at
503
- `).eq("user_id", user.id).eq("app_id", appData.id);
504
- if (error) {
505
- console.error("Failed to load user event access:", error);
506
- setUserEventAccess([]);
507
- return;
508
- }
509
- const eventAccess = data?.map((item) => ({
510
- event_id: item.event_id,
511
- event_name: "Unknown Event",
512
- // Event details not available in this query
513
- event_description: null,
514
- // Not available in this schema
515
- start_date: "",
516
- // Event date not available in this query
517
- end_date: "",
518
- // Event date not available in this query
519
- event_status: "unknown",
520
- // Not available in this schema
521
- app_id: appData.id,
522
- access_level: item.role,
523
- // Map role to access_level
524
- granted_at: item.granted_at,
525
- organisation_id: ""
526
- // Will be populated from event's organisation_id if needed
527
- })) || [];
528
- setUserEventAccess(eventAccess);
529
- } catch (error) {
530
- console.warn("Clearing corrupted localStorage data");
531
- localStorage.removeItem(STORAGE_KEYS.SELECTED_EVENT);
532
- console.error("Error loading user event access:", error);
533
- setUserEventAccess([]);
534
- } finally {
535
- setEventAccessLoading(false);
536
- }
537
- }, [supabaseClient, user, session, appName]);
538
- const getUserEventAccess = useCallback2((eventId) => {
539
- return userEventAccess.find((access) => access.event_id === eventId);
540
- }, [userEventAccess]);
541
- useEffect2(() => {
542
- if (!user || !appConfig || !session) {
543
- DebugLogger.log("RBACProvider", "Skipping permission refresh - no user, session, or app config");
544
- return;
545
- }
546
- const isSuperAdmin = user?.user_metadata?.globalRole === "super_admin";
547
- if (isSuperAdmin) {
548
- setPermissions({
549
- "admin:create": true,
550
- "admin:read": true,
551
- "admin:update": true,
552
- "admin:delete": true,
553
- "users:create": true,
554
- "users:read": true,
555
- "users:update": true,
556
- "users:delete": true,
557
- "events:create": true,
558
- "events:read": true,
559
- "events:update": true,
560
- "events:delete": true
561
- });
562
- setRoles(["super_admin"]);
563
- setAccessLevel("admin" /* ADMIN */);
564
- return;
565
- }
566
- if (selectedEventId) {
567
- refreshPermissions(selectedEventId);
568
- } else if (!appConfig.requires_event) {
569
- refreshPermissions();
570
- } else {
571
- setPermissions({});
572
- setRoles([]);
573
- setAccessLevel("viewer" /* VIEWER */);
574
- }
575
- }, [selectedEventId, user, session, appConfig, refreshPermissions]);
576
- useEffect2(() => {
577
- let isMounted = true;
578
- if (user && session) {
579
- DebugLogger.log("RBACProvider", "Loading user event access for authenticated user");
580
- loadUserEventAccess().catch((error) => {
581
- if (isMounted) {
582
- console.error("Error loading user event access:", error);
583
- }
584
- });
585
- } else {
586
- DebugLogger.log("RBACProvider", "Clearing user event access - no user or session");
587
- if (isMounted) {
588
- setUserEventAccess([]);
589
- }
590
- }
591
- return () => {
592
- isMounted = false;
593
- };
594
- }, [user, session, loadUserEventAccess]);
595
- const hasPermission = useCallback2((permission) => {
596
- const hasPerm = !!permissions[permission];
597
- return hasPerm;
598
- }, [permissions]);
599
- const hasAnyPermission = useCallback2((perms) => perms.some((p) => !!permissions[p]), [permissions]);
600
- const hasAllPermissions = useCallback2((perms) => perms.every((p) => !!permissions[p]), [permissions]);
601
- const hasRole = useCallback2((role) => {
602
- const isSuperAdmin = user?.user_metadata?.globalRole === "super_admin";
603
- if (role.toLowerCase() === "super_admin") {
604
- return isSuperAdmin;
605
- }
606
- return roles.includes(role);
607
- }, [roles, user]);
608
- const hasAccessLevel = useCallback2((level) => {
609
- const levels = Object.values(AccessLevel);
610
- return levels.indexOf(accessLevel) >= levels.indexOf(level);
611
- }, [accessLevel]);
612
- const canAccess = useCallback2((resource, action) => {
613
- const permission = `${resource}:${action}`;
614
- const hasAccess = hasPermission(permission);
615
- return hasAccess;
616
- }, [hasPermission]);
617
- const validatePermission = useCallback2(async (permission) => hasPermission(permission), [hasPermission]);
618
- const validateAccess = useCallback2(async (resource, action) => {
619
- return Promise.resolve(canAccess(resource, action));
620
- }, [canAccess]);
621
- const contextValue = useMemo2(() => ({
622
- permissions,
623
- roles,
624
- accessLevel,
625
- rbacLoading,
626
- rbacError,
627
- selectedEventId,
628
- appConfig,
629
- userEventAccess,
630
- eventAccessLoading,
631
- // Organisation context
632
- selectedOrganisationId,
633
- requireOrganisationContext: () => {
634
- if (!selectedOrganisationId) {
635
- throw new Error("Organisation context is required but not available");
636
- }
637
- return selectedOrganisationId;
638
- },
639
- hasPermission,
640
- hasAnyPermission,
641
- hasAllPermissions,
642
- hasRole,
643
- hasAccessLevel,
644
- canAccess,
645
- validatePermission,
646
- validateAccess,
647
- refreshPermissions,
648
- setSelectedEventId,
649
- // New RBAC system support
650
- rbacEnabled: enableRBAC,
651
- rbacContext: void 0,
652
- // Will be populated by useRBAC hook when enabled
653
- loadUserEventAccess,
654
- getUserEventAccess
655
- }), [
656
- permissions,
657
- roles,
658
- accessLevel,
659
- rbacLoading,
660
- rbacError,
661
- selectedEventId,
662
- appConfig,
663
- userEventAccess,
664
- eventAccessLoading,
665
- selectedOrganisationId,
666
- hasPermission,
667
- hasAnyPermission,
668
- hasAllPermissions,
669
- hasRole,
670
- hasAccessLevel,
671
- canAccess,
672
- validatePermission,
673
- validateAccess,
674
- refreshPermissions,
675
- setSelectedEventId,
676
- enableRBAC,
677
- loadUserEventAccess,
678
- getUserEventAccess
679
- ]);
680
- return /* @__PURE__ */ jsx2(RBACContext.Provider, { value: contextValue, children });
681
- }
682
- var RBACContext, useRBAC, STORAGE_KEYS, transformRBACPermissions;
683
- var init_RBACProvider = __esm({
684
- "src/providers/RBACProvider.tsx"() {
685
- "use strict";
686
- init_unified();
687
- init_debugLogger();
688
- RBACContext = createContext2(void 0);
689
- useRBAC = () => {
690
- const context = useContext2(RBACContext);
691
- if (!context) {
692
- throw new Error("useRBAC must be used within an RBACProvider");
693
- }
694
- return context;
695
- };
696
- STORAGE_KEYS = {
697
- SELECTED_EVENT: "pace-core-selected-event"
698
- };
699
- transformRBACPermissions = (rbacData, _appName) => {
700
- const permissions = {};
701
- let roles = [];
702
- let access_level = "viewer" /* VIEWER */;
703
- if (!rbacData || !Array.isArray(rbacData)) {
704
- return { permissions: {}, roles: ["viewer"], access_level: "viewer" /* VIEWER */ };
705
- }
706
- const superAdminPerm = rbacData.find((p) => p.permission_type === "all_permissions");
707
- if (superAdminPerm) {
708
- return {
709
- permissions: { "all:all": true },
710
- roles: ["super"],
711
- access_level: "super" /* SUPER */
712
- };
713
- }
714
- const eventAppPerms = rbacData.filter((p) => p.permission_type === "event_app_access");
715
- if (eventAppPerms.length > 0) {
716
- const role = eventAppPerms[0].role_name;
717
- switch (role) {
718
- case "event_admin":
719
- access_level = "admin" /* ADMIN */;
720
- roles = ["admin"];
721
- break;
722
- case "planner":
723
- access_level = "planner" /* PLANNER */;
724
- roles = ["planner"];
725
- break;
726
- case "participant":
727
- access_level = "participant" /* PARTICIPANT */;
728
- roles = ["participant"];
729
- break;
730
- case "editor":
731
- access_level = "editor" /* EDITOR */;
732
- roles = ["editor"];
733
- break;
734
- case "viewer":
735
- default:
736
- access_level = "viewer" /* VIEWER */;
737
- roles = ["viewer"];
738
- break;
739
- }
740
- const basePermissions = ["read"];
741
- if (["event_admin", "planner"].includes(role)) {
742
- basePermissions.push("create", "update");
743
- }
744
- if (role === "event_admin") {
745
- basePermissions.push("delete");
746
- }
747
- basePermissions.forEach((operation) => {
748
- permissions[`default:${operation}`] = true;
749
- });
750
- }
751
- const orgPerms = rbacData.filter((p) => p.permission_type === "organisation_access");
752
- if (orgPerms.length > 0) {
753
- const role = orgPerms[0].role_name;
754
- if (role === "org_admin") {
755
- access_level = "admin" /* ADMIN */;
756
- roles = ["admin"];
757
- ["create", "read", "update", "delete"].forEach((operation) => {
758
- permissions[`default:${operation}`] = true;
759
- });
760
- }
761
- }
762
- return { permissions, roles, access_level };
763
- };
764
- }
765
- });
766
-
767
- // src/hooks/useInactivityTracker.ts
768
- import { useState as useState3, useEffect as useEffect3, useCallback as useCallback3, useRef } from "react";
769
- function throttle(func, limit) {
770
- let inThrottle;
771
- return function(...args) {
772
- if (!inThrottle) {
773
- func.apply(this, args);
774
- inThrottle = true;
775
- setTimeout(() => inThrottle = false, limit);
776
- }
777
- };
778
- }
779
- function useInactivityTracker({
780
- idleTimeoutMs = 30 * 60 * 1e3,
781
- // 30 minutes
782
- warnBeforeMs = 60 * 1e3,
783
- // 1 minute
784
- onIdle,
785
- onWarning,
786
- onActivity,
787
- enabled = true,
788
- storageKey = "pace-core-inactivity",
789
- channelName = "pace-core-inactivity"
790
- } = {}) {
791
- const [isIdle, setIsIdle] = useState3(false);
792
- const [timeRemaining, setTimeRemaining] = useState3(idleTimeoutMs);
793
- const [showWarning, setShowWarning] = useState3(false);
794
- const [isTracking, setIsTracking] = useState3(false);
795
- useEffect3(() => {
796
- if (!enabled) {
797
- setIsTracking(false);
798
- setIsIdle(false);
799
- setShowWarning(false);
800
- setTimeRemaining(idleTimeoutMs);
801
- }
802
- }, [enabled, idleTimeoutMs]);
803
- const timeoutRef = useRef(null);
804
- const warningTimeoutRef = useRef(null);
805
- const countdownIntervalRef = useRef(null);
806
- const lastActivityRef = useRef(Date.now());
807
- const channelRef = useRef(null);
808
- const clearTimers = useCallback3(() => {
809
- if (timeoutRef.current) {
810
- clearTimeout(timeoutRef.current);
811
- timeoutRef.current = null;
812
- }
813
- if (warningTimeoutRef.current) {
814
- clearTimeout(warningTimeoutRef.current);
815
- warningTimeoutRef.current = null;
816
- }
817
- if (countdownIntervalRef.current) {
818
- clearInterval(countdownIntervalRef.current);
819
- countdownIntervalRef.current = null;
820
- }
821
- }, []);
822
- const resetActivity = useCallback3((skipActivityCallback = false) => {
823
- if (!enabled) return;
824
- const now = Date.now();
825
- lastActivityRef.current = now;
826
- clearTimers();
827
- setIsIdle(false);
828
- setShowWarning(false);
829
- setTimeRemaining(idleTimeoutMs);
830
- if (!skipActivityCallback) {
831
- onActivity?.();
832
- }
833
- const warningTime = idleTimeoutMs - warnBeforeMs;
834
- if (warningTime > 0) {
835
- warningTimeoutRef.current = setTimeout(() => {
836
- setShowWarning(true);
837
- onWarning?.();
838
- }, warningTime);
839
- }
840
- timeoutRef.current = setTimeout(() => {
841
- setIsIdle(true);
842
- onIdle?.();
843
- }, idleTimeoutMs);
844
- countdownIntervalRef.current = setInterval(() => {
845
- const elapsed = Date.now() - lastActivityRef.current;
846
- const remaining = Math.max(0, idleTimeoutMs - elapsed);
847
- setTimeRemaining(remaining);
848
- if (remaining === 0) {
849
- clearTimers();
850
- }
851
- }, 1e3);
852
- try {
853
- localStorage.setItem(storageKey, now.toString());
854
- } catch (error) {
855
- console.warn("[useInactivityTracker] Failed to persist activity time:", error);
856
- }
857
- try {
858
- if (channelRef.current) {
859
- channelRef.current.postMessage({ type: "activity", timestamp: now });
860
- }
861
- } catch (error) {
862
- console.warn("[useInactivityTracker] Failed to broadcast activity:", error);
863
- }
864
- }, [enabled, idleTimeoutMs, warnBeforeMs, onIdle, onWarning, onActivity, storageKey, clearTimers]);
865
- const startTracking = useCallback3(() => {
866
- if (!enabled) return;
867
- setIsTracking(false);
868
- setIsIdle(false);
869
- setShowWarning(false);
870
- setTimeRemaining(idleTimeoutMs);
871
- clearTimers();
872
- setIsTracking(true);
873
- try {
874
- if (typeof BroadcastChannel !== "undefined") {
875
- channelRef.current = new BroadcastChannel(channelName);
876
- channelRef.current.onmessage = (event) => {
877
- if (event.data.type === "activity") {
878
- lastActivityRef.current = event.data.timestamp;
879
- resetActivity();
880
- }
881
- };
882
- }
883
- } catch (error) {
884
- console.warn("[useInactivityTracker] Failed to set up cross-tab communication:", error);
885
- }
886
- try {
887
- const persistedTime = localStorage.getItem(storageKey);
888
- if (persistedTime) {
889
- const persistedTimestamp = parseInt(persistedTime, 10);
890
- const elapsed = Date.now() - persistedTimestamp;
891
- if (elapsed < idleTimeoutMs) {
892
- lastActivityRef.current = persistedTimestamp;
893
- const remaining = idleTimeoutMs - elapsed;
894
- setTimeRemaining(remaining);
895
- if (remaining <= warnBeforeMs) {
896
- setShowWarning(true);
897
- onWarning?.();
898
- }
899
- if (remaining <= 0) {
900
- setIsIdle(true);
901
- onIdle?.();
902
- return;
903
- }
904
- }
905
- }
906
- } catch (error) {
907
- console.warn("[useInactivityTracker] Failed to check persisted activity time:", error);
908
- }
909
- const throttledResetActivity = throttle((event) => {
910
- resetActivity();
911
- }, 100);
912
- const addEventListeners = () => {
913
- ACTIVITY_EVENTS.forEach((event) => {
914
- document.addEventListener(event, throttledResetActivity, { passive: true });
915
- });
916
- };
917
- const removeEventListeners = () => {
918
- ACTIVITY_EVENTS.forEach((event) => {
919
- document.removeEventListener(event, throttledResetActivity);
920
- });
921
- };
922
- addEventListeners();
923
- resetActivity(true);
924
- return () => {
925
- removeEventListeners();
926
- clearTimers();
927
- if (channelRef.current) {
928
- channelRef.current.close();
929
- channelRef.current = null;
930
- }
931
- };
932
- }, [enabled, isTracking, channelName, storageKey, idleTimeoutMs, warnBeforeMs, onIdle, onWarning]);
933
- const stopTracking = useCallback3(() => {
934
- setIsTracking(false);
935
- clearTimers();
936
- if (channelRef.current) {
937
- channelRef.current.close();
938
- channelRef.current = null;
939
- }
940
- }, [clearTimers]);
941
- useEffect3(() => {
942
- if (enabled) {
943
- const cleanup = startTracking();
944
- return cleanup;
945
- } else {
946
- stopTracking();
947
- }
948
- }, [enabled, idleTimeoutMs, warnBeforeMs]);
949
- useEffect3(() => {
950
- return () => {
951
- clearTimers();
952
- if (channelRef.current) {
953
- channelRef.current.close();
954
- }
955
- };
956
- }, [clearTimers]);
957
- return {
958
- isIdle,
959
- timeRemaining,
960
- showWarning,
961
- resetActivity,
962
- startTracking,
963
- stopTracking,
964
- isTracking
965
- };
966
- }
967
- var ACTIVITY_EVENTS;
968
- var init_useInactivityTracker = __esm({
969
- "src/hooks/useInactivityTracker.ts"() {
970
- "use strict";
971
- ACTIVITY_EVENTS = [
972
- "mousedown",
973
- "mousemove",
974
- "mouseup",
975
- "click",
976
- "scroll",
977
- "wheel",
978
- "touchstart",
979
- "touchmove",
980
- "touchend",
981
- "keydown",
982
- "keyup",
983
- "keypress",
984
- "focus",
985
- "blur",
986
- "visibilitychange"
987
- ];
988
- }
989
- });
990
-
991
- // src/components/InactivityWarningModal/InactivityWarningModal.tsx
992
- import { useEffect as useEffect4, useState as useState4, useCallback as useCallback4 } from "react";
993
- import { Clock, AlertTriangle } from "lucide-react";
994
- import { jsx as jsx3, jsxs } from "react/jsx-runtime";
995
- function InactivityWarningModal({
996
- isOpen,
997
- timeRemaining,
998
- onStaySignedIn,
999
- onSignOutNow,
1000
- title = "Session Timeout Warning",
1001
- description = "You've been inactive for a while. Your session will expire soon for security reasons.",
1002
- className
1003
- }) {
1004
- const [displayTime, setDisplayTime] = useState4(timeRemaining);
1005
- useEffect4(() => {
1006
- setDisplayTime(timeRemaining);
1007
- }, [timeRemaining]);
1008
- const formatTime = useCallback4((seconds) => {
1009
- const mins = Math.floor(seconds / 60);
1010
- const secs = seconds % 60;
1011
- return `${mins.toString().padStart(2, "0")}:${secs.toString().padStart(2, "0")}`;
1012
- }, []);
1013
- return /* @__PURE__ */ jsx3(Dialog, { open: isOpen, onOpenChange: (open) => !open && onStaySignedIn(), children: /* @__PURE__ */ jsxs(
1014
- DialogContent,
1015
- {
1016
- className: `sm:max-w-md ${className || ""}`,
1017
- preventCloseOnEscape: false,
1018
- preventCloseOnOutsideClick: true,
1019
- "data-testid": "inactivity-warning-modal",
1020
- children: [
1021
- /* @__PURE__ */ jsxs(DialogHeader, { children: [
1022
- /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-3", children: [
1023
- /* @__PURE__ */ jsx3("div", { className: "flex-shrink-0", children: /* @__PURE__ */ jsx3(AlertTriangle, { className: "h-6 w-6 text-acc-600" }) }),
1024
- /* @__PURE__ */ jsx3("div", { children: /* @__PURE__ */ jsx3(DialogTitle, { className: "text-lg font-semibold text-main-900", children: title }) })
1025
- ] }),
1026
- /* @__PURE__ */ jsx3(DialogDescription, { className: "text-main-700 mt-2", children: description })
1027
- ] }),
1028
- /* @__PURE__ */ jsxs("div", { className: "space-y-6", children: [
1029
- /* @__PURE__ */ jsxs("div", { className: "text-center", children: [
1030
- /* @__PURE__ */ jsxs("div", { className: "inline-flex items-center gap-2 px-4 py-3 bg-acc-50 border border-acc-200 rounded-lg", children: [
1031
- /* @__PURE__ */ jsx3(Clock, { className: "h-5 w-5 text-acc-600" }),
1032
- /* @__PURE__ */ jsx3("span", { className: "text-2xl font-mono font-bold text-acc-700", children: formatTime(displayTime) })
1033
- ] }),
1034
- /* @__PURE__ */ jsx3("p", { className: "text-sm text-main-600 mt-2", children: "Time remaining before automatic logout" })
1035
- ] }),
1036
- /* @__PURE__ */ jsxs("div", { className: "flex flex-col sm:flex-row gap-3", children: [
1037
- /* @__PURE__ */ jsx3(
1038
- Button,
1039
- {
1040
- onClick: onStaySignedIn,
1041
- className: "flex-1 bg-main-600 hover:bg-main-700 text-main-50",
1042
- size: "lg",
1043
- children: "Stay Signed In"
1044
- }
1045
- ),
1046
- /* @__PURE__ */ jsx3(
1047
- Button,
1048
- {
1049
- onClick: onSignOutNow,
1050
- variant: "outline",
1051
- className: "flex-1 border-acc-300 text-acc-700 hover:bg-acc-50",
1052
- size: "lg",
1053
- children: "Sign Out Now"
1054
- }
1055
- )
1056
- ] }),
1057
- /* @__PURE__ */ jsx3("div", { className: "text-xs text-main-500 text-center", children: /* @__PURE__ */ jsx3("p", { children: "For security reasons, you'll be automatically signed out after 30 minutes of inactivity." }) })
1058
- ] })
1059
- ]
1060
- }
1061
- ) });
1062
- }
1063
- var init_InactivityWarningModal = __esm({
1064
- "src/components/InactivityWarningModal/InactivityWarningModal.tsx"() {
1065
- "use strict";
1066
- init_Dialog();
1067
- init_Button();
1068
- }
1069
- });
1070
-
1071
- // src/providers/InactivityProvider.tsx
1072
- import { createContext as createContext3, useContext as useContext3, useState as useState5, useEffect as useEffect5, useCallback as useCallback5, useMemo as useMemo3 } from "react";
1073
- import { jsx as jsx4, jsxs as jsxs2 } from "react/jsx-runtime";
1074
- function InactivityProvider({
1075
- children,
1076
- user,
1077
- session,
1078
- supabaseClient,
1079
- idleTimeoutMs = 30 * 60 * 1e3,
1080
- // 30 minutes
1081
- warnBeforeMs = 60 * 1e3,
1082
- // 60 seconds
1083
- onIdleLogout,
1084
- renderInactivityWarning,
1085
- dangerouslyDisableInactivity = false
1086
- }) {
1087
- const [showInactivityWarning, setShowInactivityWarning] = useState5(false);
1088
- const [inactivityTimeRemaining, setInactivityTimeRemaining] = useState5(0);
1089
- useEffect5(() => {
1090
- if (typeof window !== "undefined") {
1091
- const isProduction = true;
1092
- if (isProduction && dangerouslyDisableInactivity) {
1093
- console.error("[InactivityProvider] CRITICAL: dangerouslyDisableInactivity is not allowed in production! Auto-enabling inactivity feature.");
1094
- }
1095
- if (!isProduction && dangerouslyDisableInactivity) {
1096
- console.warn("[InactivityProvider] Inactivity feature disabled for development. This will NOT work in production.");
1097
- }
1098
- }
1099
- }, [dangerouslyDisableInactivity]);
1100
- const isInactivityEnabled = typeof window !== "undefined" && (false ? !dangerouslyDisableInactivity : true);
1101
- const {
1102
- isIdle,
1103
- timeRemaining,
1104
- showWarning,
1105
- resetActivity,
1106
- startTracking,
1107
- stopTracking,
1108
- isTracking
1109
- } = useInactivityTracker({
1110
- idleTimeoutMs,
1111
- warnBeforeMs,
1112
- enabled: isInactivityEnabled && !!user && !!session,
1113
- onIdle: useCallback5(() => {
1114
- setShowInactivityWarning(false);
1115
- setInactivityTimeRemaining(0);
1116
- if (supabaseClient) {
1117
- supabaseClient.auth.signOut().catch((error) => {
1118
- console.error("[InactivityProvider] Error during idle logout:", error);
1119
- });
1120
- }
1121
- onIdleLogout?.("inactivity");
1122
- }, [supabaseClient, onIdleLogout]),
1123
- onWarning: useCallback5(() => {
1124
- setShowInactivityWarning(true);
1125
- setInactivityTimeRemaining(warnBeforeMs);
1126
- }, [warnBeforeMs]),
1127
- onActivity: useCallback5(() => {
1128
- setShowInactivityWarning(false);
1129
- setInactivityTimeRemaining(0);
1130
- }, [])
1131
- });
1132
- const handleIdleLogout = useCallback5(async () => {
1133
- setShowInactivityWarning(false);
1134
- setInactivityTimeRemaining(0);
1135
- stopTracking();
1136
- try {
1137
- if (supabaseClient) {
1138
- await supabaseClient.auth.signOut();
1139
- }
1140
- } catch (error) {
1141
- console.error("[InactivityProvider] Error during idle logout:", error);
1142
- }
1143
- onIdleLogout?.("inactivity");
1144
- }, [supabaseClient, onIdleLogout, stopTracking]);
1145
- const handleStaySignedIn = useCallback5(() => {
1146
- setShowInactivityWarning(false);
1147
- setInactivityTimeRemaining(0);
1148
- resetActivity();
1149
- }, [resetActivity]);
1150
- const handleSignOutNow = useCallback5(async () => {
1151
- setShowInactivityWarning(false);
1152
- setInactivityTimeRemaining(0);
1153
- stopTracking();
1154
- try {
1155
- if (supabaseClient) {
1156
- await supabaseClient.auth.signOut();
1157
- }
1158
- } catch (error) {
1159
- console.error("[InactivityProvider] Error during manual sign out:", error);
1160
- }
1161
- onIdleLogout?.("inactivity");
1162
- }, [supabaseClient, onIdleLogout, stopTracking]);
1163
- useEffect5(() => {
1164
- if (showWarning && timeRemaining > 0) {
1165
- setInactivityTimeRemaining(Math.ceil(timeRemaining / 1e3));
1166
- }
1167
- }, [showWarning, timeRemaining]);
1168
- const contextValue = useMemo3(() => ({
1169
- showInactivityWarning,
1170
- inactivityTimeRemaining,
1171
- isIdle,
1172
- timeRemaining,
1173
- showWarning,
1174
- isTracking,
1175
- resetActivity,
1176
- startTracking,
1177
- stopTracking,
1178
- handleIdleLogout,
1179
- handleStaySignedIn,
1180
- handleSignOutNow
1181
- }), [
1182
- showInactivityWarning,
1183
- inactivityTimeRemaining,
1184
- isIdle,
1185
- timeRemaining,
1186
- showWarning,
1187
- isTracking,
1188
- resetActivity,
1189
- startTracking,
1190
- stopTracking,
1191
- handleIdleLogout,
1192
- handleStaySignedIn,
1193
- handleSignOutNow
1194
- ]);
1195
- return /* @__PURE__ */ jsxs2(InactivityContext.Provider, { value: contextValue, children: [
1196
- children,
1197
- showInactivityWarning && (renderInactivityWarning ? renderInactivityWarning({
1198
- timeRemaining: inactivityTimeRemaining,
1199
- onStaySignedIn: handleStaySignedIn,
1200
- onSignOutNow: handleSignOutNow
1201
- }) : /* @__PURE__ */ jsx4(
1202
- InactivityWarningModal,
1203
- {
1204
- isOpen: showInactivityWarning,
1205
- timeRemaining: inactivityTimeRemaining,
1206
- onStaySignedIn: handleStaySignedIn,
1207
- onSignOutNow: handleSignOutNow
1208
- }
1209
- ))
1210
- ] });
1211
- }
1212
- var InactivityContext, useInactivity;
1213
- var init_InactivityProvider = __esm({
1214
- "src/providers/InactivityProvider.tsx"() {
1215
- "use strict";
1216
- init_useInactivityTracker();
1217
- init_InactivityWarningModal();
1218
- InactivityContext = createContext3(void 0);
1219
- useInactivity = () => {
1220
- const context = useContext3(InactivityContext);
1221
- if (!context) {
1222
- throw new Error("useInactivity must be used within an InactivityProvider");
1223
- }
1224
- return context;
1225
- };
1226
- }
1227
- });
1228
-
1229
- // src/providers/UnifiedAuthProvider.tsx
1230
- var UnifiedAuthProvider_exports = {};
1231
- __export(UnifiedAuthProvider_exports, {
1232
- UnifiedAuthProvider: () => UnifiedAuthProvider,
1233
- useUnifiedAuth: () => useUnifiedAuth
1234
- });
1235
- import { createContext as createContext4, useContext as useContext4, useMemo as useMemo4 } from "react";
1236
- import { jsx as jsx5 } from "react/jsx-runtime";
1237
- function UnifiedAuthContextProvider({
1238
- children,
1239
- appName,
1240
- ...props
1241
- }) {
1242
- const auth = useAuth();
1243
- const rbac = useRBAC();
1244
- const inactivity = useInactivity();
1245
- const contextValue = useMemo4(() => ({
1246
- ...auth,
1247
- ...rbac,
1248
- ...inactivity,
1249
- appName,
1250
- isLoading: auth.authLoading || rbac.rbacLoading,
1251
- hasErrors: !!auth.authError || !!rbac.rbacError
1252
- }), [auth, rbac, inactivity, appName]);
1253
- return /* @__PURE__ */ jsx5(UnifiedAuthContext.Provider, { value: contextValue, children });
1254
- }
1255
- function AuthAwareProviders({
1256
- children,
1257
- supabaseClient,
1258
- appName,
1259
- persistState,
1260
- enablePersistence,
1261
- requireOrganisationContext,
1262
- enableRBAC,
1263
- idleTimeoutMs,
1264
- warnBeforeMs,
1265
- onIdleLogout,
1266
- renderInactivityWarning,
1267
- dangerouslyDisableInactivity
1268
- }) {
1269
- const auth = useAuth();
1270
- return /* @__PURE__ */ jsx5(
1271
- RBACProvider,
1272
- {
1273
- supabaseClient,
1274
- user: auth.user,
1275
- session: auth.session,
1276
- appName,
1277
- enableRBAC,
1278
- persistState,
1279
- enablePersistence,
1280
- requireOrganisationContext,
1281
- children: /* @__PURE__ */ jsx5(
1282
- InactivityProvider,
1283
- {
1284
- user: auth.user,
1285
- session: auth.session,
1286
- supabaseClient,
1287
- idleTimeoutMs,
1288
- warnBeforeMs,
1289
- onIdleLogout,
1290
- renderInactivityWarning,
1291
- dangerouslyDisableInactivity,
1292
- children: /* @__PURE__ */ jsx5(
1293
- UnifiedAuthContextProvider,
1294
- {
1295
- appName,
1296
- supabaseClient,
1297
- persistState,
1298
- enablePersistence,
1299
- requireOrganisationContext,
1300
- enableRBAC,
1301
- idleTimeoutMs,
1302
- warnBeforeMs,
1303
- onIdleLogout,
1304
- renderInactivityWarning,
1305
- dangerouslyDisableInactivity,
1306
- children
1307
- }
1308
- )
1309
- }
1310
- )
1311
- }
1312
- );
1313
- }
1314
- function UnifiedAuthProvider({
1315
- children,
1316
- supabaseClient,
1317
- appName,
1318
- persistState = true,
1319
- enablePersistence,
1320
- requireOrganisationContext = true,
1321
- enableRBAC = false,
1322
- idleTimeoutMs = 30 * 60 * 1e3,
1323
- // 30 minutes
1324
- warnBeforeMs = 60 * 1e3,
1325
- // 60 seconds
1326
- onIdleLogout,
1327
- renderInactivityWarning,
1328
- dangerouslyDisableInactivity = false
1329
- }) {
1330
- return /* @__PURE__ */ jsx5(AuthProvider, { supabaseClient, children: /* @__PURE__ */ jsx5(
1331
- AuthAwareProviders,
1332
- {
1333
- supabaseClient,
1334
- appName,
1335
- persistState,
1336
- enablePersistence,
1337
- requireOrganisationContext,
1338
- enableRBAC,
1339
- idleTimeoutMs,
1340
- warnBeforeMs,
1341
- onIdleLogout,
1342
- renderInactivityWarning,
1343
- dangerouslyDisableInactivity,
1344
- children
1345
- }
1346
- ) });
1347
- }
1348
- var UnifiedAuthContext, useUnifiedAuth;
1349
- var init_UnifiedAuthProvider = __esm({
1350
- "src/providers/UnifiedAuthProvider.tsx"() {
1351
- "use strict";
1352
- init_AuthProvider();
1353
- init_RBACProvider();
1354
- init_InactivityProvider();
1355
- UnifiedAuthContext = createContext4(void 0);
1356
- useUnifiedAuth = () => {
1357
- const context = useContext4(UnifiedAuthContext);
1358
- if (!context) {
1359
- throw new Error("useUnifiedAuth must be used within a UnifiedAuthProvider");
1360
- }
1361
- return context;
1362
- };
1363
- }
1364
- });
1365
-
1366
- // src/providers/OrganisationProvider.tsx
1367
- var OrganisationProvider_exports = {};
1368
- __export(OrganisationProvider_exports, {
1369
- OrganisationProvider: () => OrganisationProvider,
1370
- useOrganisations: () => useOrganisations
1371
- });
1372
- import { createContext as createContext5, useContext as useContext5, useState as useState6, useEffect as useEffect6, useCallback as useCallback6, useMemo as useMemo5 } from "react";
1373
- import { useNavigate } from "react-router-dom";
1374
- import { jsx as jsx6, jsxs as jsxs3 } from "react/jsx-runtime";
1375
- function OrganisationProvider({ children }) {
1376
- const [selectedOrganisation, setSelectedOrganisation] = useState6(null);
1377
- const [organisations, setOrganisations] = useState6([]);
1378
- const [userMemberships, setUserMemberships] = useState6([]);
1379
- const [roleMapState, setRoleMapState] = useState6(/* @__PURE__ */ new Map());
1380
- const [isLoading, setIsLoading] = useState6(true);
1381
- const [error, setError] = useState6(null);
1382
- const [isContextReady, setIsContextReady] = useState6(false);
1383
- const { user, session, supabase, signOut } = useUnifiedAuth();
1384
- let navigate = null;
1385
- try {
1386
- navigate = useNavigate();
1387
- } catch (error2) {
1388
- navigate = null;
1389
- }
1390
- const setDatabaseOrganisationContext = useCallback6(async (organisation) => {
1391
- if (!supabase || !session) {
1392
- console.warn("[OrganisationProvider] No Supabase client or session available for setting organisation context");
1393
- setIsContextReady(false);
1394
- return;
1395
- }
1396
- try {
1397
- await setOrganisationContext(supabase, organisation.id);
1398
- DebugLogger.log("OrganisationProvider", "Database organisation context set to:", organisation.display_name);
1399
- setIsContextReady(true);
1400
- } catch (error2) {
1401
- console.error("[OrganisationProvider] Failed to set database organisation context:", error2);
1402
- setIsContextReady(false);
1403
- }
1404
- }, [supabase, session]);
1405
- useEffect6(() => {
1406
- if (selectedOrganisation && supabase && session) {
1407
- setIsContextReady(false);
1408
- (async () => {
1409
- await setDatabaseOrganisationContext(selectedOrganisation);
1410
- })();
1411
- } else {
1412
- setIsContextReady(false);
1413
- }
1414
- }, [selectedOrganisation, setDatabaseOrganisationContext, supabase, session]);
1415
- const loadUserOrganisations = useCallback6(async () => {
1416
- if (!user || !session || !supabase) {
1417
- DebugLogger.log("OrganisationProvider", "Clearing organisation state - no user, session, or supabase client");
1418
- setSelectedOrganisation(null);
1419
- setOrganisations([]);
1420
- setUserMemberships([]);
1421
- setIsLoading(false);
1422
- setError(null);
1423
- return;
1424
- }
1425
- setIsLoading(true);
1426
- setError(null);
1427
- try {
1428
- DebugLogger.log("OrganisationProvider", "Loading organisations for user:", user.id);
1429
- let memberships, membershipError;
1430
- try {
1431
- const result = await supabase.from("rbac_organisation_roles").select(`
1432
- id,
1433
- user_id,
1434
- organisation_id,
1435
- role,
1436
- status,
1437
- granted_at,
1438
- granted_by,
1439
- revoked_at,
1440
- revoked_by,
1441
- notes,
1442
- created_at,
1443
- updated_at
1444
- `).eq("user_id", user.id).eq("status", "active").is("revoked_at", null).in("role", ["org_admin", "leader", "member"]);
1445
- memberships = result.data;
1446
- membershipError = result.error;
1447
- } catch (queryError) {
1448
- membershipError = queryError;
1449
- }
1450
- if (membershipError) {
1451
- console.error("[OrganisationProvider] Error loading memberships:", membershipError);
1452
- throw membershipError;
1453
- }
1454
- DebugLogger.log("OrganisationProvider", "Raw memberships data:", memberships);
1455
- if (!memberships || memberships.length === 0) {
1456
- throw new Error("User has no active organisation memberships");
1457
- }
1458
- const organisationIds = memberships.map((m) => m.organisation_id);
1459
- const { data: organisations2, error: orgError } = await supabase.from("organisations").select("id, name, display_name, subscription_tier, settings, is_active, parent_id, created_at, updated_at").in("id", organisationIds);
1460
- if (orgError) {
1461
- console.error("[OrganisationProvider] Error loading organisations:", orgError);
1462
- throw orgError;
1463
- }
1464
- const roleMap = /* @__PURE__ */ new Map();
1465
- memberships?.forEach((membership) => {
1466
- roleMap.set(membership.organisation_id, membership.role);
1467
- });
1468
- const orgs = organisations2;
1469
- const activeOrgs = orgs.filter((org) => org.is_active);
1470
- if (activeOrgs.length === 0) {
1471
- throw new Error("User has no access to active organisations");
1472
- }
1473
- DebugLogger.log("OrganisationProvider", "Active organisations:", activeOrgs);
1474
- setOrganisations(activeOrgs);
1475
- setUserMemberships(memberships);
1476
- setRoleMapState(roleMap);
1477
- let initialOrg = null;
1478
- try {
1479
- const persistedOrgString = localStorage.getItem(STORAGE_KEYS2.SELECTED_ORGANISATION);
1480
- if (persistedOrgString) {
1481
- const persistedOrg = JSON.parse(persistedOrgString);
1482
- const validPersistedOrg = activeOrgs.find((org) => org.id === persistedOrg.id);
1483
- if (validPersistedOrg) {
1484
- initialOrg = validPersistedOrg;
1485
- DebugLogger.log("OrganisationProvider", "Restored persisted organisation:", initialOrg.display_name);
1486
- }
1487
- }
1488
- } catch (storageError) {
1489
- console.warn("[OrganisationProvider] Failed to restore persisted organisation:", storageError);
1490
- }
1491
- if (!initialOrg) {
1492
- const adminMembership = memberships.find((m) => m.role === "org_admin");
1493
- if (adminMembership) {
1494
- const foundOrg = organisations2.find((org) => org.id === adminMembership.organisation_id);
1495
- if (foundOrg) {
1496
- initialOrg = foundOrg;
1497
- DebugLogger.log("OrganisationProvider", "Selected org_admin organisation:", initialOrg.display_name);
1498
- }
1499
- }
1500
- }
1501
- if (!initialOrg) {
1502
- initialOrg = activeOrgs[0];
1503
- DebugLogger.log("OrganisationProvider", "Selected first organisation:", initialOrg.display_name);
1504
- }
1505
- if (!initialOrg) {
1506
- throw new Error("No valid organisation found for user");
1507
- }
1508
- setSelectedOrganisation(initialOrg);
1509
- localStorage.setItem(STORAGE_KEYS2.SELECTED_ORGANISATION, JSON.stringify(initialOrg));
1510
- DebugLogger.log("OrganisationProvider", "Organisation context established:", {
1511
- selectedOrganisation: initialOrg.display_name,
1512
- totalOrganisations: activeOrgs.length,
1513
- userRole: roleMap.get(initialOrg.id)
1514
- });
1515
- } catch (err) {
1516
- console.error("[OrganisationProvider] Failed to load organisations:", err);
1517
- setError(err);
1518
- setSelectedOrganisation(null);
1519
- setOrganisations([]);
1520
- setUserMemberships([]);
1521
- localStorage.removeItem(STORAGE_KEYS2.SELECTED_ORGANISATION);
1522
- } finally {
1523
- setIsLoading(false);
1524
- }
1525
- }, [user, session, supabase]);
1526
- useEffect6(() => {
1527
- loadUserOrganisations();
1528
- }, [loadUserOrganisations]);
1529
- const handleLogoutAndRedirect = useCallback6(async () => {
1530
- try {
1531
- await signOut();
1532
- if (navigate) {
1533
- navigate("/login", { replace: true });
1534
- } else {
1535
- window.location.href = "/login";
1536
- }
1537
- } catch (error2) {
1538
- console.error("[OrganisationProvider] Error during logout:", error2);
1539
- if (navigate) {
1540
- navigate("/login", { replace: true });
1541
- } else {
1542
- window.location.href = "/login";
1543
- }
1544
- }
1545
- }, [signOut, navigate]);
1546
- const ensureOrganisationContext = useCallback6(() => {
1547
- if (!selectedOrganisation) {
1548
- throw new Error("Organisation context is required but not available");
1549
- }
1550
- return selectedOrganisation;
1551
- }, [selectedOrganisation]);
1552
- const getUserRole = useCallback6((orgId) => {
1553
- const targetOrgId = orgId || selectedOrganisation?.id;
1554
- if (!targetOrgId) return "no_access";
1555
- return roleMapState.get(targetOrgId) || "no_access";
1556
- }, [roleMapState, selectedOrganisation]);
1557
- const validateOrganisationAccess = useCallback6((orgId) => {
1558
- return userMemberships.some(
1559
- (m) => m.organisation_id === orgId && m.status === "active" && m.revoked_at === null
1560
- );
1561
- }, [userMemberships]);
1562
- const switchOrganisation = useCallback6(async (orgId) => {
1563
- DebugLogger.log("OrganisationProvider", "Switching to organisation:", orgId);
1564
- if (!validateOrganisationAccess(orgId)) {
1565
- throw new Error(`User does not have access to organisation ${orgId}`);
1566
- }
1567
- const targetOrg = organisations.find((org) => org.id === orgId);
1568
- if (!targetOrg) {
1569
- throw new Error(`Organisation ${orgId} not found in user's organisations`);
1570
- }
1571
- setSelectedOrganisation(targetOrg);
1572
- localStorage.setItem(STORAGE_KEYS2.SELECTED_ORGANISATION, JSON.stringify(targetOrg));
1573
- await setDatabaseOrganisationContext(targetOrg);
1574
- DebugLogger.log("OrganisationProvider", "Switched to organisation:", targetOrg.display_name);
1575
- }, [organisations, validateOrganisationAccess, setDatabaseOrganisationContext]);
1576
- const refreshOrganisations = useCallback6(async () => {
1577
- if (!user || !session || !supabase) return;
1578
- setIsLoading(true);
1579
- }, [user, session, supabase]);
1580
- const getPrimaryOrganisation = useCallback6(() => {
1581
- const rolePriority = ["org_admin", "leader", "member"];
1582
- for (const role of rolePriority) {
1583
- const membership = userMemberships.find((m) => m.role === role);
1584
- if (membership) {
1585
- return organisations.find((org) => org.id === membership.organisation_id) || null;
1586
- }
1587
- }
1588
- return null;
1589
- }, [userMemberships, organisations]);
1590
- const isOrganisationSecure = useCallback6(() => {
1591
- return !!(selectedOrganisation && user);
1592
- }, [selectedOrganisation, user]);
1593
- const buildOrganisationHierarchy = useCallback6((orgs) => {
1594
- const orgMap = /* @__PURE__ */ new Map();
1595
- orgs.forEach((org) => orgMap.set(org.id, org));
1596
- const roots = [];
1597
- orgs.forEach((org) => {
1598
- if (!org.parent_id) {
1599
- roots.push({
1600
- organisation: org,
1601
- children: [],
1602
- depth: 0
1603
- });
1604
- }
1605
- });
1606
- return roots;
1607
- }, []);
1608
- const hasValidOrganisationContext = useMemo5(() => {
1609
- return !!(selectedOrganisation && !isLoading && !error && isContextReady);
1610
- }, [selectedOrganisation, isLoading, error, isContextReady]);
1611
- const contextValue = useMemo5(() => {
1612
- if (!selectedOrganisation) {
1613
- const placeholderOrg = {
1614
- id: "",
1615
- name: "",
1616
- display_name: "",
1617
- subscription_tier: "standard",
1618
- settings: {},
1619
- is_active: false,
1620
- created_at: "",
1621
- updated_at: ""
1622
- };
1623
- return {
1624
- selectedOrganisation: placeholderOrg,
1625
- organisations: [],
1626
- userMemberships: [],
1627
- isLoading,
1628
- error,
1629
- hasValidOrganisationContext: false,
1630
- setSelectedOrganisation: () => {
1631
- },
1632
- switchOrganisation: async () => {
1633
- },
1634
- getUserRole: () => "no_access",
1635
- validateOrganisationAccess: () => false,
1636
- refreshOrganisations: async () => {
1637
- },
1638
- ensureOrganisationContext: () => {
1639
- throw new Error("No organisation context");
1640
- },
1641
- isOrganisationSecure: () => false,
1642
- getPrimaryOrganisation: () => null
1643
- };
1644
- }
1645
- return {
1646
- selectedOrganisation,
1647
- organisations,
1648
- userMemberships,
1649
- isLoading,
1650
- error,
1651
- hasValidOrganisationContext,
1652
- setSelectedOrganisation,
1653
- switchOrganisation,
1654
- getUserRole,
1655
- validateOrganisationAccess,
1656
- refreshOrganisations,
1657
- ensureOrganisationContext,
1658
- isOrganisationSecure,
1659
- getPrimaryOrganisation
1660
- };
1661
- }, [
1662
- selectedOrganisation,
1663
- organisations,
1664
- userMemberships,
1665
- isLoading,
1666
- error,
1667
- hasValidOrganisationContext,
1668
- switchOrganisation,
1669
- getUserRole,
1670
- validateOrganisationAccess,
1671
- refreshOrganisations,
1672
- ensureOrganisationContext,
1673
- isOrganisationSecure,
1674
- getPrimaryOrganisation
1675
- ]);
1676
- if (isLoading || selectedOrganisation && !isContextReady) {
1677
- return /* @__PURE__ */ jsx6("div", { className: "organisation-loading", role: "status", "aria-label": "Loading organisation context", children: /* @__PURE__ */ jsx6("div", { className: "flex items-center justify-center min-h-screen", children: /* @__PURE__ */ jsxs3("div", { className: "text-center", children: [
1678
- /* @__PURE__ */ jsx6("div", { className: "animate-spin rounded-full h-8 w-8 border-b-2 border-primary mx-auto mb-4" }),
1679
- /* @__PURE__ */ jsx6("p", { className: "text-muted-foreground", children: isLoading ? "Loading organisation context..." : "Setting up organisation context..." })
1680
- ] }) }) });
1681
- }
1682
- if (error || user && !selectedOrganisation) {
1683
- return /* @__PURE__ */ jsx6("div", { className: "organisation-error", role: "alert", children: /* @__PURE__ */ jsx6("div", { className: "flex items-center justify-center min-h-screen", children: /* @__PURE__ */ jsxs3("div", { className: "text-center max-w-md mx-auto p-6", children: [
1684
- /* @__PURE__ */ jsx6("div", { className: "text-destructive mb-4", children: /* @__PURE__ */ jsx6("svg", { className: "h-12 w-12 mx-auto", fill: "none", viewBox: "0 0 24 24", stroke: "currentColor", children: /* @__PURE__ */ jsx6("path", { strokeLinecap: "round", strokeLinejoin: "round", strokeWidth: 2, d: "M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-2.5L13.732 4c-.77-.833-1.732-.833-2.464 0L4.35 16.5c-.77.833.192 2.5 1.732 2.5z" }) }) }),
1685
- /* @__PURE__ */ jsx6("h2", { className: "text-xl font-semibold text-foreground mb-2", children: "Organisation Access Required" }),
1686
- /* @__PURE__ */ jsx6("p", { className: "text-muted-foreground mb-4", children: error?.message || "No valid organisation context available. Please contact your administrator to be added to an organisation." }),
1687
- /* @__PURE__ */ jsx6(
1688
- "button",
1689
- {
1690
- onClick: handleLogoutAndRedirect,
1691
- className: "px-4 py-2 bg-primary text-primary-foreground rounded-md hover:bg-primary/90",
1692
- children: "Sign Out"
1693
- }
1694
- )
1695
- ] }) }) });
1696
- }
1697
- return /* @__PURE__ */ jsx6(OrganisationContext.Provider, { value: contextValue, children });
1698
- }
1699
- var OrganisationContext, STORAGE_KEYS2, useOrganisations;
1700
- var init_OrganisationProvider = __esm({
1701
- "src/providers/OrganisationProvider.tsx"() {
1702
- "use strict";
1703
- init_UnifiedAuthProvider();
1704
- init_organisationContext();
1705
- init_debugLogger();
1706
- OrganisationContext = createContext5(void 0);
1707
- STORAGE_KEYS2 = {
1708
- SELECTED_ORGANISATION: "pace-core-selected-organisation",
1709
- ORGANISATION_CONTEXT: "pace-core-organisation-context"
1710
- };
1711
- useOrganisations = () => {
1712
- const context = useContext5(OrganisationContext);
1713
- if (!context) {
1714
- throw new Error("useOrganisations must be used within an OrganisationProvider");
1715
- }
1716
- return context;
1717
- };
1718
- }
1719
- });
1720
-
1721
- export {
1722
- useAuth,
1723
- AuthProvider,
1724
- init_AuthProvider,
1725
- useRBAC,
1726
- RBACProvider,
1727
- init_RBACProvider,
1728
- useInactivityTracker,
1729
- init_useInactivityTracker,
1730
- InactivityWarningModal,
1731
- init_InactivityWarningModal,
1732
- useInactivity,
1733
- InactivityProvider,
1734
- init_InactivityProvider,
1735
- useUnifiedAuth,
1736
- UnifiedAuthProvider,
1737
- UnifiedAuthProvider_exports,
1738
- init_UnifiedAuthProvider,
1739
- OrganisationProvider,
1740
- useOrganisations,
1741
- OrganisationProvider_exports,
1742
- init_OrganisationProvider
1743
- };
1744
- //# sourceMappingURL=chunk-BEZRLNK3.js.map