@jmruthers/pace-core 0.5.136 → 0.5.139

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 (292) hide show
  1. package/dist/{DataTable-CYOHOX3O.js → DataTable-JXFCA2BJ.js} +10 -9
  2. package/dist/{EventLogo-801uofbR.d.ts → EventLogo-rFL_kRjk.d.ts} +73 -1
  3. package/dist/{UnifiedAuthProvider-5E5TUNMS.js → UnifiedAuthProvider-XIQQ7LVU.js} +4 -5
  4. package/dist/{chunk-YLKIDTUK.js → chunk-22WKWKRX.js} +4 -4
  5. package/dist/{chunk-TVYPTYOY.js → chunk-4C7EXCAR.js} +60 -24
  6. package/dist/chunk-4C7EXCAR.js.map +1 -0
  7. package/dist/{chunk-NOHEVYVX.js → chunk-5JMOHWDI.js} +417 -319
  8. package/dist/chunk-5JMOHWDI.js.map +1 -0
  9. package/dist/{chunk-FHWWBIHA.js → chunk-6DXZ6V5Q.js} +5 -5
  10. package/dist/{chunk-2TWNJ46Y.js → chunk-6LAAY47Q.js} +2 -2
  11. package/dist/{chunk-444EZN6N.js → chunk-7QCC6MCP.js} +88 -1
  12. package/dist/chunk-7QCC6MCP.js.map +1 -0
  13. package/dist/chunk-BJPBT3CU.js +21 -0
  14. package/dist/chunk-BJPBT3CU.js.map +1 -0
  15. package/dist/{chunk-L6PGMCMD.js → chunk-BOOI7GK2.js} +38 -12
  16. package/dist/chunk-BOOI7GK2.js.map +1 -0
  17. package/dist/{chunk-XARJS7CD.js → chunk-INQLMHPF.js} +2 -2
  18. package/dist/chunk-JISYG63F.js +70 -0
  19. package/dist/chunk-JISYG63F.js.map +1 -0
  20. package/dist/{chunk-SL2YQDR6.js → chunk-MA6EPSGZ.js} +2 -2
  21. package/dist/{chunk-5DPZ5EAT.js → chunk-OWAG3GSU.js} +1 -3
  22. package/dist/{chunk-LTV3XIJJ.js → chunk-T6JN6LH6.js} +4 -4
  23. package/dist/{chunk-HJGGOMQ6.js → chunk-TLT2ZR3L.js} +147 -103
  24. package/dist/chunk-TLT2ZR3L.js.map +1 -0
  25. package/dist/{chunk-4MT5BGGL.js → chunk-YCWDTTUK.js} +4 -6
  26. package/dist/{chunk-4MT5BGGL.js.map → chunk-YCWDTTUK.js.map} +1 -1
  27. package/dist/components.d.ts +1 -1
  28. package/dist/components.js +12 -11
  29. package/dist/components.js.map +1 -1
  30. package/dist/hooks.js +8 -9
  31. package/dist/hooks.js.map +1 -1
  32. package/dist/index.d.ts +2 -2
  33. package/dist/index.js +15 -14
  34. package/dist/index.js.map +1 -1
  35. package/dist/providers.js +3 -4
  36. package/dist/rbac/index.js +8 -9
  37. package/dist/schema-DTDZQe2u.d.ts +28 -0
  38. package/dist/types.d.ts +152 -3
  39. package/dist/types.js +51 -16
  40. package/dist/types.js.map +1 -1
  41. package/dist/utils.d.ts +89 -4
  42. package/dist/utils.js +214 -96
  43. package/dist/utils.js.map +1 -1
  44. package/dist/validation.d.ts +1 -343
  45. package/dist/validation.js +3 -100
  46. package/docs/api/classes/ColumnFactory.md +1 -1
  47. package/docs/api/classes/ErrorBoundary.md +1 -1
  48. package/docs/api/classes/InvalidScopeError.md +1 -1
  49. package/docs/api/classes/MissingUserContextError.md +1 -1
  50. package/docs/api/classes/OrganisationContextRequiredError.md +1 -1
  51. package/docs/api/classes/PermissionDeniedError.md +1 -1
  52. package/docs/api/classes/PublicErrorBoundary.md +1 -1
  53. package/docs/api/classes/RBACAuditManager.md +1 -1
  54. package/docs/api/classes/RBACCache.md +1 -1
  55. package/docs/api/classes/RBACEngine.md +1 -1
  56. package/docs/api/classes/RBACError.md +1 -1
  57. package/docs/api/classes/RBACNotInitializedError.md +1 -1
  58. package/docs/api/classes/SecureSupabaseClient.md +1 -1
  59. package/docs/api/classes/StorageUtils.md +1 -1
  60. package/docs/api/enums/FileCategory.md +1 -1
  61. package/docs/api/interfaces/AggregateConfig.md +1 -1
  62. package/docs/api/interfaces/BadgeProps.md +27 -0
  63. package/docs/api/interfaces/ButtonProps.md +1 -1
  64. package/docs/api/interfaces/CardProps.md +1 -1
  65. package/docs/api/interfaces/ColorPalette.md +1 -1
  66. package/docs/api/interfaces/ColorShade.md +1 -1
  67. package/docs/api/interfaces/DataAccessRecord.md +1 -1
  68. package/docs/api/interfaces/DataRecord.md +1 -1
  69. package/docs/api/interfaces/DataTableAction.md +1 -1
  70. package/docs/api/interfaces/DataTableColumn.md +1 -1
  71. package/docs/api/interfaces/DataTableProps.md +1 -1
  72. package/docs/api/interfaces/DataTableToolbarButton.md +1 -1
  73. package/docs/api/interfaces/EmptyStateConfig.md +1 -1
  74. package/docs/api/interfaces/EnhancedNavigationMenuProps.md +1 -1
  75. package/docs/api/interfaces/EventAppRoleData.md +1 -1
  76. package/docs/api/interfaces/EventLogoProps.md +1 -1
  77. package/docs/api/interfaces/ExportColumn.md +1 -1
  78. package/docs/api/interfaces/ExportOptions.md +1 -1
  79. package/docs/api/interfaces/FileDisplayProps.md +1 -1
  80. package/docs/api/interfaces/FileMetadata.md +1 -1
  81. package/docs/api/interfaces/FileReference.md +1 -1
  82. package/docs/api/interfaces/FileSizeLimits.md +1 -1
  83. package/docs/api/interfaces/FileUploadOptions.md +1 -1
  84. package/docs/api/interfaces/FileUploadProps.md +1 -1
  85. package/docs/api/interfaces/FooterProps.md +1 -1
  86. package/docs/api/interfaces/GrantEventAppRoleParams.md +1 -1
  87. package/docs/api/interfaces/InactivityWarningModalProps.md +1 -1
  88. package/docs/api/interfaces/InputProps.md +1 -1
  89. package/docs/api/interfaces/LabelProps.md +1 -1
  90. package/docs/api/interfaces/LoginFormProps.md +1 -1
  91. package/docs/api/interfaces/NavigationAccessRecord.md +1 -1
  92. package/docs/api/interfaces/NavigationContextType.md +1 -1
  93. package/docs/api/interfaces/NavigationGuardProps.md +1 -1
  94. package/docs/api/interfaces/NavigationItem.md +1 -1
  95. package/docs/api/interfaces/NavigationMenuProps.md +1 -1
  96. package/docs/api/interfaces/NavigationProviderProps.md +1 -1
  97. package/docs/api/interfaces/Organisation.md +1 -1
  98. package/docs/api/interfaces/OrganisationContextType.md +1 -1
  99. package/docs/api/interfaces/OrganisationMembership.md +1 -1
  100. package/docs/api/interfaces/OrganisationProviderProps.md +1 -1
  101. package/docs/api/interfaces/OrganisationSecurityError.md +1 -1
  102. package/docs/api/interfaces/PaceAppLayoutProps.md +1 -1
  103. package/docs/api/interfaces/PaceLoginPageProps.md +1 -1
  104. package/docs/api/interfaces/PageAccessRecord.md +1 -1
  105. package/docs/api/interfaces/PagePermissionContextType.md +1 -1
  106. package/docs/api/interfaces/PagePermissionGuardProps.md +1 -1
  107. package/docs/api/interfaces/PagePermissionProviderProps.md +1 -1
  108. package/docs/api/interfaces/PaletteData.md +1 -1
  109. package/docs/api/interfaces/PermissionEnforcerProps.md +1 -1
  110. package/docs/api/interfaces/ProtectedRouteProps.md +1 -1
  111. package/docs/api/interfaces/PublicErrorBoundaryProps.md +1 -1
  112. package/docs/api/interfaces/PublicErrorBoundaryState.md +1 -1
  113. package/docs/api/interfaces/PublicLoadingSpinnerProps.md +1 -1
  114. package/docs/api/interfaces/PublicPageFooterProps.md +1 -1
  115. package/docs/api/interfaces/PublicPageHeaderProps.md +1 -1
  116. package/docs/api/interfaces/PublicPageLayoutProps.md +1 -1
  117. package/docs/api/interfaces/RBACConfig.md +1 -1
  118. package/docs/api/interfaces/RBACLogger.md +1 -1
  119. package/docs/api/interfaces/RevokeEventAppRoleParams.md +1 -1
  120. package/docs/api/interfaces/RoleBasedRouterContextType.md +1 -1
  121. package/docs/api/interfaces/RoleBasedRouterProps.md +1 -1
  122. package/docs/api/interfaces/RoleManagementResult.md +1 -1
  123. package/docs/api/interfaces/RouteAccessRecord.md +1 -1
  124. package/docs/api/interfaces/RouteConfig.md +1 -1
  125. package/docs/api/interfaces/SecureDataContextType.md +1 -1
  126. package/docs/api/interfaces/SecureDataProviderProps.md +1 -1
  127. package/docs/api/interfaces/SessionRestorationLoaderProps.md +1 -1
  128. package/docs/api/interfaces/StorageConfig.md +1 -1
  129. package/docs/api/interfaces/StorageFileInfo.md +1 -1
  130. package/docs/api/interfaces/StorageFileMetadata.md +1 -1
  131. package/docs/api/interfaces/StorageListOptions.md +1 -1
  132. package/docs/api/interfaces/StorageListResult.md +1 -1
  133. package/docs/api/interfaces/StorageUploadOptions.md +1 -1
  134. package/docs/api/interfaces/StorageUploadResult.md +1 -1
  135. package/docs/api/interfaces/StorageUrlOptions.md +1 -1
  136. package/docs/api/interfaces/StyleImport.md +1 -1
  137. package/docs/api/interfaces/SwitchProps.md +1 -1
  138. package/docs/api/interfaces/ToastActionElement.md +1 -1
  139. package/docs/api/interfaces/ToastProps.md +1 -1
  140. package/docs/api/interfaces/UnifiedAuthContextType.md +1 -1
  141. package/docs/api/interfaces/UnifiedAuthProviderProps.md +1 -1
  142. package/docs/api/interfaces/UseInactivityTrackerOptions.md +1 -1
  143. package/docs/api/interfaces/UseInactivityTrackerReturn.md +1 -1
  144. package/docs/api/interfaces/UsePublicEventOptions.md +1 -1
  145. package/docs/api/interfaces/UsePublicEventReturn.md +1 -1
  146. package/docs/api/interfaces/UsePublicFileDisplayOptions.md +1 -1
  147. package/docs/api/interfaces/UsePublicFileDisplayReturn.md +1 -1
  148. package/docs/api/interfaces/UsePublicRouteParamsReturn.md +1 -1
  149. package/docs/api/interfaces/UseResolvedScopeOptions.md +1 -1
  150. package/docs/api/interfaces/UseResolvedScopeReturn.md +1 -1
  151. package/docs/api/interfaces/UserEventAccess.md +1 -1
  152. package/docs/api/interfaces/UserMenuProps.md +1 -1
  153. package/docs/api/interfaces/UserProfile.md +1 -1
  154. package/docs/api/modules.md +84 -15
  155. package/docs/architecture/README.md +0 -1
  156. package/docs/styles/README.md +0 -2
  157. package/examples/RBAC/CompleteRBACExample.tsx +324 -0
  158. package/examples/RBAC/EventBasedApp.tsx +239 -0
  159. package/examples/RBAC/PermissionExample.tsx +151 -0
  160. package/examples/RBAC/index.ts +13 -0
  161. package/examples/public-pages/CorrectPublicPageImplementation.tsx +301 -0
  162. package/examples/public-pages/PublicEventPage.tsx +274 -0
  163. package/examples/public-pages/PublicPageApp.tsx +308 -0
  164. package/examples/public-pages/PublicPageUsageExample.tsx +216 -0
  165. package/examples/public-pages/index.ts +14 -0
  166. package/package.json +1 -10
  167. package/src/__tests__/TEST_STANDARD.md +92 -0
  168. package/src/components/Badge/Badge.test.tsx +314 -0
  169. package/src/components/Badge/Badge.tsx +304 -0
  170. package/src/components/Badge/index.ts +3 -0
  171. package/src/components/DataTable/__tests__/DataTableCore.test-setup.ts +217 -0
  172. package/src/components/DataTable/__tests__/styles.test.ts +1 -1
  173. package/src/components/DataTable/components/ColumnFilter.tsx +8 -4
  174. package/src/components/DataTable/components/DataTableBody.tsx +461 -0
  175. package/src/components/DataTable/components/DraggableColumnHeader.tsx +144 -0
  176. package/src/components/DataTable/components/FilterRow.tsx +9 -3
  177. package/src/components/DataTable/components/PaginationControls.tsx +1 -0
  178. package/src/components/DataTable/components/VirtualizedDataTable.tsx +513 -0
  179. package/src/components/DataTable/components/__tests__/AccessDeniedPage.test.tsx +14 -68
  180. package/src/components/DataTable/components/__tests__/ColumnFilter.test.tsx +62 -0
  181. package/src/components/DataTable/components/__tests__/FilterRow.test.tsx +43 -0
  182. package/src/components/DataTable/core/ActionManager.ts +235 -0
  183. package/src/components/DataTable/core/ColumnManager.ts +205 -0
  184. package/src/components/DataTable/core/DataManager.ts +188 -0
  185. package/src/components/DataTable/core/DataTableContext.tsx +181 -0
  186. package/src/components/DataTable/core/LocalDataAdapter.ts +273 -0
  187. package/src/components/DataTable/core/PluginRegistry.ts +229 -0
  188. package/src/components/DataTable/core/StateManager.ts +311 -0
  189. package/src/components/DataTable/core/interfaces.ts +338 -0
  190. package/src/components/DataTable/styles.ts +27 -6
  191. package/src/components/DataTable/utils/__tests__/columnUtils.test.ts +94 -0
  192. package/src/components/DataTable/utils/columnUtils.ts +40 -0
  193. package/src/components/DataTable/utils/debugTools.ts +609 -0
  194. package/src/components/DataTable/utils/index.ts +1 -0
  195. package/src/components/Dialog/README.md +804 -0
  196. package/src/components/Dialog/utils/__tests__/safeHtml.unit.test.ts +611 -0
  197. package/src/components/Dialog/utils/safeHtml.ts +185 -0
  198. package/src/components/Footer/Footer.test.tsx +1 -1
  199. package/src/components/Form/Form.test.tsx +1 -1
  200. package/src/components/Form/FormErrorSummary.tsx +113 -0
  201. package/src/components/Form/FormFieldset.tsx +127 -0
  202. package/src/components/Form/FormLiveRegion.tsx +198 -0
  203. package/src/components/LoginForm/LoginForm.test.tsx +1 -1
  204. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.performance.test.tsx +76 -10
  205. package/src/components/PaceLoginPage/PaceLoginPage.tsx +1 -1
  206. package/src/components/PasswordReset/PasswordResetForm.test.tsx +597 -0
  207. package/src/components/PasswordReset/PasswordResetForm.tsx +201 -0
  208. package/src/components/PublicLayout/PublicPageDebugger.tsx +104 -0
  209. package/src/components/PublicLayout/PublicPageDiagnostic.tsx +162 -0
  210. package/src/components/PublicLayout/__tests__/PublicPageFooter.test.tsx +1 -1
  211. package/src/components/Select/Select.test.tsx +1 -1
  212. package/src/components/Select/Select.tsx +20 -8
  213. package/src/components/Table/__tests__/Table.test.tsx +1 -1
  214. package/src/components/index.ts +3 -0
  215. package/src/hooks/__tests__/useFileUrl.unit.test.ts +83 -85
  216. package/src/index.ts +4 -0
  217. package/src/rbac/hooks/useCan.test.ts +24 -0
  218. package/src/rbac/hooks/usePermissions.ts +49 -12
  219. package/src/styles/core.css +3 -0
  220. package/src/utils/appConfig.ts +47 -0
  221. package/src/utils/appIdResolver.test.ts +499 -0
  222. package/src/utils/appIdResolver.ts +130 -0
  223. package/src/utils/appNameResolver.simple.test.ts +212 -0
  224. package/src/utils/appNameResolver.test.ts +121 -0
  225. package/src/utils/appNameResolver.ts +191 -0
  226. package/src/utils/audit.ts +127 -0
  227. package/src/utils/auth-utils.ts +96 -0
  228. package/src/utils/bundleAnalysis.ts +129 -0
  229. package/src/utils/cn.ts +7 -0
  230. package/src/utils/debugLogger.ts +67 -0
  231. package/src/utils/deviceFingerprint.ts +215 -0
  232. package/src/utils/dynamicUtils.ts +105 -0
  233. package/src/utils/file-reference.test.ts +788 -0
  234. package/src/utils/file-reference.ts +519 -0
  235. package/src/utils/formatDate.test.ts +237 -0
  236. package/src/utils/formatting.ts +133 -0
  237. package/src/utils/index.ts +7 -0
  238. package/src/utils/lazyLoad.tsx +44 -0
  239. package/src/utils/logger.ts +179 -0
  240. package/src/utils/organisationContext.test.ts +322 -0
  241. package/src/utils/organisationContext.ts +153 -0
  242. package/src/utils/performanceBenchmark.ts +64 -0
  243. package/src/utils/performanceBudgets.ts +110 -0
  244. package/src/utils/permissionTypes.ts +37 -0
  245. package/src/utils/permissionUtils.test.ts +393 -0
  246. package/src/utils/permissionUtils.ts +34 -0
  247. package/src/utils/sanitization.ts +264 -0
  248. package/src/utils/schemaUtils.ts +37 -0
  249. package/src/utils/secureDataAccess.test.ts +711 -0
  250. package/src/utils/secureDataAccess.ts +377 -0
  251. package/src/utils/secureErrors.ts +79 -0
  252. package/src/utils/secureStorage.ts +244 -0
  253. package/src/utils/security.ts +156 -0
  254. package/src/utils/securityMonitor.ts +45 -0
  255. package/src/utils/sessionTracking.ts +126 -0
  256. package/src/utils/validation.ts +111 -0
  257. package/src/utils/validationUtils.ts +120 -0
  258. package/src/validation/index.ts +2 -2
  259. package/dist/chunk-444EZN6N.js.map +0 -1
  260. package/dist/chunk-APIBCTL2.js +0 -670
  261. package/dist/chunk-APIBCTL2.js.map +0 -1
  262. package/dist/chunk-HJGGOMQ6.js.map +0 -1
  263. package/dist/chunk-K2WWTH7O.js +0 -94
  264. package/dist/chunk-K2WWTH7O.js.map +0 -1
  265. package/dist/chunk-L6PGMCMD.js.map +0 -1
  266. package/dist/chunk-LMC26NLJ.js +0 -84
  267. package/dist/chunk-LMC26NLJ.js.map +0 -1
  268. package/dist/chunk-NOHEVYVX.js.map +0 -1
  269. package/dist/chunk-TVYPTYOY.js.map +0 -1
  270. package/dist/validation-8npbysjg.d.ts +0 -177
  271. /package/dist/{DataTable-CYOHOX3O.js.map → DataTable-JXFCA2BJ.js.map} +0 -0
  272. /package/dist/{UnifiedAuthProvider-5E5TUNMS.js.map → UnifiedAuthProvider-XIQQ7LVU.js.map} +0 -0
  273. /package/dist/{chunk-YLKIDTUK.js.map → chunk-22WKWKRX.js.map} +0 -0
  274. /package/dist/{chunk-FHWWBIHA.js.map → chunk-6DXZ6V5Q.js.map} +0 -0
  275. /package/dist/{chunk-2TWNJ46Y.js.map → chunk-6LAAY47Q.js.map} +0 -0
  276. /package/dist/{chunk-XARJS7CD.js.map → chunk-INQLMHPF.js.map} +0 -0
  277. /package/dist/{chunk-SL2YQDR6.js.map → chunk-MA6EPSGZ.js.map} +0 -0
  278. /package/dist/{chunk-5DPZ5EAT.js.map → chunk-OWAG3GSU.js.map} +0 -0
  279. /package/dist/{chunk-LTV3XIJJ.js.map → chunk-T6JN6LH6.js.map} +0 -0
  280. /package/examples/{components → components 2}/DataTable/HierarchicalActionsExample.tsx +0 -0
  281. /package/examples/{components → components 2}/DataTable/HierarchicalExample.tsx +0 -0
  282. /package/examples/{components → components 2}/DataTable/InitialPageSizeExample.tsx +0 -0
  283. /package/examples/{components → components 2}/DataTable/PerformanceExample.tsx +0 -0
  284. /package/examples/{components → components 2}/DataTable/index.ts +0 -0
  285. /package/examples/{components → components 2}/Dialog/BasicHtmlTest.tsx +0 -0
  286. /package/examples/{components → components 2}/Dialog/DebugHtmlExample.tsx +0 -0
  287. /package/examples/{components → components 2}/Dialog/HtmlDialogExample.tsx +0 -0
  288. /package/examples/{components → components 2}/Dialog/ScrollableDialogExample.tsx +0 -0
  289. /package/examples/{components → components 2}/Dialog/SimpleHtmlTest.tsx +0 -0
  290. /package/examples/{components → components 2}/Dialog/SmartDialogExample.tsx +0 -0
  291. /package/examples/{components → components 2}/Dialog/index.ts +0 -0
  292. /package/examples/{components → components 2}/index.ts +0 -0
