@jmruthers/pace-core 0.5.135 → 0.5.136
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{DataTable-C7GaRZye.d.ts → DataTable-CWAZZcXC.d.ts} +1 -1
- package/dist/{DataTable-A36PJG6N.js → DataTable-CYOHOX3O.js} +25 -13
- package/dist/{PublicLoadingSpinner-CUAnTvcg.d.ts → EventLogo-801uofbR.d.ts} +51 -135
- package/dist/UnifiedAuthProvider-5E5TUNMS.js +17 -0
- package/dist/{UnifiedAuthProvider-BVKmQd9u.d.ts → UnifiedAuthProvider-DJxGTftH.d.ts} +1 -1
- package/dist/{api-TNIBJWLM.js → api-45XYYO2A.js} +4 -3
- package/dist/{audit-T36HM7IM.js → audit-64X3VJXB.js} +3 -2
- package/dist/{chunk-CTJRBUX2.js → chunk-2TWNJ46Y.js} +2 -2
- package/dist/{chunk-UJI6WSMD.js → chunk-444EZN6N.js} +3 -3
- package/dist/chunk-444EZN6N.js.map +1 -0
- package/dist/{chunk-3CG5L6RN.js → chunk-4MT5BGGL.js} +90 -73
- package/dist/chunk-4MT5BGGL.js.map +1 -0
- package/dist/{chunk-PYUXFQJ3.js → chunk-56XJ3TU6.js} +2 -2
- package/dist/chunk-56XJ3TU6.js.map +1 -0
- package/dist/chunk-5DPZ5EAT.js +60 -0
- package/dist/chunk-5DPZ5EAT.js.map +1 -0
- package/dist/{chunk-66C4BSAY.js → chunk-ANBQRTPX.js} +9 -2
- package/dist/chunk-ANBQRTPX.js.map +1 -0
- package/dist/chunk-APIBCTL2.js +670 -0
- package/dist/chunk-APIBCTL2.js.map +1 -0
- package/dist/{chunk-GKHF54DI.js → chunk-BESYRHQM.js} +10 -4
- package/dist/chunk-BESYRHQM.js.map +1 -0
- package/dist/{chunk-WP5I5GLN.js → chunk-BVYWGZVV.js} +112 -97
- package/dist/chunk-BVYWGZVV.js.map +1 -0
- package/dist/{chunk-GEVIB2UB.js → chunk-ERISIBYU.js} +14 -5
- package/dist/chunk-ERISIBYU.js.map +1 -0
- package/dist/{chunk-CQZU6TFE.js → chunk-FHWWBIHA.js} +100 -62
- package/dist/chunk-FHWWBIHA.js.map +1 -0
- package/dist/{chunk-O3NWNXDY.js → chunk-FMUCXFII.js} +2 -2
- package/dist/chunk-FMUCXFII.js.map +1 -0
- package/dist/{chunk-GVDR7WNV.js → chunk-HJGGOMQ6.js} +194 -518
- package/dist/chunk-HJGGOMQ6.js.map +1 -0
- package/dist/{chunk-BDZUMRBD.js → chunk-K2WWTH7O.js} +13 -6
- package/dist/chunk-K2WWTH7O.js.map +1 -0
- package/dist/{chunk-BYXRHAIF.js → chunk-L6PGMCMD.js} +23 -14
- package/dist/chunk-L6PGMCMD.js.map +1 -0
- package/dist/chunk-LMC26NLJ.js +84 -0
- package/dist/chunk-LMC26NLJ.js.map +1 -0
- package/dist/{chunk-M6DDYFUD.js → chunk-LS353YLY.js} +19 -16
- package/dist/chunk-LS353YLY.js.map +1 -0
- package/dist/{chunk-ZYZCRSBD.js → chunk-LTV3XIJJ.js} +16 -11
- package/dist/chunk-LTV3XIJJ.js.map +1 -0
- package/dist/{chunk-HMNOSGVA.js → chunk-NOHEVYVX.js} +377 -666
- package/dist/chunk-NOHEVYVX.js.map +1 -0
- package/dist/{chunk-JCQZ6LA7.js → chunk-Q5QRDWKI.js} +9 -3
- package/dist/chunk-Q5QRDWKI.js.map +1 -0
- package/dist/chunk-S5OFRT4M.js +94 -0
- package/dist/chunk-S5OFRT4M.js.map +1 -0
- package/dist/{chunk-3DBFLLLU.js → chunk-SBVILCCA.js} +14 -9
- package/dist/chunk-SBVILCCA.js.map +1 -0
- package/dist/{chunk-TGIY2AR2.js → chunk-SL2YQDR6.js} +4 -3
- package/dist/{chunk-TGIY2AR2.js.map → chunk-SL2YQDR6.js.map} +1 -1
- package/dist/{chunk-VZ5OR6HD.js → chunk-TVYPTYOY.js} +55 -179
- package/dist/chunk-TVYPTYOY.js.map +1 -0
- package/dist/{chunk-ZV77RZMU.js → chunk-XARJS7CD.js} +2 -2
- package/dist/chunk-XDNLUEXI.js +138 -0
- package/dist/chunk-XDNLUEXI.js.map +1 -0
- package/dist/{chunk-F64FFPOZ.js → chunk-YLKIDTUK.js} +26 -20
- package/dist/chunk-YLKIDTUK.js.map +1 -0
- package/dist/{chunk-5F3NDPJV.js → chunk-ZZ2SS7NI.js} +10 -5
- package/dist/chunk-ZZ2SS7NI.js.map +1 -0
- package/dist/components.d.ts +7 -287
- package/dist/components.js +26 -157
- package/dist/components.js.map +1 -1
- package/dist/{file-reference-C9isKNPn.d.ts → file-reference-C6Gkn77H.d.ts} +1 -1
- package/dist/{formatting-DFcCxUEk.d.ts → formatting-CvUXy2mF.d.ts} +1 -1
- package/dist/hooks.d.ts +3 -3
- package/dist/hooks.js +22 -16
- package/dist/hooks.js.map +1 -1
- package/dist/index.d.ts +101 -9
- package/dist/index.js +43 -31
- package/dist/index.js.map +1 -1
- package/dist/providers.d.ts +1 -1
- package/dist/providers.js +5 -4
- package/dist/rbac/index.js +13 -12
- package/dist/styles/index.js +2 -1
- package/dist/theming/runtime.d.ts +2 -19
- package/dist/theming/runtime.js +2 -1
- package/dist/{types-D5rqZQXk.d.ts → types-Dfz9dmVH.d.ts} +12 -1
- package/dist/types.d.ts +2 -2
- package/dist/types.js +1 -1
- package/dist/{useInactivityTracker-MRUU55XI.js → useInactivityTracker-TO6ZOF35.js} +3 -2
- package/dist/{usePublicRouteParams-Dyt1tzI9.d.ts → usePublicRouteParams-B7PabvuH.d.ts} +1 -1
- package/dist/utils.d.ts +195 -232
- package/dist/utils.js +173 -331
- package/dist/utils.js.map +1 -1
- package/dist/{validation-DnhrNMju.d.ts → validation-8npbysjg.d.ts} +26 -8
- package/dist/validation.d.ts +261 -10
- package/dist/validation.js +82 -440
- package/dist/validation.js.map +1 -1
- package/docs/api/classes/ColumnFactory.md +1 -1
- package/docs/api/classes/ErrorBoundary.md +6 -6
- package/docs/api/classes/InvalidScopeError.md +1 -1
- package/docs/api/classes/MissingUserContextError.md +1 -1
- package/docs/api/classes/OrganisationContextRequiredError.md +1 -1
- package/docs/api/classes/PermissionDeniedError.md +1 -1
- package/docs/api/classes/PublicErrorBoundary.md +1 -1
- package/docs/api/classes/RBACAuditManager.md +6 -6
- package/docs/api/classes/RBACCache.md +1 -1
- package/docs/api/classes/RBACEngine.md +7 -7
- package/docs/api/classes/RBACError.md +1 -1
- package/docs/api/classes/RBACNotInitializedError.md +1 -1
- package/docs/api/classes/SecureSupabaseClient.md +1 -1
- package/docs/api/classes/StorageUtils.md +1 -1
- package/docs/api/enums/FileCategory.md +1 -1
- package/docs/api/interfaces/AggregateConfig.md +4 -4
- package/docs/api/interfaces/ButtonProps.md +1 -1
- package/docs/api/interfaces/CardProps.md +1 -1
- package/docs/api/interfaces/ColorPalette.md +1 -1
- package/docs/api/interfaces/ColorShade.md +29 -4
- package/docs/api/interfaces/DataAccessRecord.md +9 -9
- package/docs/api/interfaces/DataRecord.md +1 -1
- package/docs/api/interfaces/DataTableAction.md +18 -18
- package/docs/api/interfaces/DataTableColumn.md +61 -1
- package/docs/api/interfaces/DataTableProps.md +1 -1
- package/docs/api/interfaces/DataTableToolbarButton.md +7 -7
- package/docs/api/interfaces/EmptyStateConfig.md +5 -5
- package/docs/api/interfaces/EnhancedNavigationMenuProps.md +14 -14
- package/docs/api/interfaces/EventAppRoleData.md +1 -1
- package/docs/api/interfaces/EventLogoProps.md +152 -0
- package/docs/api/interfaces/ExportColumn.md +1 -1
- package/docs/api/interfaces/ExportOptions.md +8 -8
- package/docs/api/interfaces/FileDisplayProps.md +15 -15
- package/docs/api/interfaces/FileMetadata.md +1 -1
- package/docs/api/interfaces/FileReference.md +1 -1
- package/docs/api/interfaces/FileSizeLimits.md +1 -1
- package/docs/api/interfaces/FileUploadOptions.md +1 -1
- package/docs/api/interfaces/FileUploadProps.md +1 -1
- package/docs/api/interfaces/FooterProps.md +1 -1
- package/docs/api/interfaces/GrantEventAppRoleParams.md +1 -1
- package/docs/api/interfaces/InactivityWarningModalProps.md +1 -1
- package/docs/api/interfaces/InputProps.md +1 -1
- package/docs/api/interfaces/LabelProps.md +1 -1
- package/docs/api/interfaces/LoginFormProps.md +1 -1
- package/docs/api/interfaces/NavigationAccessRecord.md +10 -10
- package/docs/api/interfaces/NavigationContextType.md +9 -9
- package/docs/api/interfaces/NavigationGuardProps.md +10 -10
- package/docs/api/interfaces/NavigationItem.md +1 -1
- package/docs/api/interfaces/NavigationMenuProps.md +1 -1
- package/docs/api/interfaces/NavigationProviderProps.md +7 -7
- package/docs/api/interfaces/Organisation.md +1 -1
- package/docs/api/interfaces/OrganisationContextType.md +1 -1
- package/docs/api/interfaces/OrganisationMembership.md +1 -1
- package/docs/api/interfaces/OrganisationProviderProps.md +1 -1
- package/docs/api/interfaces/OrganisationSecurityError.md +1 -1
- package/docs/api/interfaces/PaceAppLayoutProps.md +27 -27
- package/docs/api/interfaces/PaceLoginPageProps.md +4 -4
- package/docs/api/interfaces/PageAccessRecord.md +8 -8
- package/docs/api/interfaces/PagePermissionContextType.md +8 -8
- package/docs/api/interfaces/PagePermissionGuardProps.md +11 -11
- package/docs/api/interfaces/PagePermissionProviderProps.md +7 -7
- package/docs/api/interfaces/PaletteData.md +4 -4
- package/docs/api/interfaces/PermissionEnforcerProps.md +11 -11
- package/docs/api/interfaces/ProtectedRouteProps.md +6 -6
- package/docs/api/interfaces/PublicErrorBoundaryProps.md +1 -1
- package/docs/api/interfaces/PublicErrorBoundaryState.md +1 -1
- package/docs/api/interfaces/PublicLoadingSpinnerProps.md +1 -1
- package/docs/api/interfaces/PublicPageFooterProps.md +1 -1
- package/docs/api/interfaces/PublicPageHeaderProps.md +1 -1
- package/docs/api/interfaces/PublicPageLayoutProps.md +1 -1
- package/docs/api/interfaces/RBACConfig.md +1 -1
- package/docs/api/interfaces/RBACLogger.md +1 -1
- package/docs/api/interfaces/RevokeEventAppRoleParams.md +1 -1
- package/docs/api/interfaces/RoleBasedRouterContextType.md +8 -8
- package/docs/api/interfaces/RoleBasedRouterProps.md +10 -10
- package/docs/api/interfaces/RoleManagementResult.md +1 -1
- package/docs/api/interfaces/RouteAccessRecord.md +10 -10
- package/docs/api/interfaces/RouteConfig.md +10 -10
- package/docs/api/interfaces/SecureDataContextType.md +9 -9
- package/docs/api/interfaces/SecureDataProviderProps.md +8 -8
- package/docs/api/interfaces/SessionRestorationLoaderProps.md +21 -0
- package/docs/api/interfaces/StorageConfig.md +1 -1
- package/docs/api/interfaces/StorageFileInfo.md +1 -1
- package/docs/api/interfaces/StorageFileMetadata.md +1 -1
- package/docs/api/interfaces/StorageListOptions.md +1 -1
- package/docs/api/interfaces/StorageListResult.md +1 -1
- package/docs/api/interfaces/StorageUploadOptions.md +1 -1
- package/docs/api/interfaces/StorageUploadResult.md +1 -1
- package/docs/api/interfaces/StorageUrlOptions.md +1 -1
- package/docs/api/interfaces/StyleImport.md +1 -1
- package/docs/api/interfaces/SwitchProps.md +1 -1
- package/docs/api/interfaces/ToastActionElement.md +1 -1
- package/docs/api/interfaces/ToastProps.md +1 -1
- package/docs/api/interfaces/UnifiedAuthContextType.md +53 -53
- package/docs/api/interfaces/UnifiedAuthProviderProps.md +13 -13
- package/docs/api/interfaces/UseInactivityTrackerOptions.md +9 -9
- package/docs/api/interfaces/UseInactivityTrackerReturn.md +8 -8
- package/docs/api/interfaces/UsePublicEventOptions.md +3 -3
- package/docs/api/interfaces/UsePublicEventReturn.md +5 -5
- package/docs/api/interfaces/UsePublicFileDisplayOptions.md +4 -4
- package/docs/api/interfaces/UsePublicFileDisplayReturn.md +9 -9
- package/docs/api/interfaces/UsePublicRouteParamsReturn.md +1 -1
- package/docs/api/interfaces/UseResolvedScopeOptions.md +4 -4
- package/docs/api/interfaces/UseResolvedScopeReturn.md +4 -4
- package/docs/api/interfaces/UserEventAccess.md +11 -11
- package/docs/api/interfaces/UserMenuProps.md +1 -1
- package/docs/api/interfaces/UserProfile.md +1 -1
- package/docs/api/modules.md +514 -212
- package/docs/api-reference/components.md +106 -26
- package/docs/architecture/README.md +0 -2
- package/docs/implementation-guides/data-tables.md +277 -13
- package/docs/implementation-guides/forms.md +1 -16
- package/docs/implementation-guides/permission-enforcement.md +8 -2
- package/examples/README.md +30 -14
- package/examples/STRUCTURE.md +125 -0
- package/examples/components/DataTable/HierarchicalActionsExample.tsx +421 -0
- package/examples/components/DataTable/HierarchicalExample.tsx +475 -0
- package/examples/components/DataTable/InitialPageSizeExample.tsx +177 -0
- package/examples/components/DataTable/PerformanceExample.tsx +506 -0
- package/examples/components/DataTable/index.ts +13 -0
- package/examples/components/Dialog/BasicHtmlTest.tsx +55 -0
- package/examples/components/Dialog/DebugHtmlExample.tsx +68 -0
- package/examples/components/Dialog/HtmlDialogExample.tsx +202 -0
- package/examples/components/Dialog/ScrollableDialogExample.tsx +290 -0
- package/examples/components/Dialog/SimpleHtmlTest.tsx +61 -0
- package/examples/components/Dialog/SmartDialogExample.tsx +322 -0
- package/examples/components/Dialog/index.ts +15 -0
- package/examples/components/index.ts +11 -0
- package/examples/features/index.ts +12 -0
- package/examples/{public-pages → features/public-pages}/CorrectPublicPageImplementation.tsx +1 -1
- package/examples/{public-pages → features/public-pages}/PublicEventPage.tsx +1 -1
- package/examples/{public-pages → features/public-pages}/PublicPageApp.tsx +1 -1
- package/examples/{public-pages → features/public-pages}/PublicPageUsageExample.tsx +1 -1
- package/examples/index.ts +11 -3
- package/package.json +30 -10
- package/src/components/Alert/Alert.tsx +1 -1
- package/src/components/Avatar/Avatar.tsx +1 -1
- package/src/components/Button/Button.tsx +1 -1
- package/src/components/Card/Card.tsx +1 -1
- package/src/components/Checkbox/Checkbox.tsx +1 -1
- package/src/components/DataTable/DataTable.test.tsx +1 -1
- package/src/components/DataTable/DataTable.tsx +1 -30
- package/src/components/DataTable/__tests__/DataTable.grouping-aggregation.test.tsx +562 -0
- package/src/components/DataTable/__tests__/styles.test.ts +2 -2
- package/src/components/DataTable/components/ActionButtons.tsx +0 -15
- package/src/components/DataTable/components/DataTableCore.tsx +4 -185
- package/src/components/DataTable/components/DataTableErrorBoundary.tsx +1 -1
- package/src/components/DataTable/components/DataTableModals.tsx +1 -27
- package/src/components/DataTable/components/EditableRow.tsx +1 -1
- package/src/components/DataTable/components/ImportModal.tsx +2 -14
- package/src/components/DataTable/components/PaginationControls.tsx +1 -1
- package/src/components/DataTable/components/UnifiedTableBody.tsx +109 -82
- package/src/components/DataTable/components/__tests__/ActionButtons.test.tsx +1 -1
- package/src/components/DataTable/components/__tests__/DataTableErrorBoundary.test.tsx +1 -1
- package/src/components/DataTable/components/__tests__/DataTableModals.test.tsx +1 -1
- package/src/components/DataTable/components/__tests__/ImportModal.test.tsx +1 -1
- package/src/components/DataTable/examples/GroupingAggregationExample.tsx +273 -0
- package/src/components/DataTable/examples/HierarchicalActionsExample.tsx +1 -1
- package/src/components/DataTable/examples/__tests__/HierarchicalActionsExample.test.tsx +1 -1
- package/src/components/DataTable/hooks/useColumnOrderPersistence.ts +1 -1
- package/src/components/DataTable/hooks/useColumnVisibilityPersistence.ts +1 -1
- package/src/components/DataTable/hooks/useDataTablePermissions.ts +2 -23
- package/src/components/DataTable/index.ts +4 -0
- package/src/components/DataTable/styles.ts +1 -1
- package/src/components/DataTable/types.ts +13 -0
- package/src/components/DataTable/utils/__tests__/exportUtils.test.ts +1 -1
- package/src/components/DataTable/utils/aggregationUtils.ts +161 -0
- package/src/components/DataTable/utils/exportUtils.ts +1 -1
- package/src/components/DataTable/utils/flexibleImport.ts +1 -11
- package/src/components/DataTable/utils/index.ts +1 -0
- package/src/components/DataTable/utils/paginationUtils.ts +1 -1
- package/src/components/Dialog/Dialog.tsx +2 -2
- package/src/components/ErrorBoundary/ErrorBoundary.test.tsx +35 -7
- package/src/components/ErrorBoundary/ErrorBoundary.tsx +5 -4
- package/src/components/EventSelector/EventSelector.tsx +3 -2
- package/src/components/FileDisplay/FileDisplay.tsx +2 -36
- package/src/components/FileUpload/FileUpload.test.tsx +2 -2
- package/src/components/FileUpload/FileUpload.tsx +2 -2
- package/src/components/Footer/Footer.tsx +1 -1
- package/src/components/Form/Form.test.tsx +4 -509
- package/src/components/Form/Form.tsx +1 -1
- package/src/components/Form/FormField.tsx +1 -1
- package/src/components/Form/index.ts +0 -12
- package/src/components/Header/Header.tsx +1 -1
- package/src/components/Input/Input.tsx +1 -1
- package/src/components/Label/Label.tsx +1 -1
- package/src/components/LoginForm/LoginForm.tsx +1 -1
- package/src/components/NavigationMenu/NavigationMenu.test.tsx +19 -3
- package/src/components/NavigationMenu/NavigationMenu.tsx +9 -8
- package/src/components/OrganisationSelector/OrganisationSelector.tsx +4 -3
- package/src/components/PaceAppLayout/PaceAppLayout.tsx +14 -12
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.integration.test.tsx +0 -16
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.security.test.tsx +0 -1
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.unit.test.tsx +0 -9
- package/src/components/PaceLoginPage/PaceLoginPage.test.tsx +35 -3
- package/src/components/PaceLoginPage/PaceLoginPage.tsx +13 -12
- package/src/components/PasswordReset/PasswordChangeForm.tsx +1 -1
- package/src/components/PasswordReset/index.ts +0 -2
- package/src/components/Progress/Progress.tsx +1 -1
- package/src/components/ProtectedRoute/ProtectedRoute.test.tsx +35 -8
- package/src/components/ProtectedRoute/ProtectedRoute.tsx +3 -2
- package/src/components/PublicLayout/PublicErrorBoundary.tsx +1 -1
- package/src/components/PublicLayout/PublicLoadingSpinner.tsx +1 -1
- package/src/components/PublicLayout/PublicPageContextChecker.tsx +44 -43
- package/src/components/PublicLayout/PublicPageFooter.tsx +1 -1
- package/src/components/PublicLayout/PublicPageHeader.tsx +1 -15
- package/src/components/PublicLayout/PublicPageProvider.tsx +3 -2
- package/src/components/PublicLayout/__tests__/PublicPageContextChecker.test.tsx +2 -0
- package/src/components/PublicLayout/index.ts +4 -2
- package/src/components/Select/Select.tsx +1 -1
- package/src/components/{SessionRestorationLoader.tsx → SessionRestorationLoader/SessionRestorationLoader.tsx} +3 -2
- package/src/components/SessionRestorationLoader/index.ts +3 -0
- package/src/components/Switch/Switch.tsx +1 -1
- package/src/components/Table/Table.tsx +1 -1
- package/src/components/Toast/Toast.tsx +1 -1
- package/src/components/Tooltip/Tooltip.tsx +1 -1
- package/src/components/index.ts +4 -10
- package/src/hooks/__tests__/hooks.integration.test.tsx +37 -22
- package/src/hooks/__tests__/useComponentPerformance.unit.test.tsx +33 -17
- package/src/hooks/__tests__/useDataTablePerformance.unit.test.ts +28 -3
- package/src/hooks/__tests__/useFileDisplay.unit.test.ts +36 -9
- package/src/hooks/__tests__/useInactivityTracker.unit.test.ts +26 -2
- package/src/hooks/__tests__/usePerformanceMonitor.unit.test.ts +19 -6
- package/src/hooks/__tests__/usePermissionCache.simple.test.ts +17 -4
- package/src/hooks/__tests__/usePermissionCache.unit.test.ts +17 -4
- package/src/hooks/__tests__/usePublicEvent.simple.test.ts +26 -6
- package/src/hooks/__tests__/usePublicFileDisplay.test.ts +16 -6
- package/src/hooks/__tests__/useSecureDataAccess.unit.test.tsx +3 -3
- package/src/hooks/__tests__/useSessionRestoration.unit.test.tsx +17 -3
- package/src/hooks/public/usePublicEvent.ts +7 -6
- package/src/hooks/public/usePublicEventLogo.ts +7 -4
- package/src/hooks/public/usePublicFileDisplay.ts +6 -150
- package/src/hooks/useComponentPerformance.ts +4 -1
- package/src/hooks/useDataTablePerformance.ts +4 -3
- package/src/hooks/useEventTheme.test.ts +18 -5
- package/src/hooks/useEventTheme.ts +4 -1
- package/src/hooks/useEvents.ts +2 -0
- package/src/hooks/useFileDisplay.ts +9 -8
- package/src/hooks/useFileReference.ts +4 -1
- package/src/hooks/useFileUrl.ts +4 -1
- package/src/hooks/useInactivityTracker.ts +5 -4
- package/src/hooks/useOrganisationSecurity.test.ts +33 -12
- package/src/hooks/useOrganisationSecurity.ts +8 -7
- package/src/hooks/usePerformanceMonitor.ts +6 -3
- package/src/hooks/usePermissionCache.ts +13 -6
- package/src/hooks/useSecureDataAccess.test.ts +2 -2
- package/src/hooks/useSecureDataAccess.ts +9 -8
- package/src/hooks/useSessionRestoration.ts +4 -1
- package/src/hooks/useStorage.ts +4 -1
- package/src/index.ts +16 -7
- package/src/providers/services/AuthServiceProvider.tsx +3 -2
- package/src/providers/services/EventServiceProvider.tsx +2 -1
- package/src/providers/services/InactivityServiceProvider.tsx +2 -1
- package/src/providers/services/OrganisationServiceProvider.tsx +2 -1
- package/src/providers/services/UnifiedAuthProvider.tsx +4 -3
- package/src/providers/services/__tests__/AuthServiceProvider.integration.test.tsx +22 -2
- package/src/providers/services/__tests__/UnifiedAuthProvider.integration.test.tsx +24 -2
- package/src/rbac/__tests__/cache-invalidation.test.ts +20 -6
- package/src/rbac/api.ts +5 -2
- package/src/rbac/audit-enhanced.ts +6 -6
- package/src/rbac/audit.test.ts +60 -38
- package/src/rbac/audit.ts +8 -8
- package/src/rbac/cache-invalidation.ts +7 -4
- package/src/rbac/components/EnhancedNavigationMenu.tsx +11 -5
- package/src/rbac/components/NavigationGuard.tsx +7 -3
- package/src/rbac/components/NavigationProvider.tsx +6 -3
- package/src/rbac/components/PagePermissionGuard.tsx +28 -16
- package/src/rbac/components/PagePermissionProvider.tsx +4 -1
- package/src/rbac/components/PermissionEnforcer.tsx +9 -3
- package/src/rbac/components/RoleBasedRouter.tsx +3 -1
- package/src/rbac/components/SecureDataProvider.tsx +7 -3
- package/src/rbac/components/__tests__/EnhancedNavigationMenu.test.tsx +87 -61
- package/src/rbac/components/__tests__/NavigationGuard.test.tsx +83 -33
- package/src/rbac/components/__tests__/NavigationProvider.test.tsx +36 -13
- package/src/rbac/components/__tests__/PagePermissionGuard.test.tsx +2 -2
- package/src/rbac/components/__tests__/PagePermissionProvider.test.tsx +22 -8
- package/src/rbac/components/__tests__/PermissionEnforcer.test.tsx +19 -6
- package/src/rbac/components/__tests__/SecureDataProvider.fixed.test.tsx +43 -17
- package/src/rbac/components/__tests__/SecureDataProvider.test.tsx +42 -17
- package/src/rbac/engine.ts +15 -7
- package/src/rbac/hooks/usePermissions.ts +7 -3
- package/src/rbac/hooks/useResolvedScope.test.ts +2 -2
- package/src/rbac/hooks/useResolvedScope.ts +10 -7
- package/src/rbac/permissions.ts +5 -2
- package/src/rbac/security.test.ts +27 -16
- package/src/rbac/security.ts +5 -4
- package/src/services/AuthService.ts +22 -21
- package/src/services/EventService.ts +12 -12
- package/src/services/InactivityService.ts +5 -4
- package/src/services/OrganisationService.ts +26 -25
- package/src/services/__tests__/AuthService.test.ts +51 -19
- package/src/services/__tests__/EventService.test.ts +37 -5
- package/src/services/__tests__/InactivityService.test.ts +38 -4
- package/src/services/__tests__/OrganisationService.test.ts +3 -8
- package/src/services/base/BaseService.ts +3 -1
- package/src/theming/__tests__/runtime.test.ts +21 -12
- package/src/theming/parseEventColours.ts +5 -19
- package/src/theming/runtime.ts +8 -4
- package/src/types/validation.ts +2 -29
- package/src/utils/__tests__/appConfig.unit.test.ts +1 -1
- package/src/utils/__tests__/audit.unit.test.ts +1 -1
- package/src/utils/__tests__/auth-utils.unit.test.ts +1 -1
- package/src/utils/__tests__/bundleAnalysis.unit.test.ts +19 -19
- package/src/utils/__tests__/cn.unit.test.ts +1 -1
- package/src/utils/__tests__/debugLogger.test.ts +1 -1
- package/src/utils/__tests__/deviceFingerprint.unit.test.ts +1 -1
- package/src/utils/__tests__/dynamicUtils.unit.test.ts +1 -1
- package/src/utils/__tests__/formatting.unit.test.ts +1 -1
- package/src/utils/__tests__/lazyLoad.unit.test.tsx +1 -1
- package/src/utils/__tests__/logger.unit.test.ts +1 -1
- package/src/utils/__tests__/organisationContext.unit.test.ts +1 -1
- package/src/utils/__tests__/performanceBenchmark.test.ts +1 -1
- package/src/utils/__tests__/performanceBudgets.unit.test.ts +1 -1
- package/src/utils/__tests__/permissionTypes.unit.test.ts +1 -1
- package/src/utils/__tests__/permissionUtils.unit.test.ts +1 -1
- package/src/utils/__tests__/sanitization.unit.test.ts +1 -1
- package/src/utils/__tests__/schemaUtils.unit.test.ts +1 -1
- package/src/utils/__tests__/secureDataAccess.unit.test.ts +1 -1
- package/src/utils/__tests__/secureErrors.unit.test.ts +33 -15
- package/src/utils/__tests__/secureStorage.unit.test.ts +1 -1
- package/src/utils/__tests__/security.unit.test.ts +40 -18
- package/src/utils/__tests__/securityMonitor.unit.test.ts +1 -1
- package/src/utils/__tests__/sessionTracking.unit.test.ts +40 -29
- package/src/utils/__tests__/validationUtils.unit.test.ts +19 -6
- package/src/utils/{appIdResolver.test.ts → app/appIdResolver.test.ts} +28 -30
- package/src/utils/{appIdResolver.ts → app/appIdResolver.ts} +8 -5
- package/src/utils/{appNameResolver.test.ts → app/appNameResolver.test.ts} +1 -1
- package/src/utils/{appNameResolver.ts → app/appNameResolver.ts} +5 -1
- package/src/utils/{organisationContext.ts → context/organisationContext.ts} +6 -3
- package/src/utils/{sessionTracking.ts → context/sessionTracking.ts} +11 -12
- package/src/utils/{logger.ts → core/logger.ts} +4 -2
- package/src/utils/{deviceFingerprint.ts → device/deviceFingerprint.ts} +1 -1
- package/src/utils/{lazyLoad.tsx → dynamic/lazyLoad.tsx} +2 -2
- package/src/utils/{file-reference.test.ts → file-reference/__tests__/file-reference.test.ts} +5 -5
- package/src/utils/{file-reference.ts → file-reference/index.ts} +20 -38
- package/src/utils/index.ts +32 -54
- package/src/utils/{secureErrors.ts → security/secureErrors.ts} +6 -3
- package/src/utils/{security.ts → security/security.ts} +5 -2
- package/src/utils/storage/__tests__/helpers.unit.test.ts +1 -4
- package/src/utils/storage/helpers.ts +15 -8
- package/src/{components/Dialog/utils/__tests__/safeHtml.unit.test.ts → utils/validation/__tests__/htmlSanitization.unit.test.ts} +9 -15
- package/src/{validation → utils/validation}/csrf.ts +1 -1
- package/src/{components/Dialog/utils/safeHtml.ts → utils/validation/htmlSanitization.ts} +9 -10
- package/src/utils/validation/index.ts +79 -0
- package/src/utils/{sanitization.ts → validation/sanitization.ts} +71 -2
- package/src/{validation/schemaUtils.ts → utils/validation/schema.ts} +11 -6
- package/src/{validation → utils/validation}/sqlInjectionProtection.ts +2 -0
- package/src/utils/{validationUtils.ts → validation/validationUtils.ts} +4 -1
- package/src/validation/index.ts +3 -34
- package/dist/UnifiedAuthProvider-CQDZRJIS.js +0 -16
- package/dist/chunk-24MKLB7U.js +0 -81
- package/dist/chunk-24MKLB7U.js.map +0 -1
- package/dist/chunk-3CG5L6RN.js.map +0 -1
- package/dist/chunk-3DBFLLLU.js.map +0 -1
- package/dist/chunk-5F3NDPJV.js.map +0 -1
- package/dist/chunk-66C4BSAY.js.map +0 -1
- package/dist/chunk-BDZUMRBD.js.map +0 -1
- package/dist/chunk-BYXRHAIF.js.map +0 -1
- package/dist/chunk-CDQ3PX7L.js +0 -18
- package/dist/chunk-CDQ3PX7L.js.map +0 -1
- package/dist/chunk-CQZU6TFE.js.map +0 -1
- package/dist/chunk-F64FFPOZ.js.map +0 -1
- package/dist/chunk-GEVIB2UB.js.map +0 -1
- package/dist/chunk-GKHF54DI.js.map +0 -1
- package/dist/chunk-GVDR7WNV.js.map +0 -1
- package/dist/chunk-HMNOSGVA.js.map +0 -1
- package/dist/chunk-JCQZ6LA7.js.map +0 -1
- package/dist/chunk-M6DDYFUD.js.map +0 -1
- package/dist/chunk-O3NWNXDY.js.map +0 -1
- package/dist/chunk-PYUXFQJ3.js.map +0 -1
- package/dist/chunk-UJI6WSMD.js.map +0 -1
- package/dist/chunk-VZ5OR6HD.js.map +0 -1
- package/dist/chunk-WP5I5GLN.js.map +0 -1
- package/dist/chunk-ZYZCRSBD.js.map +0 -1
- package/src/components/Dialog/README.md +0 -804
- package/src/components/Form/FormErrorSummary.tsx +0 -113
- package/src/components/Form/FormFieldset.tsx +0 -127
- package/src/components/Form/FormLiveRegion.tsx +0 -198
- package/src/components/PasswordReset/PasswordResetForm.test.tsx +0 -597
- package/src/components/PasswordReset/PasswordResetForm.tsx +0 -201
- package/src/components/PublicLayout/PublicPageDebugger.tsx +0 -104
- package/src/components/PublicLayout/PublicPageDiagnostic.tsx +0 -162
- package/src/components/PublicLayout/__tests__/PublicPageDebugger.test.tsx +0 -185
- package/src/examples/CorrectPublicPageImplementation.tsx +0 -304
- package/src/examples/PublicEventPage.tsx +0 -287
- package/src/examples/PublicPageApp.tsx +0 -321
- package/src/examples/PublicPageUsageExample.tsx +0 -218
- package/src/utils/schemaUtils.ts +0 -37
- package/src/validation/__tests__/common.unit.test.ts +0 -101
- package/src/validation/__tests__/csrf.unit.test.ts +0 -365
- package/src/validation/__tests__/passwordSchema.unit.test.ts +0 -203
- package/src/validation/__tests__/sanitization.unit.test.ts +0 -250
- package/src/validation/__tests__/schemaUtils.unit.test.ts +0 -451
- package/src/validation/__tests__/sqlInjectionProtection.unit.test.ts +0 -462
- package/src/validation/__tests__/user.unit.test.ts +0 -440
- package/src/validation/sanitization.ts +0 -96
- /package/dist/{DataTable-A36PJG6N.js.map → DataTable-CYOHOX3O.js.map} +0 -0
- /package/dist/{UnifiedAuthProvider-CQDZRJIS.js.map → UnifiedAuthProvider-5E5TUNMS.js.map} +0 -0
- /package/dist/{api-TNIBJWLM.js.map → api-45XYYO2A.js.map} +0 -0
- /package/dist/{audit-T36HM7IM.js.map → audit-64X3VJXB.js.map} +0 -0
- /package/dist/{chunk-CTJRBUX2.js.map → chunk-2TWNJ46Y.js.map} +0 -0
- /package/dist/{chunk-ZV77RZMU.js.map → chunk-XARJS7CD.js.map} +0 -0
- /package/dist/{useInactivityTracker-MRUU55XI.js.map → useInactivityTracker-TO6ZOF35.js.map} +0 -0
- /package/examples/{public-pages → features/public-pages}/index.ts +0 -0
- /package/examples/{RBAC → features/rbac}/CompleteRBACExample.tsx +0 -0
- /package/examples/{RBAC → features/rbac}/EventBasedApp.tsx +0 -0
- /package/examples/{RBAC → features/rbac}/PermissionExample.tsx +0 -0
- /package/examples/{RBAC → features/rbac}/index.ts +0 -0
- /package/src/utils/{appConfig.ts → app/appConfig.ts} +0 -0
- /package/src/utils/{appNameResolver.simple.test.ts → app/appNameResolver.simple.test.ts} +0 -0
- /package/src/utils/{audit.ts → audit/audit.ts} +0 -0
- /package/src/utils/{organisationContext.test.ts → context/organisationContext.test.ts} +0 -0
- /package/src/utils/{cn.ts → core/cn.ts} +0 -0
- /package/src/utils/{debugLogger.ts → core/debugLogger.ts} +0 -0
- /package/src/utils/{dynamicUtils.ts → dynamic/dynamicUtils.ts} +0 -0
- /package/src/utils/{formatDate.test.ts → formatting/formatDate.test.ts} +0 -0
- /package/src/utils/{formatting.ts → formatting/formatting.ts} +0 -0
- /package/src/utils/{bundleAnalysis.ts → performance/bundleAnalysis.ts} +0 -0
- /package/src/utils/{performanceBenchmark.ts → performance/performanceBenchmark.ts} +0 -0
- /package/src/utils/{performanceBudgets.ts → performance/performanceBudgets.ts} +0 -0
- /package/src/utils/{permissionTypes.ts → permissions/permissionTypes.ts} +0 -0
- /package/src/utils/{permissionUtils.test.ts → permissions/permissionUtils.test.ts} +0 -0
- /package/src/utils/{permissionUtils.ts → permissions/permissionUtils.ts} +0 -0
- /package/src/utils/{auth-utils.ts → security/auth-utils.ts} +0 -0
- /package/src/utils/{secureDataAccess.test.ts → security/secureDataAccess.test.ts} +0 -0
- /package/src/utils/{secureDataAccess.ts → security/secureDataAccess.ts} +0 -0
- /package/src/utils/{secureStorage.ts → security/secureStorage.ts} +0 -0
- /package/src/utils/{securityMonitor.ts → security/securityMonitor.ts} +0 -0
- /package/src/{validation → utils/validation}/common.ts +0 -0
- /package/src/{validation → utils/validation}/passwordSchema.ts +0 -0
- /package/src/{validation → utils/validation}/user.ts +0 -0
- /package/src/utils/{validation.ts → validation/validation.ts} +0 -0
|
@@ -1,113 +0,0 @@
|
|
|
1
|
-
|
|
2
|
-
/**
|
|
3
|
-
* @file FormErrorSummary Component
|
|
4
|
-
* @package @jmruthers/pace-core
|
|
5
|
-
* @module Components/Form
|
|
6
|
-
* @since 0.1.0
|
|
7
|
-
*
|
|
8
|
-
* A form error summary component that displays validation errors in a user-friendly format.
|
|
9
|
-
* Provides a centralized location for users to see all form errors at once.
|
|
10
|
-
*
|
|
11
|
-
* Features:
|
|
12
|
-
* - Displays all form errors in one place
|
|
13
|
-
* - Configurable error display format
|
|
14
|
-
* - Optional field name display
|
|
15
|
-
* - Accessible error presentation
|
|
16
|
-
* - Consistent styling with Alert component
|
|
17
|
-
* - Automatic error filtering
|
|
18
|
-
* - Responsive design
|
|
19
|
-
*
|
|
20
|
-
* @example
|
|
21
|
-
* ```tsx
|
|
22
|
-
* // Basic error summary
|
|
23
|
-
* <FormErrorSummary errors={formErrors} />
|
|
24
|
-
*
|
|
25
|
-
* // With custom title
|
|
26
|
-
* <FormErrorSummary
|
|
27
|
-
* errors={formErrors}
|
|
28
|
-
* title="Please correct the following issues:"
|
|
29
|
-
* />
|
|
30
|
-
*
|
|
31
|
-
* // Show field names with errors
|
|
32
|
-
* <FormErrorSummary
|
|
33
|
-
* errors={formErrors}
|
|
34
|
-
* showFieldNames={true}
|
|
35
|
-
* />
|
|
36
|
-
*
|
|
37
|
-
* // In a form with React Hook Form
|
|
38
|
-
* const { formState: { errors } } = useForm();
|
|
39
|
-
*
|
|
40
|
-
* <Form onSubmit={handleSubmit}>
|
|
41
|
-
* <FormField name="email" label="Email" />
|
|
42
|
-
* <FormField name="password" label="Password" />
|
|
43
|
-
* <FormErrorSummary errors={errors} />
|
|
44
|
-
* <Button type="submit">Submit</Button>
|
|
45
|
-
* </Form>
|
|
46
|
-
* ```
|
|
47
|
-
*
|
|
48
|
-
* @accessibility
|
|
49
|
-
* - Uses Alert component for consistent accessibility
|
|
50
|
-
* - Proper heading hierarchy with h3
|
|
51
|
-
* - List structure for error items
|
|
52
|
-
* - Screen reader friendly error announcements
|
|
53
|
-
* - High contrast support via Alert component
|
|
54
|
-
*
|
|
55
|
-
* @dependencies
|
|
56
|
-
* - Alert component for consistent styling
|
|
57
|
-
* - React 18+ - Component framework
|
|
58
|
-
* - Tailwind CSS - Styling
|
|
59
|
-
*/
|
|
60
|
-
|
|
61
|
-
import { Alert } from '../Alert';
|
|
62
|
-
|
|
63
|
-
export interface FormErrorSummaryProps {
|
|
64
|
-
/** Form validation errors object */
|
|
65
|
-
errors?: Record<string, any>;
|
|
66
|
-
/** Title displayed above the error list */
|
|
67
|
-
title?: string;
|
|
68
|
-
/** Whether to show field names alongside error messages */
|
|
69
|
-
showFieldNames?: boolean;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* FormErrorSummary component for displaying form validation errors
|
|
74
|
-
*
|
|
75
|
-
* This component provides a user-friendly way to display all form errors
|
|
76
|
-
* in one centralized location, improving the user experience by making
|
|
77
|
-
* it clear what needs to be fixed.
|
|
78
|
-
*
|
|
79
|
-
* @param props - Component configuration
|
|
80
|
-
* @param props.errors - Form validation errors object
|
|
81
|
-
* @param props.title - Title displayed above the error list
|
|
82
|
-
* @param props.showFieldNames - Whether to show field names with errors
|
|
83
|
-
* @returns JSX.Element - The rendered error summary or null if no errors
|
|
84
|
-
*/
|
|
85
|
-
export function FormErrorSummary({
|
|
86
|
-
errors = {},
|
|
87
|
-
title = "Please fix the following errors:",
|
|
88
|
-
showFieldNames = false
|
|
89
|
-
}: FormErrorSummaryProps) {
|
|
90
|
-
const errorEntries = Object.entries(errors).filter(([, value]) => Boolean(value));
|
|
91
|
-
|
|
92
|
-
if (errorEntries.length === 0) {
|
|
93
|
-
return null;
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
return (
|
|
97
|
-
<Alert variant="destructive" className="mb-4">
|
|
98
|
-
<div>
|
|
99
|
-
<h3>{title}</h3>
|
|
100
|
-
<ul className="list-disc list-inside space-y-1">
|
|
101
|
-
{errorEntries.map(([field, error], index) => {
|
|
102
|
-
const message = typeof error === 'object' && error.message ? error.message : String(error);
|
|
103
|
-
return (
|
|
104
|
-
<li key={index} className="text-sm">
|
|
105
|
-
{showFieldNames ? `${field}: ${message}` : message}
|
|
106
|
-
</li>
|
|
107
|
-
);
|
|
108
|
-
})}
|
|
109
|
-
</ul>
|
|
110
|
-
</div>
|
|
111
|
-
</Alert>
|
|
112
|
-
);
|
|
113
|
-
}
|
|
@@ -1,127 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @file FormFieldset Component
|
|
3
|
-
* @package @jmruthers/pace-core
|
|
4
|
-
* @module Components/Form
|
|
5
|
-
* @since 0.1.0
|
|
6
|
-
*
|
|
7
|
-
* A form fieldset component that groups related form fields together with a legend
|
|
8
|
-
* and optional description. Provides semantic structure and accessibility for form sections.
|
|
9
|
-
*
|
|
10
|
-
* Features:
|
|
11
|
-
* - Semantic fieldset grouping for related form fields
|
|
12
|
-
* - Accessible legend with required indicator
|
|
13
|
-
* - Optional description text with proper ARIA association
|
|
14
|
-
* - Consistent styling with design system
|
|
15
|
-
* - Forwarded ref support
|
|
16
|
-
* - Customizable styling via className
|
|
17
|
-
* - Required field indication
|
|
18
|
-
* - Responsive design
|
|
19
|
-
*
|
|
20
|
-
* @example
|
|
21
|
-
* ```tsx
|
|
22
|
-
* // Basic fieldset
|
|
23
|
-
* <FormFieldset legend="Personal Information">
|
|
24
|
-
* <FormField name="firstName" label="First Name" />
|
|
25
|
-
* <FormField name="lastName" label="Last Name" />
|
|
26
|
-
* <FormField name="email" label="Email" />
|
|
27
|
-
* </FormFieldset>
|
|
28
|
-
*
|
|
29
|
-
* // Fieldset with description
|
|
30
|
-
* <FormFieldset
|
|
31
|
-
* legend="Contact Details"
|
|
32
|
-
* description="We'll use this information to contact you about your order"
|
|
33
|
-
* >
|
|
34
|
-
* <FormField name="phone" label="Phone Number" />
|
|
35
|
-
* <FormField name="address" label="Address" />
|
|
36
|
-
* </FormFieldset>
|
|
37
|
-
*
|
|
38
|
-
* // Required fieldset
|
|
39
|
-
* <FormFieldset
|
|
40
|
-
* legend="Account Settings"
|
|
41
|
-
* required={true}
|
|
42
|
-
* description="These settings are required for account creation"
|
|
43
|
-
* >
|
|
44
|
-
* <FormField name="username" label="Username" required />
|
|
45
|
-
* <FormField name="password" label="Password" type="password" required />
|
|
46
|
-
* </FormFieldset>
|
|
47
|
-
*
|
|
48
|
-
* // With custom styling
|
|
49
|
-
* <FormFieldset
|
|
50
|
-
* legend="Preferences"
|
|
51
|
-
* className="bg-sec-50 border-sec-200"
|
|
52
|
-
* >
|
|
53
|
-
* <FormField name="newsletter" label="Subscribe to newsletter" type="checkbox" />
|
|
54
|
-
* <FormField name="notifications" label="Enable notifications" type="checkbox" />
|
|
55
|
-
* </FormFieldset>
|
|
56
|
-
* ```
|
|
57
|
-
*
|
|
58
|
-
* @accessibility
|
|
59
|
-
* - Proper semantic HTML with fieldset and legend
|
|
60
|
-
* - ARIA describedby association for descriptions
|
|
61
|
-
* - Required field indication with asterisk
|
|
62
|
-
* - Screen reader friendly structure
|
|
63
|
-
* - Keyboard navigation support
|
|
64
|
-
* - WCAG 2.1 AA compliant
|
|
65
|
-
*
|
|
66
|
-
* @dependencies
|
|
67
|
-
* - React 18+ - Component framework and hooks
|
|
68
|
-
* - Tailwind CSS - Styling
|
|
69
|
-
* - cn utility - Class name merging
|
|
70
|
-
*/
|
|
71
|
-
|
|
72
|
-
import React from 'react';
|
|
73
|
-
import { cn } from '../../utils/cn';
|
|
74
|
-
|
|
75
|
-
export interface FormFieldsetProps extends React.FieldsetHTMLAttributes<HTMLFieldSetElement> {
|
|
76
|
-
/** The legend text for the fieldset */
|
|
77
|
-
legend: string;
|
|
78
|
-
/** Optional description text below the legend */
|
|
79
|
-
description?: string;
|
|
80
|
-
/** Whether the fieldset contains required fields */
|
|
81
|
-
required?: boolean;
|
|
82
|
-
/** Form fields to group within the fieldset */
|
|
83
|
-
children: React.ReactNode;
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
/**
|
|
87
|
-
* FormFieldset component for grouping related form fields
|
|
88
|
-
*
|
|
89
|
-
* This component provides semantic grouping for form fields with proper
|
|
90
|
-
* accessibility support and consistent styling. Use it to organize
|
|
91
|
-
* related form inputs into logical sections.
|
|
92
|
-
*
|
|
93
|
-
* @param props - Component configuration
|
|
94
|
-
* @param props.legend - The legend text for the fieldset
|
|
95
|
-
* @param props.description - Optional description text
|
|
96
|
-
* @param props.required - Whether the fieldset contains required fields
|
|
97
|
-
* @param props.children - Form fields to group
|
|
98
|
-
* @param ref - Forwarded ref to the fieldset element
|
|
99
|
-
* @returns JSX.Element - The rendered fieldset with legend and fields
|
|
100
|
-
*/
|
|
101
|
-
export const FormFieldset = React.forwardRef<HTMLFieldSetElement, FormFieldsetProps>(
|
|
102
|
-
({ legend, description, required, className, children, ...props }, ref) => {
|
|
103
|
-
const descriptionId = React.useId();
|
|
104
|
-
|
|
105
|
-
return (
|
|
106
|
-
<fieldset
|
|
107
|
-
ref={ref}
|
|
108
|
-
className={cn("space-y-4 border border-input rounded-lg p-4", className)}
|
|
109
|
-
aria-describedby={description ? descriptionId : undefined}
|
|
110
|
-
{...props}
|
|
111
|
-
>
|
|
112
|
-
<legend className="px-2">
|
|
113
|
-
{legend}
|
|
114
|
-
{required && <span className="text-destructive ml-1">*</span>}
|
|
115
|
-
</legend>
|
|
116
|
-
{description && (
|
|
117
|
-
<p id={descriptionId} className="text-muted-foreground">
|
|
118
|
-
{description}
|
|
119
|
-
</p>
|
|
120
|
-
)}
|
|
121
|
-
{children}
|
|
122
|
-
</fieldset>
|
|
123
|
-
);
|
|
124
|
-
}
|
|
125
|
-
);
|
|
126
|
-
|
|
127
|
-
FormFieldset.displayName = "FormFieldset";
|
|
@@ -1,198 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @file FormLiveRegion Component
|
|
3
|
-
* @package @jmruthers/pace-core
|
|
4
|
-
* @module Components/Form
|
|
5
|
-
* @since 0.1.0
|
|
6
|
-
*
|
|
7
|
-
* An accessible live region component for form validation announcements.
|
|
8
|
-
* Provides real-time feedback to screen readers about form validation states
|
|
9
|
-
* and error messages, improving accessibility for users with assistive technologies.
|
|
10
|
-
*
|
|
11
|
-
* Features:
|
|
12
|
-
* - Real-time form validation announcements
|
|
13
|
-
* - Configurable politeness levels (polite/assertive)
|
|
14
|
-
* - Customizable delay timing
|
|
15
|
-
* - Field-level and form-level announcements
|
|
16
|
-
* - Success message customization
|
|
17
|
-
* - Automatic error counting and field identification
|
|
18
|
-
* - Accessibility compliant with ARIA live regions
|
|
19
|
-
* - Integration with React Hook Form
|
|
20
|
-
*
|
|
21
|
-
* @example
|
|
22
|
-
* ```tsx
|
|
23
|
-
* // Basic live region
|
|
24
|
-
* const { control, formState } = useForm();
|
|
25
|
-
*
|
|
26
|
-
* <Form onSubmit={handleSubmit}>
|
|
27
|
-
* <FormField name="email" label="Email" />
|
|
28
|
-
* <FormField name="password" label="Password" />
|
|
29
|
-
* <FormLiveRegion form={{ control, formState }} />
|
|
30
|
-
* <Button type="submit">Submit</Button>
|
|
31
|
-
* </Form>
|
|
32
|
-
*
|
|
33
|
-
* // With custom configuration
|
|
34
|
-
* <FormLiveRegion
|
|
35
|
-
* form={{ control, formState }}
|
|
36
|
-
* politeness="assertive"
|
|
37
|
-
* delay={1000}
|
|
38
|
-
* successMessage="Your form was submitted successfully!"
|
|
39
|
-
* enableFieldAnnouncements={true}
|
|
40
|
-
* />
|
|
41
|
-
*
|
|
42
|
-
* // In a complex form
|
|
43
|
-
* function ComplexForm() {
|
|
44
|
-
* const methods = useForm({
|
|
45
|
-
* defaultValues: { name: '', email: '', message: '' }
|
|
46
|
-
* });
|
|
47
|
-
*
|
|
48
|
-
* return (
|
|
49
|
-
* <FormProvider {...methods}>
|
|
50
|
-
* <form onSubmit={methods.handleSubmit(onSubmit)}>
|
|
51
|
-
* <FormField name="name" label="Name" />
|
|
52
|
-
* <FormField name="email" label="Email" />
|
|
53
|
-
* <FormField name="message" label="Message" />
|
|
54
|
-
* <FormLiveRegion
|
|
55
|
-
* form={methods}
|
|
56
|
-
* politeness="polite"
|
|
57
|
-
* delay={300}
|
|
58
|
-
* />
|
|
59
|
-
* <Button type="submit">Send Message</Button>
|
|
60
|
-
* </form>
|
|
61
|
-
* </FormProvider>
|
|
62
|
-
* );
|
|
63
|
-
* }
|
|
64
|
-
* ```
|
|
65
|
-
*
|
|
66
|
-
* @accessibility
|
|
67
|
-
* - ARIA live region for screen reader announcements
|
|
68
|
-
* - Configurable politeness levels for different message types
|
|
69
|
-
* - Automatic error counting and field identification
|
|
70
|
-
* - Non-intrusive validation feedback
|
|
71
|
-
* - WCAG 2.1 AA compliant
|
|
72
|
-
* - Screen reader friendly error announcements
|
|
73
|
-
*
|
|
74
|
-
* @dependencies
|
|
75
|
-
* - react-hook-form - Form state management
|
|
76
|
-
* - React 18+ - Hooks and effects
|
|
77
|
-
* - Tailwind CSS - Styling
|
|
78
|
-
*/
|
|
79
|
-
|
|
80
|
-
import React, { useEffect, useState } from 'react';
|
|
81
|
-
import { UseFormReturn, FieldValues } from 'react-hook-form';
|
|
82
|
-
import { cn } from '../../utils/cn';
|
|
83
|
-
|
|
84
|
-
/**
|
|
85
|
-
* Props for the FormLiveRegion component
|
|
86
|
-
*/
|
|
87
|
-
export interface FormLiveRegionProps<T extends FieldValues> {
|
|
88
|
-
/** React Hook Form instance */
|
|
89
|
-
form: UseFormReturn<T>;
|
|
90
|
-
/** ARIA live region politeness level */
|
|
91
|
-
politeness?: 'polite' | 'assertive';
|
|
92
|
-
/** Delay before announcing messages (in ms) */
|
|
93
|
-
delay?: number;
|
|
94
|
-
/** Custom success message */
|
|
95
|
-
successMessage?: string;
|
|
96
|
-
/** Custom class name */
|
|
97
|
-
className?: string;
|
|
98
|
-
/** Enable field-level validation announcements */
|
|
99
|
-
enableFieldAnnouncements?: boolean;
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
/**
|
|
103
|
-
* FormLiveRegion component for accessible form validation announcements
|
|
104
|
-
*/
|
|
105
|
-
export function FormLiveRegion<T extends FieldValues>({
|
|
106
|
-
form,
|
|
107
|
-
politeness = 'polite',
|
|
108
|
-
delay = 500,
|
|
109
|
-
successMessage = 'Form submitted successfully',
|
|
110
|
-
className,
|
|
111
|
-
enableFieldAnnouncements = true
|
|
112
|
-
}: FormLiveRegionProps<T>): React.ReactElement {
|
|
113
|
-
const [message, setMessage] = useState<string>('');
|
|
114
|
-
const { formState } = form;
|
|
115
|
-
|
|
116
|
-
// Handle form-level announcements
|
|
117
|
-
useEffect(() => {
|
|
118
|
-
let timeoutId: NodeJS.Timeout;
|
|
119
|
-
|
|
120
|
-
if (formState.isSubmitting) {
|
|
121
|
-
setMessage('Submitting form...');
|
|
122
|
-
} else if (formState.isSubmitSuccessful) {
|
|
123
|
-
timeoutId = setTimeout(() => {
|
|
124
|
-
setMessage(successMessage);
|
|
125
|
-
}, delay);
|
|
126
|
-
} else if (formState.errors && Object.keys(formState.errors).length > 0) {
|
|
127
|
-
const errorCount = Object.keys(formState.errors).length;
|
|
128
|
-
const errorFields = Object.keys(formState.errors).join(', ');
|
|
129
|
-
setMessage(`Form has ${errorCount} error${errorCount === 1 ? '' : 's'} in: ${errorFields}`);
|
|
130
|
-
} else if (formState.isValid && formState.isDirty) {
|
|
131
|
-
setMessage('Form is valid');
|
|
132
|
-
} else {
|
|
133
|
-
setMessage('');
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
return () => {
|
|
137
|
-
if (timeoutId) {
|
|
138
|
-
clearTimeout(timeoutId);
|
|
139
|
-
}
|
|
140
|
-
};
|
|
141
|
-
}, [formState, delay, successMessage]);
|
|
142
|
-
|
|
143
|
-
// Handle field-level announcements
|
|
144
|
-
useEffect(() => {
|
|
145
|
-
if (!enableFieldAnnouncements) return;
|
|
146
|
-
|
|
147
|
-
let timeoutId: NodeJS.Timeout;
|
|
148
|
-
|
|
149
|
-
const fieldErrors = formState.errors;
|
|
150
|
-
const touchedFields = formState.touchedFields;
|
|
151
|
-
|
|
152
|
-
// Find newly touched fields with errors
|
|
153
|
-
const newErrors = Object.keys(fieldErrors).filter(field => {
|
|
154
|
-
// Type-safe access to touchedFields
|
|
155
|
-
const touchedField = touchedFields as Record<string, boolean>;
|
|
156
|
-
return touchedField[field] && fieldErrors[field as keyof typeof fieldErrors];
|
|
157
|
-
});
|
|
158
|
-
|
|
159
|
-
if (newErrors.length > 0) {
|
|
160
|
-
timeoutId = setTimeout(() => {
|
|
161
|
-
const errorField = newErrors[0];
|
|
162
|
-
const errorData = fieldErrors[errorField as keyof typeof fieldErrors];
|
|
163
|
-
const errorMessage = errorData?.message;
|
|
164
|
-
if (errorMessage) {
|
|
165
|
-
setMessage(`${errorField}: ${errorMessage}`);
|
|
166
|
-
}
|
|
167
|
-
}, delay);
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
return () => {
|
|
171
|
-
if (timeoutId) {
|
|
172
|
-
clearTimeout(timeoutId);
|
|
173
|
-
}
|
|
174
|
-
};
|
|
175
|
-
}, [formState.errors, formState.touchedFields, enableFieldAnnouncements, delay]);
|
|
176
|
-
|
|
177
|
-
if (!message) {
|
|
178
|
-
return <></>;
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
return (
|
|
182
|
-
<div
|
|
183
|
-
role="status"
|
|
184
|
-
aria-live={politeness}
|
|
185
|
-
aria-atomic="true"
|
|
186
|
-
className={cn('sr-only', className)}
|
|
187
|
-
style={{
|
|
188
|
-
position: 'absolute',
|
|
189
|
-
left: '-10000px',
|
|
190
|
-
width: '1px',
|
|
191
|
-
height: '1px',
|
|
192
|
-
overflow: 'hidden'
|
|
193
|
-
}}
|
|
194
|
-
>
|
|
195
|
-
{message}
|
|
196
|
-
</div>
|
|
197
|
-
);
|
|
198
|
-
}
|