@@ -0,0 +1,274 @@
1
+ /**
2
+ * @file Public Event Page Example
3
+ * @package @jmruthers/pace-core
4
+ * @module Examples/PublicPages
5
+ * @since 1.0.0
6
+ *
7
+ * A complete example of how to implement a public event page using the public pages feature.
8
+ * This example demonstrates all the key components and hooks working together.
9
+ *
10
+ * @example
11
+ * ```tsx
12
+ * import { PublicEventPage } from '@jmruthers/pace-core/examples';
13
+ *
14
+ * function App() {
15
+ * return (
16
+ * <BrowserRouter>
17
+ * <Routes>
18
+ * <Route path="/public/event/:eventCode" element={<PublicEventPage />} />
19
+ * </Routes>
20
+ * </BrowserRouter>
21
+ * );
22
+ * }
23
+ * ```
24
+ *
25
+ * @dependencies
26
+ * - React 18+ - Component framework
27
+ * - React Router - Routing
28
+ * - Public pages components - Layout and data access
29
+ * - Tailwind CSS - Styling
30
+ */
31
+
32
+ import React from 'react';
33
+ import {
34
+ PublicPageLayout,
35
+ PublicPageHeader,
36
+ PublicPageFooter,
37
+ EventLogo,
38
+ usePublicEvent,
39
+ usePublicRouteParams,
40
+ PublicLoadingSpinner,
41
+ PublicErrorBoundary
42
+ } from '../../src';
43
+
44
+ /**
45
+ * Complete public event page example
46
+ *
47
+ * This component demonstrates how to create a public event page that displays
48
+ * event information, dishes, and branding without requiring authentication.
49
+ *
50
+ * @returns React element with complete public event page
51
+ */
52
+ export function PublicEventPage() {
53
+ const { eventCode } = usePublicRouteParams();
54
+ const { event, isLoading: eventLoading, error: eventError, refetch: refetchEvent } = usePublicEvent(eventCode || '');
55
+
56
+ // Handle loading state
57
+ if (eventLoading) {
58
+ return (
59
+ <PublicLoadingSpinner
60
+ message="Loading event details..."
61
+ />
62
+ );
63
+ }
64
+
65
+ // Handle error state
66
+ if (eventError) {
67
+ return (
68
+ <div className="min-h-screen bg-main-50 flex items-center justify-center">
69
+ <div className="max-w-md mx-auto text-center px-4">
70
+ <div className="mb-6">
71
+ <div className="mx-auto flex items-center justify-center h-12 w-12 rounded-full bg-acc-100 mb-4">
72
+ <svg className="h-6 w-6 text-acc-600" fill="none" viewBox="0 0 24 24" stroke="currentColor">
73
+ <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.964-.833-2.732 0L3.732 16.5c-.77.833.192 2.5 1.732 2.5z" />
74
+ </svg>
75
+ </div>
76
+ <h1 className="text-2xl font-bold text-sec-900 mb-2">
77
+ Event Not Found
78
+ </h1>
79
+ <p className="text-sec-600 mb-6">
80
+ The event code "{eventCode}" is invalid or the event is not available for public viewing.
81
+ </p>
82
+ <button
83
+ onClick={refetchEvent}
84
+ className="px-4 py-2 bg-main-600 text-main-50 rounded-md hover:bg-main-700 transition-colors"
85
+ >
86
+ Try Again
87
+ </button>
88
+ </div>
89
+ </div>
90
+ </div>
91
+ );
92
+ }
93
+
94
+ // Handle missing event
95
+ if (!event) {
96
+ return (
97
+ <div className="min-h-screen bg-main-50 flex items-center justify-center">
98
+ <div className="max-w-md mx-auto text-center px-4">
99
+ <h1 className="text-2xl font-bold text-sec-900 mb-4">
100
+ Event Not Available
101
+ </h1>
102
+ <p className="text-sec-600 mb-6">
103
+ This event is not available for public viewing.
104
+ </p>
105
+ <button
106
+ onClick={refetchEvent}
107
+ className="px-4 py-2 bg-main-600 text-main-50 rounded-md hover:bg-main-700 transition-colors"
108
+ >
109
+ Try Again
110
+ </button>
111
+ </div>
112
+ </div>
113
+ );
114
+ }
115
+
116
+ return (
117
+ <PublicErrorBoundary>
118
+ <PublicPageLayout eventCode={eventCode || ''} event={event}>
119
+ <PublicPageHeader
120
+ event={event}
121
+ eventCode={eventCode || ''}
122
+ title="Event Details"
123
+ description="Public information about this event"
124
+ />
125
+
126
+ <main className="max-w-6xl mx-auto px-4 py-8">
127
+ {/* Event Overview */}
128
+ <div className="grid grid-cols-1 lg:grid-cols-3 gap-8 mb-12">
129
+ {/* Event Information */}
130
+ <div className="lg:col-span-2 space-y-6">
131
+ <div>
132
+ <h2 className="text-2xl font-bold text-sec-900 mb-4">Event Information</h2>
133
+ <div className="bg-sec-50 rounded-lg p-6 space-y-4">
134
+ <div className="grid grid-cols-1 md:grid-cols-2 gap-4">
135
+ <div>
136
+ <h3 className="font-semibold text-sec-700">Date</h3>
137
+ <p className="text-sec-900">
138
+ {event.event_date ? new Date(event.event_date).toLocaleDateString('en-AU', {
139
+ weekday: 'long',
140
+ year: 'numeric',
141
+ month: 'long',
142
+ day: 'numeric'
143
+ }) : 'TBA'}
144
+ </p>
145
+ </div>
146
+ <div>
147
+ <h3 className="font-semibold text-sec-700">Venue</h3>
148
+ <p className="text-sec-900">{event.event_venue || 'TBA'}</p>
149
+ </div>
150
+ <div>
151
+ <h3 className="font-semibold text-sec-700">Participants</h3>
152
+ <p className="text-sec-900">{event.event_participants || 'TBA'}</p>
153
+ </div>
154
+ <div>
155
+ <h3 className="font-semibold text-sec-700">Event Code</h3>
156
+ <p className="text-sec-900 font-mono">{event.event_code}</p>
157
+ </div>
158
+ </div>
159
+
160
+ {event.event_news && (
161
+ <div>
162
+ <h3 className="font-semibold text-sec-700 mb-2">Event News</h3>
163
+ <p className="text-sec-900">{event.event_news}</p>
164
+ </div>
165
+ )}
166
+ </div>
167
+ </div>
168
+ </div>
169
+
170
+ {/* Event Logo */}
171
+ <div className="flex justify-center lg:justify-start">
172
+ <div className="text-center">
173
+ <EventLogo
174
+ eventId={event.event_id}
175
+ eventName={event.event_name}
176
+ organisationId={event.organisation_id}
177
+ size="2xl"
178
+ className="rounded-lg shadow-lg"
179
+ />
180
+ <p className="mt-4 text-sm text-sec-600">Event Logo</p>
181
+ </div>
182
+ </div>
183
+ </div>
184
+
185
+ {/* Note about dishes */}
186
+ <div className="mb-12">
187
+ <div className="bg-main-50 border border-main-200 rounded-lg p-6">
188
+ <h3 className="font-semibold text-main-900 mb-2">Event Dishes</h3>
189
+ <p className="text-main-800">
190
+ For CAKE applications, implement your own dishes functionality using the
191
+ <code className="bg-main-100 px-1 rounded">usePublicDishes</code> hook
192
+ as described in the CAKE implementation guide.
193
+ </p>
194
+ </div>
195
+ </div>
196
+
197
+ {/* Event Footer Information */}
198
+ {event.event_footer && (
199
+ <div className="bg-sec-50 rounded-lg p-6">
200
+ <h3 className="font-semibold text-sec-900 mb-2">Additional Information</h3>
201
+ <p className="text-sec-700">{event.event_footer}</p>
202
+ </div>
203
+ )}
204
+ </main>
205
+
206
+ <PublicPageFooter event={event} />
207
+ </PublicPageLayout>
208
+ </PublicErrorBoundary>
209
+ );
210
+ }
211
+
212
+ /**
213
+ * Compact version of the public event page for smaller displays
214
+ */
215
+ export function PublicEventPageCompact() {
216
+ const { eventCode } = usePublicRouteParams();
217
+ const { event, isLoading, error } = usePublicEvent(eventCode || '');
218
+
219
+ if (isLoading) return <PublicLoadingSpinner message="Loading..." />;
220
+ if (error || !event) return <div>Event not found</div>;
221
+
222
+ return (
223
+ <PublicErrorBoundary>
224
+ <PublicPageLayout eventCode={eventCode || ''} event={event}>
225
+ <PublicPageHeader
226
+ event={event}
227
+ eventCode={eventCode || ''}
228
+ showEventLogo={true}
229
+ showAppLogo={true}
230
+ />
231
+
232
+ <main className="max-w-4xl mx-auto px-4 py-6">
233
+ <div className="text-center mb-6">
234
+ <EventLogo
235
+ eventId={event.event_id}
236
+ eventName={event.event_name}
237
+ organisationId={event.organisation_id}
238
+ size="xl"
239
+ className="mx-auto mb-4"
240
+ />
241
+ <h1 className="text-2xl font-bold text-sec-900">{event.event_name}</h1>
242
+ {event.event_date && (
243
+ <p className="text-sec-600 mt-2">
244
+ {new Date(event.event_date).toLocaleDateString('en-AU')}
245
+ </p>
246
+ )}
247
+ </div>
248
+
249
+ <div className="grid grid-cols-1 md:grid-cols-2 gap-4 mb-6">
250
+ <div className="bg-sec-50 rounded p-4">
251
+ <h3 className="font-semibold text-sec-700">Venue</h3>
252
+ <p className="text-sec-900">{event.event_venue || 'TBA'}</p>
253
+ </div>
254
+ <div className="bg-sec-50 rounded p-4">
255
+ <h3 className="font-semibold text-sec-700">Participants</h3>
256
+ <p className="text-sec-900">{event.event_participants || 'TBA'}</p>
257
+ </div>
258
+ </div>
259
+
260
+ {event.event_news && (
261
+ <div className="bg-main-50 border border-main-200 rounded p-4">
262
+ <h3 className="font-semibold text-main-900 mb-2">Event News</h3>
263
+ <p className="text-main-800">{event.event_news}</p>
264
+ </div>
265
+ )}
266
+ </main>
267
+
268
+ <PublicPageFooter event={event} />
269
+ </PublicPageLayout>
270
+ </PublicErrorBoundary>
271
+ );
272
+ }
273
+
274
+ export default PublicEventPage;
@@ -0,0 +1,308 @@
1
+ /**
2
+ * @file Public Page App Example
3
+ * @package @jmruthers/pace-core
4
+ * @module Examples/PublicPages
5
+ * @since 1.0.0
6
+ *
7
+ * A complete example showing how to properly implement public pages
8
+ * that are completely isolated from the authentication context.
9
+ *
10
+ * This example demonstrates the correct architecture for public pages:
11
+ * 1. Separate routing structure outside authentication context
12
+ * 2. PublicPageProvider for environment variables
13
+ * 3. No authentication context initialization
14
+ * 4. Proper error handling and loading states
15
+ *
16
+ * @example
17
+ * ```tsx
18
+ * import { PublicPageApp } from '@jmruthers/pace-core/examples';
19
+ *
20
+ * // In your main App.tsx, use this for public routes
21
+ * function App() {
22
+ * return (
23
+ * <BrowserRouter>
24
+ * <Routes>
25
+ * <Route path="/public/*" element={<PublicPageApp />} />
26
+ * <Route path="/*" element={<AuthenticatedApp />} />
27
+ * </Routes>
28
+ * </BrowserRouter>
29
+ * );
30
+ * }
31
+ * ```
32
+ */
33
+
34
+ import React from 'react';
35
+ import { Routes, Route } from 'react-router-dom';
36
+ import {
37
+ PublicPageProvider,
38
+ PublicPageLayout,
39
+ PublicPageHeader,
40
+ PublicPageFooter,
41
+ EventLogo,
42
+ usePublicEvent,
43
+ usePublicRouteParams,
44
+ PublicLoadingSpinner,
45
+ PublicErrorBoundary
46
+ } from '../../src';
47
+
48
+ /**
49
+ * Complete public page application that works independently
50
+ * of the main application's authentication context.
51
+ *
52
+ * This component should be used for all public routes that don't
53
+ * require authentication, such as:
54
+ * - Event pages: /events/:eventCode/recipe-grid-report
55
+ * - Public reports: /reports/:reportId
56
+ * - Public information: /info/:pageId
57
+ */
58
+ export function PublicPageApp() {
59
+ return (
60
+ <PublicPageProvider>
61
+ <Routes>
62
+ {/* Public event pages */}
63
+ <Route path="/events/:eventCode/recipe-grid-report" element={<PublicRecipeGridReportPage />} />
64
+ <Route path="/events/:eventCode" element={<PublicEventPage />} />
65
+
66
+ {/* Add more public routes as needed */}
67
+ <Route path="/info/:pageId" element={<PublicInfoPage />} />
68
+ <Route path="/reports/:reportId" element={<PublicReportPage />} />
69
+ </Routes>
70
+ </PublicPageProvider>
71
+ );
72
+ }
73
+
74
+ /**
75
+ * Public recipe grid report page
76
+ *
77
+ * This is the specific page mentioned in the bug report.
78
+ * It demonstrates the correct usage pattern for public pages.
79
+ */
80
+ function PublicRecipeGridReportPage() {
81
+ // Step 1: Extract event code from URL (no auth context triggered)
82
+ const { eventCode } = usePublicRouteParams({ fetchEventData: false });
83
+
84
+ // Step 2: Fetch event data using the event code
85
+ const { event, isLoading, error, refetch } = usePublicEvent(eventCode || '');
86
+
87
+ // Step 3: Handle loading state
88
+ if (isLoading) {
89
+ return (
90
+ <PublicLoadingSpinner
91
+ message="Loading recipe grid report..."
92
+ />
93
+ );
94
+ }
95
+
96
+ // Step 4: Handle error state
97
+ if (error) {
98
+ return (
99
+ <div className="min-h-screen bg-main-50 flex items-center justify-center">
100
+ <div className="max-w-md mx-auto text-center px-4">
101
+ <div className="mb-6">
102
+ <div className="mx-auto flex items-center justify-center h-12 w-12 rounded-full bg-acc-100 mb-4">
103
+ <svg className="h-6 w-6 text-acc-600" fill="none" viewBox="0 0 24 24" stroke="currentColor">
104
+ <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.964-.833-2.732 0L3.732 16.5c-.77.833.192 2.5 1.732 2.5z" />
105
+ </svg>
106
+ </div>
107
+ <h1 className="text-2xl font-bold text-sec-900 mb-2">
108
+ Recipe Grid Report Not Found
109
+ </h1>
110
+ <p className="text-sec-600 mb-6">
111
+ The event code "{eventCode}" is invalid or the recipe grid report is not available for public viewing.
112
+ </p>
113
+ <button
114
+ onClick={refetch}
115
+ className="px-4 py-2 bg-main-600 text-main-50 rounded-md hover:bg-main-700 transition-colors"
116
+ >
117
+ Try Again
118
+ </button>
119
+ </div>
120
+ </div>
121
+ </div>
122
+ );
123
+ }
124
+
125
+ // Step 5: Handle missing event
126
+ if (!event) {
127
+ return (
128
+ <div className="min-h-screen bg-main-50 flex items-center justify-center">
129
+ <div className="max-w-md mx-auto text-center px-4">
130
+ <h1 className="text-2xl font-bold text-sec-900 mb-4">
131
+ Recipe Grid Report Not Available
132
+ </h1>
133
+ <p className="text-sec-600 mb-6">
134
+ This recipe grid report is not available for public viewing.
135
+ </p>
136
+ <button
137
+ onClick={refetch}
138
+ className="px-4 py-2 bg-main-600 text-main-50 rounded-md hover:bg-main-700 transition-colors"
139
+ >
140
+ Try Again
141
+ </button>
142
+ </div>
143
+ </div>
144
+ );
145
+ }
146
+
147
+ // Step 6: Render the public page with event data
148
+ return (
149
+ <PublicPageLayout eventCode={eventCode || ''} event={event}>
150
+ <PublicPageHeader
151
+ event={event}
152
+ eventCode={eventCode || ''}
153
+ title="Recipe Grid Report"
154
+ description="Public recipe grid report for this event"
155
+ />
156
+
157
+ <main className="max-w-6xl mx-auto px-4 py-8">
158
+ {/* Recipe Grid Report Content */}
159
+ <div className="mb-12">
160
+ <div className="bg-main-50 border border-main-200 rounded-lg p-6">
161
+ <h3 className="font-semibold text-main-900 mb-2">Recipe Grid Report</h3>
162
+ <p className="text-main-800">
163
+ This is where your recipe grid report content would go.
164
+ The public page is now working correctly without authentication context conflicts.
165
+ </p>
166
+ <div className="mt-4 text-sm text-main-700">
167
+ <p><strong>Event Code:</strong> {eventCode}</p>
168
+ <p><strong>Event ID:</strong> {event.event_id}</p>
169
+ <p><strong>Event Name:</strong> {event.event_name}</p>
170
+ </div>
171
+ </div>
172
+ </div>
173
+
174
+ {/* Event Information */}
175
+ <div className="grid grid-cols-1 lg:grid-cols-3 gap-8 mb-12">
176
+ <div className="lg:col-span-2 space-y-6">
177
+ <div>
178
+ <h2 className="text-2xl font-bold text-sec-900 mb-4">Event Information</h2>
179
+ <div className="bg-sec-50 rounded-lg p-6 space-y-4">
180
+ <div className="grid grid-cols-1 md:grid-cols-2 gap-4">
181
+ <div>
182
+ <h3 className="font-semibold text-sec-700">Date</h3>
183
+ <p className="text-sec-900">
184
+ {event.event_date ? new Date(event.event_date).toLocaleDateString('en-AU', {
185
+ weekday: 'long',
186
+ year: 'numeric',
187
+ month: 'long',
188
+ day: 'numeric'
189
+ }) : 'TBA'}
190
+ </p>
191
+ </div>
192
+ <div>
193
+ <h3 className="font-semibold text-sec-700">Venue</h3>
194
+ <p className="text-sec-900">{event.event_venue || 'TBA'}</p>
195
+ </div>
196
+ <div>
197
+ <h3 className="font-semibold text-sec-700">Participants</h3>
198
+ <p className="text-sec-900">{event.event_participants || 'TBA'}</p>
199
+ </div>
200
+ <div>
201
+ <h3 className="font-semibold text-sec-700">Event Code</h3>
202
+ <p className="text-sec-900 font-mono">{event.event_code}</p>
203
+ </div>
204
+ </div>
205
+ </div>
206
+ </div>
207
+ </div>
208
+
209
+ {/* Event Logo */}
210
+ <div className="flex justify-center lg:justify-start">
211
+ <div className="text-center">
212
+ <EventLogo
213
+ eventId={event.event_id}
214
+ eventName={event.event_name}
215
+ organisationId={event.organisation_id}
216
+ size="2xl"
217
+ className="rounded-lg shadow-lg"
218
+ />
219
+ <p className="mt-4 text-sm text-sec-600">Event Logo</p>
220
+ </div>
221
+ </div>
222
+ </div>
223
+ </main>
224
+
225
+ <PublicPageFooter event={event} />
226
+ </PublicPageLayout>
227
+ );
228
+ }
229
+
230
+ /**
231
+ * Generic public event page
232
+ */
233
+ function PublicEventPage() {
234
+ const { eventCode } = usePublicRouteParams({ fetchEventData: false });
235
+ const { event, isLoading, error, refetch } = usePublicEvent(eventCode || '');
236
+
237
+ if (isLoading) return <PublicLoadingSpinner message="Loading event..." />;
238
+ if (error || !event) return <div>Event not found</div>;
239
+
240
+ return (
241
+ <PublicPageLayout eventCode={eventCode || ''} event={event}>
242
+ <PublicPageHeader
243
+ event={event}
244
+ eventCode={eventCode || ''}
245
+ title="Event Details"
246
+ description="Public information about this event"
247
+ />
248
+
249
+ <main className="max-w-4xl mx-auto px-4 py-6">
250
+ <div className="text-center mb-6">
251
+ <EventLogo
252
+ eventId={event.event_id}
253
+ eventName={event.event_name}
254
+ organisationId={event.organisation_id}
255
+ size="xl"
256
+ className="mx-auto mb-4"
257
+ />
258
+ <h1 className="text-2xl font-bold text-sec-900">{event.event_name}</h1>
259
+ {event.event_date && (
260
+ <p className="text-sec-600 mt-2">
261
+ {new Date(event.event_date).toLocaleDateString('en-AU')}
262
+ </p>
263
+ )}
264
+ </div>
265
+ </main>
266
+
267
+ <PublicPageFooter event={event} />
268
+ </PublicPageLayout>
269
+ );
270
+ }
271
+
272
+ /**
273
+ * Generic public info page
274
+ */
275
+ function PublicInfoPage() {
276
+ return (
277
+ <div className="min-h-screen bg-main-50 flex items-center justify-center">
278
+ <div className="max-w-md mx-auto text-center px-4">
279
+ <h1 className="text-2xl font-bold text-sec-900 mb-4">
280
+ Public Information Page
281
+ </h1>
282
+ <p className="text-sec-600">
283
+ This is a public information page that doesn't require authentication.
284
+ </p>
285
+ </div>
286
+ </div>
287
+ );
288
+ }
289
+
290
+ /**
291
+ * Generic public report page
292
+ */
293
+ function PublicReportPage() {
294
+ return (
295
+ <div className="min-h-screen bg-main-50 flex items-center justify-center">
296
+ <div className="max-w-md mx-auto text-center px-4">
297
+ <h1 className="text-2xl font-bold text-sec-900 mb-4">
298
+ Public Report Page
299
+ </h1>
300
+ <p className="text-sec-600">
301
+ This is a public report page that doesn't require authentication.
302
+ </p>
303
+ </div>
304
+ </div>
305
+ );
306
+ }
307
+
308
+ export default PublicPageApp;