@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.
- package/dist/{DataTable-EEUDXPE5.js → DataTable-GX3XERFJ.js} +8 -4
- package/dist/{DataTable-C1AEm9Cx.d.ts → DataTable-ltTFXHS3.d.ts} +3 -1
- package/dist/{chunk-VYG4AXYW.js → chunk-5EL3KHOQ.js} +2 -2
- package/dist/{chunk-ETEJVKYK.js → chunk-6CR3MRZN.js} +1426 -62
- package/dist/chunk-6CR3MRZN.js.map +1 -0
- package/dist/chunk-AUE24LVR.js +268 -0
- package/dist/chunk-AUE24LVR.js.map +1 -0
- package/dist/chunk-COBPIXXQ.js +379 -0
- package/dist/chunk-COBPIXXQ.js.map +1 -0
- package/dist/{chunk-EWKPTNPO.js → chunk-GSNM5D6H.js} +388 -86
- package/dist/chunk-GSNM5D6H.js.map +1 -0
- package/dist/{chunk-2V3Y6YBC.js → chunk-OEGRKULD.js} +1 -42
- package/dist/chunk-OEGRKULD.js.map +1 -0
- package/dist/chunk-OYRY44Q2.js +62 -0
- package/dist/chunk-OYRY44Q2.js.map +1 -0
- package/dist/{chunk-RRUYHORU.js → chunk-T3XIA4AJ.js} +297 -433
- package/dist/chunk-T3XIA4AJ.js.map +1 -0
- package/dist/{chunk-HEMJ4SUJ.js → chunk-TGDCLPP2.js} +11 -7
- package/dist/{chunk-HEMJ4SUJ.js.map → chunk-TGDCLPP2.js.map} +1 -1
- package/dist/{chunk-HNDFPXUU.js → chunk-U6JDHVC2.js} +6 -4
- package/dist/{chunk-HNDFPXUU.js.map → chunk-U6JDHVC2.js.map} +1 -1
- package/dist/{chunk-TIVL4UQ7.js → chunk-XJK2J4N6.js} +6 -4
- package/dist/{chunk-TIVL4UQ7.js.map → chunk-XJK2J4N6.js.map} +1 -1
- package/dist/components.d.ts +2 -2
- package/dist/components.js +21 -20
- package/dist/components.js.map +1 -1
- package/dist/hooks.d.ts +1 -1
- package/dist/hooks.js +7 -7
- package/dist/index.d.ts +2 -2
- package/dist/index.js +26 -25
- package/dist/index.js.map +1 -1
- package/dist/providers.js +8 -7
- package/dist/rbac/index.d.ts +806 -806
- package/dist/rbac/index.js +937 -1179
- package/dist/rbac/index.js.map +1 -1
- package/dist/{types-DiRQsGJs.d.ts → types-BRDU7N6w.d.ts} +12 -1
- package/dist/utils.d.ts +2 -2
- package/dist/utils.js +6 -6
- package/docs/api/classes/ErrorBoundary.md +1 -1
- package/docs/api/classes/PublicErrorBoundary.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 +1 -1
- package/docs/api/interfaces/DataTableAction.md +21 -8
- package/docs/api/interfaces/DataTableColumn.md +1 -1
- package/docs/api/interfaces/DataTableProps.md +46 -33
- package/docs/api/interfaces/DataTableToolbarButton.md +7 -7
- package/docs/api/interfaces/EmptyStateConfig.md +5 -5
- package/docs/api/interfaces/EventContextType.md +1 -1
- package/docs/api/interfaces/EventLogoProps.md +1 -1
- package/docs/api/interfaces/EventProviderProps.md +1 -1
- package/docs/api/interfaces/FileSizeLimits.md +1 -1
- package/docs/api/interfaces/FileUploadProps.md +1 -1
- package/docs/api/interfaces/FooterProps.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/NavigationItem.md +1 -1
- package/docs/api/interfaces/NavigationMenuProps.md +1 -1
- package/docs/api/interfaces/Organisation.md +1 -1
- package/docs/api/interfaces/OrganisationContextType.md +1 -1
- package/docs/api/interfaces/OrganisationMembership.md +2 -2
- package/docs/api/interfaces/OrganisationProviderProps.md +1 -1
- package/docs/api/interfaces/OrganisationSecurityError.md +1 -1
- package/docs/api/interfaces/PaceAppLayoutProps.md +1 -1
- package/docs/api/interfaces/PaceLoginPageProps.md +1 -1
- package/docs/api/interfaces/PaletteData.md +1 -1
- 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/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/ToastActionElement.md +1 -1
- package/docs/api/interfaces/ToastProps.md +1 -1
- package/docs/api/interfaces/UnifiedAuthContextType.md +1 -1
- package/docs/api/interfaces/UnifiedAuthProviderProps.md +1 -1
- package/docs/api/interfaces/UseInactivityTrackerOptions.md +1 -1
- package/docs/api/interfaces/UseInactivityTrackerReturn.md +1 -1
- package/docs/api/interfaces/UsePublicEventLogoOptions.md +1 -1
- package/docs/api/interfaces/UsePublicEventLogoReturn.md +1 -1
- package/docs/api/interfaces/UsePublicEventOptions.md +1 -1
- package/docs/api/interfaces/UsePublicEventReturn.md +1 -1
- package/docs/api/interfaces/UsePublicRouteParamsReturn.md +1 -1
- package/docs/api/interfaces/UserEventAccess.md +1 -1
- package/docs/api/interfaces/UserMenuProps.md +1 -1
- package/docs/api/interfaces/UserProfile.md +1 -1
- package/docs/api/modules.md +3 -3
- package/package.json +5 -2
- package/src/__tests__/REBUILD_PLAN.md +223 -0
- package/src/__tests__/TESTING_GUIDELINES.md +341 -0
- package/src/__tests__/fixtures/mocks.ts +93 -0
- package/src/__tests__/helpers/component-test-utils.tsx +145 -0
- package/src/__tests__/helpers/test-utils.tsx +117 -0
- package/src/__tests__/integration/UserProfile.test.tsx +128 -0
- package/src/__tests__/setup.ts +37 -225
- package/src/__tests__/templates/component.test.template.tsx +97 -75
- package/src/__tests__/templates/hook.test.template.ts +173 -0
- package/src/__tests__/types/test.types.ts +106 -0
- package/src/components/Alert/Alert.test.tsx +496 -0
- package/src/components/Avatar/Avatar.test.tsx +484 -0
- package/src/components/Button/Button.test.tsx +662 -0
- package/src/components/Card/Card.test.tsx +593 -0
- package/src/components/Checkbox/Checkbox.test.tsx +461 -0
- package/src/components/DataTable/DataTable.tsx +9 -1
- package/src/components/DataTable/components/AccessDeniedPage.tsx +168 -0
- package/src/components/DataTable/components/ActionButtons.tsx +18 -1
- package/src/components/DataTable/components/DataTableCore.tsx +97 -11
- package/src/components/DataTable/components/DataTableToolbar.tsx +22 -10
- package/src/components/DataTable/components/UnifiedTableBody.tsx +33 -4
- package/src/components/DataTable/examples/HierarchicalActionsExample.tsx +1 -0
- package/src/components/DataTable/examples/HierarchicalExample.tsx +3 -0
- package/src/components/DataTable/examples/InitialPageSizeExample.tsx +3 -0
- package/src/components/DataTable/examples/PerformanceExample.tsx +3 -0
- package/src/components/DataTable/types.ts +39 -1
- package/src/components/Dialog/Dialog.test.tsx +1139 -0
- package/src/components/ErrorBoundary/ErrorBoundary.test.tsx +752 -0
- package/src/components/FileUpload/FileUpload.test.tsx +665 -0
- package/src/hooks/useCounter.test.ts +135 -0
- package/src/rbac/index.ts +3 -3
- package/dist/chunk-2V3Y6YBC.js.map +0 -1
- package/dist/chunk-BEZRLNK3.js +0 -1744
- package/dist/chunk-BEZRLNK3.js.map +0 -1
- package/dist/chunk-ETEJVKYK.js.map +0 -1
- package/dist/chunk-EWKPTNPO.js.map +0 -1
- package/dist/chunk-OHXGNT3K.js +0 -21
- package/dist/chunk-OHXGNT3K.js.map +0 -1
- package/dist/chunk-RRUYHORU.js.map +0 -1
- package/src/__tests__/README.md +0 -404
- package/src/__tests__/debug-provider.unit.test.tsx +0 -67
- package/src/__tests__/e2e/workflows.test.tsx +0 -373
- package/src/__tests__/hybridPermissions.unit.test.tsx +0 -474
- package/src/__tests__/index.integration.test.ts +0 -491
- package/src/__tests__/mocks/MockAuthProvider-standalone.tsx +0 -47
- package/src/__tests__/mocks/MockAuthProvider.tsx +0 -63
- package/src/__tests__/mocks/enhancedSupabaseMock.ts +0 -252
- package/src/__tests__/mocks/index.test.ts +0 -23
- package/src/__tests__/mocks/index.ts +0 -16
- package/src/__tests__/mocks/mockAuth.ts +0 -155
- package/src/__tests__/mocks/mockSupabase.ts +0 -83
- package/src/__tests__/mocks/mockSupabaseClient.ts +0 -63
- package/src/__tests__/mocks/providers.tsx +0 -22
- package/src/__tests__/patterns/__tests__/testPatterns.test.ts +0 -394
- package/src/__tests__/patterns/testPatterns.ts +0 -124
- package/src/__tests__/performance/componentPerformance.performance.test.ts +0 -27
- package/src/__tests__/performance/index.ts +0 -24
- package/src/__tests__/performance/performanceValidation.performance.test.ts +0 -15
- package/src/__tests__/security/security.unit.test.tsx +0 -7
- package/src/__tests__/security/securityValidation.security.test.tsx +0 -153
- package/src/__tests__/setupTests.d.ts +0 -1
- package/src/__tests__/shared/componentTestUtils.tsx +0 -475
- package/src/__tests__/shared/errorHandlingTestUtils.ts +0 -107
- package/src/__tests__/shared/index.ts +0 -81
- package/src/__tests__/shared/integrationTestUtils.tsx +0 -375
- package/src/__tests__/shared/performanceTestUtils.tsx +0 -476
- package/src/__tests__/shared/testUtils.optimized.tsx +0 -685
- package/src/__tests__/simple.test.tsx +0 -20
- package/src/__tests__/test-utils/dataFactories.ts +0 -60
- package/src/__tests__/test-utils/index.ts +0 -6
- package/src/__tests__/typeSafety.unit.test.ts +0 -65
- package/src/__tests__/unifiedAuth.unit.test.tsx +0 -151
- package/src/__tests__/utils/accessibilityHelpers.ts +0 -254
- package/src/__tests__/utils/assertions.ts +0 -50
- package/src/__tests__/utils/deterministicHelpers.ts +0 -31
- package/src/__tests__/utils/edgeCaseConfig.test.ts +0 -75
- package/src/__tests__/utils/edgeCaseConfig.ts +0 -98
- package/src/__tests__/utils/mockHelpers.ts +0 -149
- package/src/__tests__/utils/mockLoader.ts +0 -101
- package/src/__tests__/utils/performanceHelpers.ts +0 -55
- package/src/__tests__/utils/performanceTestHelpers.ts +0 -68
- package/src/__tests__/utils/testDataFactories.ts +0 -28
- package/src/__tests__/utils/testIsolation.ts +0 -67
- package/src/__tests__/utils/visualTestHelpers.ts +0 -20
- package/src/__tests__/visual/__snapshots__/componentSnapshots.visual.test.tsx.snap +0 -68
- package/src/__tests__/visual/__snapshots__/componentVisuals.visual.test.tsx.snap +0 -14
- package/src/__tests__/visual/__snapshots__/visualRegression.test.tsx.snap +0 -217
- package/src/__tests__/visual/__snapshots__/visualRegression.visual.test.tsx.snap +0 -24
- package/src/__tests__/visual/componentSnapshots.visual.test.tsx +0 -33
- package/src/__tests__/visual/componentVisuals.visual.test.tsx +0 -12
- package/src/__tests__/visual/visualRegression.visual.test.tsx +0 -20
- package/src/components/Alert/__tests__/Alert.unit.test.tsx +0 -381
- package/src/components/Avatar/__tests__/Avatar.unit.test.tsx +0 -232
- package/src/components/Button/__tests__/Button.accessibility.test.tsx +0 -131
- package/src/components/Button/__tests__/Button.comprehensive.test.tsx +0 -721
- package/src/components/Button/__tests__/Button.unit.test.tsx +0 -189
- package/src/components/Button/__tests__/EventSelector.integration.test.tsx +0 -285
- package/src/components/Card/__tests__/Card.accessibility.test.tsx +0 -394
- package/src/components/Card/__tests__/Card.comprehensive.test.tsx +0 -599
- package/src/components/Card/__tests__/Card.integration.test.tsx +0 -673
- package/src/components/Card/__tests__/Card.performance.test.tsx +0 -546
- package/src/components/Card/__tests__/Card.unit.test.tsx +0 -330
- package/src/components/Card/__tests__/Card.visual.test.tsx +0 -599
- package/src/components/Card/__tests__/README.md +0 -211
- package/src/components/Checkbox/__tests__/Checkbox.unit.test.tsx +0 -520
- package/src/components/DataTable/__tests__/DataTable.errorHandling.test.tsx +0 -251
- package/src/components/DataTable/__tests__/DataTable.hierarchical.test.tsx +0 -680
- package/src/components/DataTable/__tests__/DataTable.infinite-loop.test.tsx +0 -323
- package/src/components/DataTable/__tests__/DataTable.integration.test.tsx +0 -716
- package/src/components/DataTable/__tests__/DataTable.performance.test.tsx +0 -589
- package/src/components/DataTable/__tests__/DataTable.permissions.test.tsx +0 -316
- package/src/components/DataTable/__tests__/DataTable.regressionFixes.test.tsx +0 -546
- package/src/components/DataTable/__tests__/DataTable.selection.controlled.test.tsx +0 -386
- package/src/components/DataTable/__tests__/DataTable.selection.test.tsx +0 -338
- package/src/components/DataTable/__tests__/DataTable.sorting.test.tsx +0 -321
- package/src/components/DataTable/__tests__/DataTable.userWorkflows.test.tsx +0 -320
- package/src/components/DataTable/__tests__/DataTable.workflowValidation.test.tsx +0 -583
- package/src/components/DataTable/__tests__/DataTable.workflows.test.tsx +0 -711
- package/src/components/DataTable/__tests__/performance-regression.test.tsx +0 -777
- package/src/components/DataTable/__tests__/performance.test.tsx +0 -365
- package/src/components/DataTable/components/__tests__/ActionButtons.unit.test.tsx +0 -150
- package/src/components/DataTable/components/__tests__/BulkOperationsDropdown.test.tsx +0 -224
- package/src/components/DataTable/components/__tests__/ColumnVisibilityDropdown.unit.test.tsx +0 -244
- package/src/components/DataTable/components/__tests__/DataTable.accessibility.test.tsx +0 -629
- package/src/components/DataTable/components/__tests__/DataTable.integration.test.tsx +0 -470
- package/src/components/DataTable/components/__tests__/DataTable.performance.test.tsx +0 -160
- package/src/components/DataTable/components/__tests__/DataTable.real.test.tsx +0 -251
- package/src/components/DataTable/components/__tests__/DataTable.security.test.tsx +0 -171
- package/src/components/DataTable/components/__tests__/DataTable.unit.test.tsx +0 -290
- package/src/components/DataTable/components/__tests__/DataTableBody.unit.test.tsx +0 -147
- package/src/components/DataTable/components/__tests__/DataTableErrorBoundary.unit.test.tsx +0 -182
- package/src/components/DataTable/components/__tests__/DataTableModals.unit.test.tsx +0 -123
- package/src/components/DataTable/components/__tests__/EditableRow.unit.test.tsx +0 -660
- package/src/components/DataTable/components/__tests__/EmptyState.unit.test.tsx +0 -256
- package/src/components/DataTable/components/__tests__/ExpandButton.test.tsx +0 -498
- package/src/components/DataTable/components/__tests__/FilterRow.unit.test.tsx +0 -112
- package/src/components/DataTable/components/__tests__/FilteringToggle.unit.test.tsx +0 -133
- package/src/components/DataTable/components/__tests__/GroupHeader.unit.test.tsx +0 -172
- package/src/components/DataTable/components/__tests__/GroupingDropdown.unit.test.tsx +0 -222
- package/src/components/DataTable/components/__tests__/ImportModal.unit.test.tsx +0 -780
- package/src/components/DataTable/components/__tests__/LoadingState.unit.test.tsx +0 -65
- package/src/components/DataTable/components/__tests__/PaginationControls.unit.test.tsx +0 -634
- package/src/components/DataTable/components/__tests__/StateComponents.unit.test.tsx +0 -48
- package/src/components/DataTable/components/__tests__/UnifiedTableBody.hierarchical.test.tsx +0 -541
- package/src/components/DataTable/components/__tests__/ViewRowModal.unit.test.tsx +0 -228
- package/src/components/DataTable/components/__tests__/VirtualizedDataTable.unit.test.tsx +0 -568
- package/src/components/DataTable/core/__tests__/ActionManager.unit.test.ts +0 -405
- package/src/components/DataTable/core/__tests__/ArchitectureIntegration.unit.test.tsx +0 -445
- package/src/components/DataTable/core/__tests__/ColumnFactory.unit.test.ts +0 -288
- package/src/components/DataTable/core/__tests__/ColumnManager.unit.test.ts +0 -623
- package/src/components/DataTable/core/__tests__/DataManager.unit.test.ts +0 -431
- package/src/components/DataTable/core/__tests__/DataTableContext.unit.test.tsx +0 -433
- package/src/components/DataTable/core/__tests__/LocalDataAdapter.unit.test.ts +0 -422
- package/src/components/DataTable/core/__tests__/PluginRegistry.unit.test.tsx +0 -207
- package/src/components/DataTable/core/__tests__/StateManager.unit.test.ts +0 -278
- package/src/components/DataTable/examples/__tests__/PerformanceExample.unit.test.tsx +0 -281
- package/src/components/DataTable/hooks/__tests__/useColumnOrderPersistence.unit.test.ts +0 -407
- package/src/components/DataTable/hooks/__tests__/useColumnReordering.unit.test.ts +0 -679
- package/src/components/DataTable/utils/__tests__/debugTools.unit.test.ts +0 -267
- package/src/components/DataTable/utils/__tests__/errorHandling.unit.test.ts +0 -467
- package/src/components/DataTable/utils/__tests__/exportUtils.unit.test.ts +0 -380
- package/src/components/DataTable/utils/__tests__/flexibleImport.unit.test.ts +0 -233
- package/src/components/DataTable/utils/__tests__/performanceUtils.unit.test.ts +0 -414
- package/src/components/Dialog/__tests__/Dialog.accessibility.test.tsx +0 -521
- package/src/components/Dialog/__tests__/Dialog.auto-size.example.tsx +0 -157
- package/src/components/Dialog/__tests__/Dialog.enhanced.test.tsx +0 -538
- package/src/components/Dialog/__tests__/Dialog.unit.test.tsx +0 -1373
- package/src/components/Dialog/examples/__tests__/SmartDialogExample.unit.test.tsx +0 -151
- package/src/components/Dialog/utils/__tests__/safeHtml.unit.test.ts +0 -611
- package/src/components/ErrorBoundary/__tests__/ErrorBoundary.accessibility.test.tsx +0 -517
- package/src/components/ErrorBoundary/__tests__/ErrorBoundary.integration.test.tsx +0 -572
- package/src/components/ErrorBoundary/__tests__/ErrorBoundary.unit.test.tsx +0 -579
- package/src/components/EventSelector/__tests__/EventSelector.test.tsx +0 -528
- package/src/components/FileUpload/__tests__/FileUpload.integration.test.tsx +0 -992
- package/src/components/FileUpload/__tests__/FileUpload.real.test.tsx +0 -927
- package/src/components/FileUpload/__tests__/FileUpload.test.tsx +0 -855
- package/src/components/FileUpload/__tests__/FileUpload.unit.test.tsx +0 -1311
- package/src/components/FileUpload/__tests__/FileUpload.unmocked.test.tsx +0 -937
- package/src/components/Footer/__tests__/Footer.accessibility.test.tsx +0 -359
- package/src/components/Footer/__tests__/Footer.integration.test.tsx +0 -353
- package/src/components/Footer/__tests__/Footer.performance.test.tsx +0 -309
- package/src/components/Footer/__tests__/Footer.unit.test.tsx +0 -309
- package/src/components/Footer/__tests__/Footer.visual.test.tsx +0 -335
- package/src/components/Form/__tests__/Form.accessibility.test.tsx +0 -820
- package/src/components/Form/__tests__/Form.unit.test.tsx +0 -305
- package/src/components/Form/__tests__/FormErrorSummary.unit.test.tsx +0 -285
- package/src/components/Form/__tests__/FormFieldset.unit.test.tsx +0 -241
- package/src/components/Header/__tests__/Header.accessibility.test.tsx +0 -382
- package/src/components/Header/__tests__/Header.comprehensive.test.tsx +0 -509
- package/src/components/Header/__tests__/Header.unit.test.tsx +0 -335
- package/src/components/InactivityWarningModal/InactivityWarningModal.test.tsx +0 -196
- package/src/components/InactivityWarningModal/__tests__/InactivityWarningModal.unit.test.tsx +0 -224
- package/src/components/Input/__tests__/Input.accessibility.test.tsx +0 -632
- package/src/components/Input/__tests__/Input.unit.test.tsx +0 -1121
- package/src/components/Label/__tests__/Label.accessibility.test.tsx +0 -239
- package/src/components/Label/__tests__/Label.unit.test.tsx +0 -331
- package/src/components/LoadingSpinner/__tests__/LoadingSpinner.accessibility.test.tsx +0 -116
- package/src/components/LoadingSpinner/__tests__/LoadingSpinner.unit.test.tsx +0 -144
- package/src/components/LoginForm/__tests__/LoginForm.accessibility.test.tsx +0 -201
- package/src/components/LoginForm/__tests__/LoginForm.unit.test.tsx +0 -119
- package/src/components/NavigationMenu/__tests__/NavigationMenu.accessibility.test.tsx +0 -378
- package/src/components/NavigationMenu/__tests__/NavigationMenu.enhanced.test.tsx +0 -768
- package/src/components/NavigationMenu/__tests__/NavigationMenu.integration.test.tsx +0 -576
- package/src/components/NavigationMenu/__tests__/NavigationMenu.performance.test.tsx +0 -585
- package/src/components/NavigationMenu/__tests__/NavigationMenu.real.component.test.tsx +0 -783
- package/src/components/NavigationMenu/__tests__/NavigationMenu.security.enhanced.test.tsx +0 -810
- package/src/components/NavigationMenu/__tests__/NavigationMenu.security.test.tsx +0 -494
- package/src/components/NavigationMenu/__tests__/NavigationMenu.unit.test.tsx +0 -331
- package/src/components/NavigationMenu/__tests__/NavigationMenu.userWorkflows.test.tsx +0 -347
- package/src/components/NavigationMenu/__tests__/NavigationMenu.workflows.test.tsx +0 -584
- package/src/components/OrganisationSelector/__tests__/OrganisationSelector.unit.test.tsx +0 -664
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.accessibility.test.tsx +0 -288
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.integration.test.tsx +0 -893
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.performance.test.tsx +0 -629
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.security.test.tsx +0 -782
- package/src/components/PaceAppLayout/__tests__/PaceAppLayout.unit.test.tsx +0 -904
- package/src/components/PaceLoginPage/__tests__/PaceLoginPage.accessibility.test.tsx +0 -463
- package/src/components/PaceLoginPage/__tests__/PaceLoginPage.integration.test.tsx +0 -586
- package/src/components/PaceLoginPage/__tests__/PaceLoginPage.unit.test.tsx +0 -533
- package/src/components/PasswordReset/__tests__/PasswordChangeForm.accessibility.test.tsx +0 -408
- package/src/components/PasswordReset/__tests__/PasswordChangeForm.unit.test.tsx +0 -561
- package/src/components/PasswordReset/__tests__/PasswordReset.integration.test.tsx +0 -304
- package/src/components/PasswordReset/__tests__/PasswordResetForm.accessibility.test.tsx +0 -20
- package/src/components/PasswordReset/__tests__/PasswordResetForm.unit.test.tsx +0 -523
- package/src/components/PasswordReset/__tests__/__mocks__/UnifiedAuthProvider.ts +0 -29
- package/src/components/Print/__tests__/Print.comprehensive.test.tsx +0 -331
- package/src/components/PrintButton/__tests__/PrintButton.unit.test.tsx +0 -429
- package/src/components/PrintButton/__tests__/PrintButtonGroup.unit.test.tsx +0 -277
- package/src/components/PrintButton/__tests__/PrintToolbar.unit.test.tsx +0 -264
- package/src/components/PrintCard/__tests__/PrintCard.unit.test.tsx +0 -233
- package/src/components/PrintCard/__tests__/PrintCardContent.test.tsx +0 -284
- package/src/components/PrintCard/__tests__/PrintCardGrid.unit.test.tsx +0 -214
- package/src/components/PrintCard/__tests__/PrintCardImage.unit.test.tsx +0 -264
- package/src/components/PrintDataTable/__tests__/PrintDataTable.unit.test.tsx +0 -361
- package/src/components/PrintDataTable/__tests__/PrintTableGroup.unit.test.tsx +0 -314
- package/src/components/PrintDataTable/__tests__/PrintTableRow.unit.test.tsx +0 -362
- package/src/components/PrintFooter/__tests__/PrintFooter.unit.test.tsx +0 -500
- package/src/components/PrintFooter/__tests__/PrintFooterContent.unit.test.tsx +0 -321
- package/src/components/PrintFooter/__tests__/PrintFooterInfo.unit.test.tsx +0 -335
- package/src/components/PrintFooter/__tests__/PrintPageNumber.unit.test.tsx +0 -340
- package/src/components/PrintGrid/__tests__/PrintGrid.unit.test.tsx +0 -340
- package/src/components/PrintGrid/__tests__/PrintGridBreakpoint.unit.test.tsx +0 -261
- package/src/components/PrintGrid/__tests__/PrintGridContainer.unit.test.tsx +0 -338
- package/src/components/PrintGrid/__tests__/PrintGridItem.unit.test.tsx +0 -338
- package/src/components/PrintHeader/__tests__/PrintCoverHeader.unit.test.tsx +0 -309
- package/src/components/PrintHeader/__tests__/PrintHeader.unit.test.tsx +0 -202
- package/src/components/PrintLayout/__tests__/PrintLayout.unit.test.tsx +0 -238
- package/src/components/PrintPageBreak/__tests__/PrintPageBreak.unit.test.tsx +0 -263
- package/src/components/PrintPageBreak/__tests__/PrintPageBreakGroup.unit.test.tsx +0 -239
- package/src/components/PrintPageBreak/__tests__/PrintPageBreakIndicator.unit.test.tsx +0 -235
- package/src/components/PrintSection/__tests__/PrintColumn.unit.test.tsx +0 -385
- package/src/components/PrintSection/__tests__/PrintDivider.unit.test.tsx +0 -373
- package/src/components/PrintSection/__tests__/PrintSection.unit.test.tsx +0 -390
- package/src/components/PrintSection/__tests__/PrintSectionContent.unit.test.tsx +0 -321
- package/src/components/PrintSection/__tests__/PrintSectionHeader.unit.test.tsx +0 -334
- package/src/components/PrintText/__tests__/PrintText.unit.test.tsx +0 -351
- package/src/components/Progress/__tests__/Progress.accessibility.test.tsx +0 -240
- package/src/components/Progress/__tests__/Progress.unit.test.tsx +0 -242
- package/src/components/PublicLayout/__tests__/EventLogo.test.tsx +0 -761
- package/src/components/PublicLayout/__tests__/PublicErrorBoundary.simplified.test.tsx +0 -228
- package/src/components/PublicLayout/__tests__/PublicErrorBoundary.test.tsx +0 -228
- package/src/components/PublicLayout/__tests__/PublicLoadingSpinner.test.tsx +0 -459
- package/src/components/PublicLayout/__tests__/PublicPageFooter.test.tsx +0 -362
- package/src/components/PublicLayout/__tests__/PublicPageHeader.test.tsx +0 -522
- package/src/components/PublicLayout/__tests__/PublicPageLayout.test.tsx +0 -599
- package/src/components/PublicLayout/__tests__/PublicPageProvider.test.tsx +0 -513
- package/src/components/RBAC/__tests__/PagePermissionGuard.unit.test.tsx +0 -683
- package/src/components/RBAC/__tests__/RBAC.integration.test.tsx +0 -573
- package/src/components/RBAC/__tests__/RBACGuard.unit.test.tsx +0 -467
- package/src/components/RBAC/__tests__/RBACProvider.accessibility.test.tsx +0 -475
- package/src/components/RBAC/__tests__/RBACProvider.advanced.test.tsx +0 -569
- package/src/components/RBAC/__tests__/RBACProvider.integration.test.tsx +0 -352
- package/src/components/RBAC/__tests__/RBACProvider.unit.test.tsx +0 -128
- package/src/components/RBAC/__tests__/RoleBasedContent.unit.test.tsx +0 -657
- package/src/components/Select/__tests__/SearchableSelect.unit.test.tsx +0 -437
- package/src/components/Select/__tests__/Select.accessibility.test.tsx +0 -1202
- package/src/components/Select/__tests__/Select.actual.test.tsx +0 -774
- package/src/components/Select/__tests__/Select.comprehensive.test.tsx +0 -837
- package/src/components/Select/__tests__/Select.enhanced.test.tsx +0 -1101
- package/src/components/Select/__tests__/Select.integration.test.tsx +0 -772
- package/src/components/Select/__tests__/Select.performance.test.tsx +0 -695
- package/src/components/Select/__tests__/Select.real-world.test.tsx +0 -1046
- package/src/components/Select/__tests__/Select.search-algorithms.test.tsx +0 -968
- package/src/components/Select/__tests__/Select.unit.test.tsx +0 -647
- package/src/components/Select/__tests__/Select.utils.test.tsx +0 -890
- package/src/components/Table/__tests__/Table.accessibility.test.tsx +0 -233
- package/src/components/Table/__tests__/Table.unit.test.tsx +0 -235
- package/src/components/Toast/__tests__/Toast.accessibility.test.tsx +0 -238
- package/src/components/Toast/__tests__/Toast.integration.test.tsx +0 -699
- package/src/components/Toast/__tests__/Toast.unit.test.tsx +0 -750
- package/src/components/Tooltip/__tests__/Tooltip.accessibility.test.tsx +0 -121
- package/src/components/Tooltip/__tests__/Tooltip.unit.test.tsx +0 -185
- package/src/components/UserMenu/__tests__/UserMenu.accessibility.test.tsx +0 -139
- package/src/components/UserMenu/__tests__/UserMenu.integration.test.tsx +0 -188
- package/src/components/UserMenu/__tests__/UserMenu.unit.test.tsx +0 -458
- package/src/components/__tests__/EdgeCaseTesting.enhanced.test.tsx +0 -524
- package/src/components/__tests__/ErrorTesting.enhanced.test.tsx +0 -455
- package/src/components/__tests__/SuperAdminGuard.test.tsx +0 -456
- package/src/components/__tests__/SuperAdminGuard.unit.test.tsx +0 -456
- package/src/components/examples/__tests__/PermissionExample.unit.test.tsx +0 -360
- package/src/hooks/__tests__/hooks.integration.test.tsx +0 -575
- package/src/hooks/__tests__/useApiFetch.unit.test.ts +0 -115
- package/src/hooks/__tests__/useComponentPerformance.unit.test.tsx +0 -133
- package/src/hooks/__tests__/useDebounce.unit.test.ts +0 -82
- package/src/hooks/__tests__/useFocusTrap.unit.test.tsx +0 -293
- package/src/hooks/__tests__/useInactivityTracker.unit.test.ts +0 -385
- package/src/hooks/__tests__/useOrganisationPermissions.unit.test.tsx +0 -286
- package/src/hooks/__tests__/useOrganisationSecurity.unit.test.tsx +0 -838
- package/src/hooks/__tests__/usePermissionCache.unit.test.ts +0 -627
- package/src/hooks/__tests__/useRBAC.unit.test.ts +0 -911
- package/src/hooks/__tests__/useSecureDataAccess.unit.test.tsx +0 -537
- package/src/hooks/__tests__/useToast.unit.test.tsx +0 -62
- package/src/hooks/__tests__/useZodForm.unit.test.tsx +0 -37
- package/src/hooks/public/__tests__/usePublicEvent.test.tsx +0 -397
- package/src/hooks/public/__tests__/usePublicEventLogo.test.tsx +0 -690
- package/src/hooks/public/__tests__/usePublicRouteParams.test.tsx +0 -449
- package/src/providers/__tests__/EventProvider.unit.test.tsx +0 -768
- package/src/providers/__tests__/OrganisationProvider.basic.test.tsx +0 -116
- package/src/providers/__tests__/OrganisationProvider.unit.test.tsx +0 -1312
- package/src/providers/__tests__/UnifiedAuthProvider.inactivity.test.tsx +0 -601
- package/src/providers/__tests__/UnifiedAuthProvider.unit.test.tsx +0 -683
- package/src/providers/__tests__/index.unit.test.ts +0 -78
- package/src/rbac/__tests__/PagePermissionGuard.test.tsx +0 -673
- package/src/rbac/__tests__/README.md +0 -170
- package/src/rbac/__tests__/RoleBasedRouter.test.tsx +0 -709
- package/src/rbac/__tests__/TestContext.tsx +0 -72
- package/src/rbac/__tests__/__mocks__/cache.ts +0 -144
- package/src/rbac/__tests__/__mocks__/supabase.ts +0 -152
- package/src/rbac/__tests__/adapters-hooks-comprehensive.test.tsx +0 -782
- package/src/rbac/__tests__/adapters-hooks.test.tsx +0 -561
- package/src/rbac/__tests__/adapters.comprehensive.test.tsx +0 -963
- package/src/rbac/__tests__/adapters.test.tsx +0 -444
- package/src/rbac/__tests__/api.test.ts +0 -620
- package/src/rbac/__tests__/audit-observability-comprehensive.test.ts +0 -792
- package/src/rbac/__tests__/audit-observability.test.ts +0 -549
- package/src/rbac/__tests__/audit.test.ts +0 -616
- package/src/rbac/__tests__/build-contract-compliance-simple.test.ts +0 -230
- package/src/rbac/__tests__/cache-invalidation-comprehensive.test.ts +0 -889
- package/src/rbac/__tests__/cache-invalidation.test.ts +0 -457
- package/src/rbac/__tests__/cache.test.ts +0 -458
- package/src/rbac/__tests__/components-navigation-guard.enhanced.test.tsx +0 -859
- package/src/rbac/__tests__/components-navigation-guard.test.tsx +0 -895
- package/src/rbac/__tests__/components-navigation-provider.test.tsx +0 -692
- package/src/rbac/__tests__/components-page-permission-guard.test.tsx +0 -673
- package/src/rbac/__tests__/components-page-permission-provider.test.tsx +0 -614
- package/src/rbac/__tests__/components-permission-enforcer.enhanced.fixed.test.tsx +0 -836
- package/src/rbac/__tests__/components-permission-enforcer.enhanced.test.tsx +0 -837
- package/src/rbac/__tests__/components-permission-enforcer.test.tsx +0 -825
- package/src/rbac/__tests__/components-role-based-router.test.tsx +0 -709
- package/src/rbac/__tests__/components-secure-data-provider.test.tsx +0 -607
- package/src/rbac/__tests__/config.test.ts +0 -583
- package/src/rbac/__tests__/core-logic-unit.test.ts +0 -190
- package/src/rbac/__tests__/core-permission-logic-comprehensive.test.ts +0 -1467
- package/src/rbac/__tests__/core-permission-logic-fixed.test.ts +0 -151
- package/src/rbac/__tests__/core-permission-logic-simple.test.ts +0 -968
- package/src/rbac/__tests__/core-permission-logic.test.ts +0 -966
- package/src/rbac/__tests__/edge-cases-comprehensive.test.ts +0 -988
- package/src/rbac/__tests__/edge-cases.test.ts +0 -654
- package/src/rbac/__tests__/engine.test.ts +0 -361
- package/src/rbac/__tests__/engine.unit.test.ts +0 -361
- package/src/rbac/__tests__/hooks.enhanced.test.tsx +0 -979
- package/src/rbac/__tests__/hooks.fixed.test.tsx +0 -475
- package/src/rbac/__tests__/hooks.test.tsx +0 -385
- package/src/rbac/__tests__/index.test.ts +0 -269
- package/src/rbac/__tests__/integration.enhanced.test.tsx +0 -824
- package/src/rbac/__tests__/page-permission-guard-super-admin.test.tsx +0 -261
- package/src/rbac/__tests__/performance.enhanced.test.tsx +0 -724
- package/src/rbac/__tests__/permissions.test.ts +0 -383
- package/src/rbac/__tests__/requires-event.test.ts +0 -330
- package/src/rbac/__tests__/scope-isolation-comprehensive.test.ts +0 -1349
- package/src/rbac/__tests__/scope-isolation.test.ts +0 -755
- package/src/rbac/__tests__/secure-client-rls-comprehensive.test.ts +0 -592
- package/src/rbac/__tests__/secure-client-rls.test.ts +0 -377
- package/src/rbac/__tests__/security.test.ts +0 -296
- package/src/rbac/__tests__/setup.ts +0 -228
- package/src/rbac/__tests__/test-utils-enhanced.tsx +0 -400
- package/src/rbac/__tests__/types.test.ts +0 -685
- package/src/rbac/components/__tests__/EnhancedNavigationMenu.test.tsx +0 -631
- package/src/rbac/components/__tests__/NavigationProvider.test.tsx +0 -667
- package/src/rbac/components/__tests__/PagePermissionProvider.test.tsx +0 -647
- package/src/rbac/components/__tests__/SecureDataProvider.test.tsx +0 -496
- package/src/rbac/testing/__tests__/index.test.tsx +0 -342
- package/src/rbac/utils/__tests__/eventContext.test.ts +0 -428
- package/src/rbac/utils/__tests__/eventContext.unit.test.ts +0 -428
- package/src/styles/__tests__/styles.unit.test.ts +0 -164
- package/src/test-dom-cleanup.test.tsx +0 -38
- package/src/theming/__tests__/README.md +0 -335
- package/src/theming/__tests__/runtime.accessibility.test.ts +0 -474
- package/src/theming/__tests__/runtime.error.test.ts +0 -616
- package/src/theming/__tests__/runtime.integration.test.ts +0 -376
- package/src/theming/__tests__/runtime.performance.test.ts +0 -411
- package/src/theming/__tests__/runtime.unit.test.ts +0 -470
- package/src/types/__tests__/database.unit.test.ts +0 -489
- package/src/types/__tests__/guards.unit.test.ts +0 -146
- package/src/types/__tests__/index.unit.test.ts +0 -77
- package/src/types/__tests__/organisation.unit.test.ts +0 -713
- package/src/types/__tests__/rbac.unit.test.ts +0 -621
- package/src/types/__tests__/security.unit.test.ts +0 -347
- package/src/types/__tests__/supabase.unit.test.ts +0 -658
- package/src/types/__tests__/theme.unit.test.ts +0 -218
- package/src/types/__tests__/unified.unit.test.ts +0 -537
- package/src/types/__tests__/validation.unit.test.ts +0 -616
- package/src/utils/__tests__/appConfig.unit.test.ts +0 -55
- package/src/utils/__tests__/appNameResolver.unit.test.ts +0 -137
- package/src/utils/__tests__/audit.unit.test.ts +0 -69
- package/src/utils/__tests__/auth-utils.unit.test.ts +0 -70
- package/src/utils/__tests__/bundleAnalysis.unit.test.ts +0 -317
- package/src/utils/__tests__/cn.unit.test.ts +0 -34
- package/src/utils/__tests__/deviceFingerprint.unit.test.ts +0 -480
- package/src/utils/__tests__/dynamicUtils.unit.test.ts +0 -322
- package/src/utils/__tests__/formatDate.unit.test.ts +0 -109
- package/src/utils/__tests__/formatting.unit.test.ts +0 -66
- package/src/utils/__tests__/index.unit.test.ts +0 -251
- package/src/utils/__tests__/lazyLoad.unit.test.tsx +0 -304
- package/src/utils/__tests__/organisationContext.unit.test.ts +0 -192
- package/src/utils/__tests__/performanceBudgets.unit.test.ts +0 -259
- package/src/utils/__tests__/permissionTypes.unit.test.ts +0 -250
- package/src/utils/__tests__/permissionUtils.unit.test.ts +0 -362
- package/src/utils/__tests__/sanitization.unit.test.ts +0 -346
- package/src/utils/__tests__/schemaUtils.unit.test.ts +0 -441
- package/src/utils/__tests__/secureDataAccess.unit.test.ts +0 -334
- package/src/utils/__tests__/secureErrors.unit.test.ts +0 -377
- package/src/utils/__tests__/secureStorage.unit.test.ts +0 -293
- package/src/utils/__tests__/security.unit.test.ts +0 -127
- package/src/utils/__tests__/securityMonitor.unit.test.ts +0 -280
- package/src/utils/__tests__/sessionTracking.unit.test.ts +0 -370
- package/src/utils/__tests__/validation.unit.test.ts +0 -84
- package/src/utils/__tests__/validationUtils.unit.test.ts +0 -571
- package/src/utils/print/__tests__/PrintDataProcessor.unit.test.ts +0 -219
- package/src/utils/print/__tests__/usePrintOptimization.unit.test.tsx +0 -353
- package/src/utils/storage/__tests__/config.unit.test.ts +0 -206
- package/src/utils/storage/__tests__/helpers.unit.test.ts +0 -648
- package/src/utils/storage/__tests__/index.unit.test.ts +0 -167
- package/src/utils/storage/__tests__/types.unit.test.ts +0 -441
- package/src/validation/__tests__/common.unit.test.ts +0 -101
- package/src/validation/__tests__/csrf.unit.test.ts +0 -302
- package/src/validation/__tests__/passwordSchema.unit.test.ts +0 -98
- package/src/validation/__tests__/sqlInjectionProtection.unit.test.ts +0 -466
- /package/dist/{DataTable-EEUDXPE5.js.map → DataTable-GX3XERFJ.js.map} +0 -0
- /package/dist/{chunk-VYG4AXYW.js.map → chunk-5EL3KHOQ.js.map} +0 -0
|
@@ -1,1101 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @file Enhanced Select Component Tests
|
|
3
|
-
* @package @jmruthers/pace-core
|
|
4
|
-
* @module Components/Select/Select.enhanced
|
|
5
|
-
* @since 0.1.0
|
|
6
|
-
*
|
|
7
|
-
* Enhanced tests for the Select component that complement existing test coverage
|
|
8
|
-
* with additional edge cases, advanced scenarios, and specific functionality testing.
|
|
9
|
-
* This file focuses on areas that may not be fully covered in existing tests.
|
|
10
|
-
*/
|
|
11
|
-
|
|
12
|
-
import React from 'react';
|
|
13
|
-
import { screen, fireEvent, waitFor } from '@testing-library/react';
|
|
14
|
-
import userEvent from '@testing-library/user-event';
|
|
15
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
16
|
-
import '@testing-library/jest-dom';
|
|
17
|
-
import { renderWithProviders, rerenderWithProviders } from '../../../__tests__/shared';
|
|
18
|
-
import {
|
|
19
|
-
Select,
|
|
20
|
-
SelectGroup,
|
|
21
|
-
SelectValue,
|
|
22
|
-
SelectTrigger,
|
|
23
|
-
SelectContent,
|
|
24
|
-
SelectLabel,
|
|
25
|
-
SelectItem,
|
|
26
|
-
SelectSeparator,
|
|
27
|
-
} from '../Select';
|
|
28
|
-
|
|
29
|
-
// Mock lucide-react icons
|
|
30
|
-
vi.mock('lucide-react', () => ({
|
|
31
|
-
Search: () => <div data-testid="search-icon">🔍</div>,
|
|
32
|
-
X: () => <div data-testid="clear-icon">✕</div>,
|
|
33
|
-
ChevronDown: () => <div data-testid="chevron-down-icon">▼</div>,
|
|
34
|
-
Check: () => <div data-testid="check-icon">✓</div>,
|
|
35
|
-
}));
|
|
36
|
-
|
|
37
|
-
// Enhanced mock for Radix UI Select with more realistic behavior
|
|
38
|
-
vi.mock('@radix-ui/react-select', async () => {
|
|
39
|
-
const actual = await vi.importActual('@radix-ui/react-select');
|
|
40
|
-
return {
|
|
41
|
-
...actual,
|
|
42
|
-
Root: ({ children, onValueChange, value, defaultValue, disabled, ...props }: any) => (
|
|
43
|
-
<div
|
|
44
|
-
data-testid="select-root"
|
|
45
|
-
data-value={value || defaultValue}
|
|
46
|
-
data-disabled={disabled}
|
|
47
|
-
{...props}
|
|
48
|
-
>
|
|
49
|
-
{children}
|
|
50
|
-
</div>
|
|
51
|
-
),
|
|
52
|
-
Trigger: React.forwardRef(({ children, disabled, onClick, ...props }: any, ref) => (
|
|
53
|
-
<button
|
|
54
|
-
ref={ref}
|
|
55
|
-
data-testid="select-trigger"
|
|
56
|
-
disabled={disabled}
|
|
57
|
-
onClick={(e) => {
|
|
58
|
-
if (!disabled) {
|
|
59
|
-
onClick?.(e);
|
|
60
|
-
// Simulate opening the dropdown
|
|
61
|
-
const event = new CustomEvent('select-open');
|
|
62
|
-
document.dispatchEvent(event);
|
|
63
|
-
}
|
|
64
|
-
}}
|
|
65
|
-
{...props}
|
|
66
|
-
>
|
|
67
|
-
{children}
|
|
68
|
-
</button>
|
|
69
|
-
)),
|
|
70
|
-
Content: ({ children, searchable, position, ...props }: any) => {
|
|
71
|
-
const [searchValue, setSearchValue] = React.useState('');
|
|
72
|
-
const [isOpen, setIsOpen] = React.useState(false);
|
|
73
|
-
|
|
74
|
-
React.useEffect(() => {
|
|
75
|
-
const handleOpen = () => setIsOpen(true);
|
|
76
|
-
const handleClose = () => setIsOpen(false);
|
|
77
|
-
|
|
78
|
-
document.addEventListener('select-open', handleOpen);
|
|
79
|
-
document.addEventListener('select-close', handleClose);
|
|
80
|
-
|
|
81
|
-
return () => {
|
|
82
|
-
document.removeEventListener('select-open', handleOpen);
|
|
83
|
-
document.removeEventListener('select-close', handleClose);
|
|
84
|
-
};
|
|
85
|
-
}, []);
|
|
86
|
-
|
|
87
|
-
return (
|
|
88
|
-
<div
|
|
89
|
-
data-testid="select-content"
|
|
90
|
-
data-searchable={searchable}
|
|
91
|
-
data-position={position}
|
|
92
|
-
data-open={isOpen}
|
|
93
|
-
{...props}
|
|
94
|
-
>
|
|
95
|
-
{searchable && (
|
|
96
|
-
<div data-testid="search-container">
|
|
97
|
-
<div data-testid="search-icon">🔍</div>
|
|
98
|
-
<input
|
|
99
|
-
data-testid="select-search-input"
|
|
100
|
-
type="text"
|
|
101
|
-
placeholder="Search..."
|
|
102
|
-
aria-label="Search options"
|
|
103
|
-
value={searchValue}
|
|
104
|
-
onChange={(e) => setSearchValue(e.target.value)}
|
|
105
|
-
onKeyDown={(e) => {
|
|
106
|
-
if (e.key === 'Escape') {
|
|
107
|
-
setSearchValue('');
|
|
108
|
-
}
|
|
109
|
-
}}
|
|
110
|
-
/>
|
|
111
|
-
{searchValue && (
|
|
112
|
-
<button
|
|
113
|
-
data-testid="select-clear-search"
|
|
114
|
-
aria-label="Clear search"
|
|
115
|
-
type="button"
|
|
116
|
-
onClick={() => setSearchValue('')}
|
|
117
|
-
>
|
|
118
|
-
✕
|
|
119
|
-
</button>
|
|
120
|
-
)}
|
|
121
|
-
</div>
|
|
122
|
-
)}
|
|
123
|
-
<div data-testid="select-viewport">
|
|
124
|
-
{children}
|
|
125
|
-
</div>
|
|
126
|
-
</div>
|
|
127
|
-
);
|
|
128
|
-
},
|
|
129
|
-
Viewport: ({ children, ...props }: any) => (
|
|
130
|
-
<div data-testid="select-viewport" {...props}>
|
|
131
|
-
{children}
|
|
132
|
-
</div>
|
|
133
|
-
),
|
|
134
|
-
Item: ({ children, value, disabled, onClick, ...props }: any) => (
|
|
135
|
-
<div
|
|
136
|
-
data-testid="select-item"
|
|
137
|
-
role="option"
|
|
138
|
-
data-value={value}
|
|
139
|
-
aria-disabled={disabled}
|
|
140
|
-
onClick={() => {
|
|
141
|
-
if (!disabled) {
|
|
142
|
-
onClick?.();
|
|
143
|
-
// Simulate value change
|
|
144
|
-
const event = new CustomEvent('select-value-change', { detail: value });
|
|
145
|
-
document.dispatchEvent(event);
|
|
146
|
-
}
|
|
147
|
-
}}
|
|
148
|
-
{...props}
|
|
149
|
-
>
|
|
150
|
-
{children}
|
|
151
|
-
</div>
|
|
152
|
-
),
|
|
153
|
-
ItemText: ({ children, ...props }: any) => (
|
|
154
|
-
<span data-testid="select-item-text" {...props}>
|
|
155
|
-
{children}
|
|
156
|
-
</span>
|
|
157
|
-
),
|
|
158
|
-
ItemIndicator: ({ children, ...props }: any) => (
|
|
159
|
-
<span data-testid="select-item-indicator" {...props}>
|
|
160
|
-
{children}
|
|
161
|
-
</span>
|
|
162
|
-
),
|
|
163
|
-
Value: ({ children, placeholder, ...props }: any) => (
|
|
164
|
-
<span data-testid="select-value" data-placeholder={placeholder} {...props}>
|
|
165
|
-
{children || placeholder}
|
|
166
|
-
</span>
|
|
167
|
-
),
|
|
168
|
-
Label: ({ children, ...props }: any) => (
|
|
169
|
-
<div data-testid="select-label" {...props}>
|
|
170
|
-
{children}
|
|
171
|
-
</div>
|
|
172
|
-
),
|
|
173
|
-
Group: ({ children, ...props }: any) => (
|
|
174
|
-
<div data-testid="select-group" {...props}>
|
|
175
|
-
{children}
|
|
176
|
-
</div>
|
|
177
|
-
),
|
|
178
|
-
Separator: ({ ...props }: any) => (
|
|
179
|
-
<div data-testid="select-separator" {...props} />
|
|
180
|
-
),
|
|
181
|
-
Icon: ({ children, ...props }: any) => (
|
|
182
|
-
<span data-testid="select-icon" {...props}>
|
|
183
|
-
{children}
|
|
184
|
-
</span>
|
|
185
|
-
),
|
|
186
|
-
Portal: ({ children }: any) => <>{children}</>,
|
|
187
|
-
};
|
|
188
|
-
});
|
|
189
|
-
|
|
190
|
-
describe('Select Component - Enhanced Tests', () => {
|
|
191
|
-
const user = userEvent.setup();
|
|
192
|
-
|
|
193
|
-
beforeEach(() => {
|
|
194
|
-
vi.clearAllMocks();
|
|
195
|
-
});
|
|
196
|
-
|
|
197
|
-
afterEach(() => {
|
|
198
|
-
vi.clearAllTimers();
|
|
199
|
-
});
|
|
200
|
-
|
|
201
|
-
describe('Advanced Search Functionality', () => {
|
|
202
|
-
it('handles complex search patterns with special characters', async () => {
|
|
203
|
-
renderWithProviders(
|
|
204
|
-
<Select>
|
|
205
|
-
<SelectTrigger>
|
|
206
|
-
<SelectValue placeholder="Select an option" />
|
|
207
|
-
</SelectTrigger>
|
|
208
|
-
<SelectContent searchable searchStrategy="contains">
|
|
209
|
-
<SelectItem value="c++">C++ Programming</SelectItem>
|
|
210
|
-
<SelectItem value="c#">C# Programming</SelectItem>
|
|
211
|
-
<SelectItem value="react-js">React.js Framework</SelectItem>
|
|
212
|
-
<SelectItem value="node-js">Node.js Runtime</SelectItem>
|
|
213
|
-
</SelectContent>
|
|
214
|
-
</Select>
|
|
215
|
-
);
|
|
216
|
-
|
|
217
|
-
// Click trigger to open the select
|
|
218
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
219
|
-
await user.click(trigger);
|
|
220
|
-
|
|
221
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
222
|
-
|
|
223
|
-
// Test special characters
|
|
224
|
-
fireEvent.change(searchInput, { target: { value: '++' } });
|
|
225
|
-
expect(searchInput).toHaveValue('++');
|
|
226
|
-
|
|
227
|
-
await user.clear(searchInput);
|
|
228
|
-
fireEvent.change(searchInput, { target: { value: '#' } });
|
|
229
|
-
expect(searchInput).toHaveValue('#');
|
|
230
|
-
|
|
231
|
-
await user.clear(searchInput);
|
|
232
|
-
fireEvent.change(searchInput, { target: { value: '.js' } });
|
|
233
|
-
expect(searchInput).toHaveValue('.js');
|
|
234
|
-
});
|
|
235
|
-
|
|
236
|
-
it('handles unicode and emoji search correctly', async () => {
|
|
237
|
-
renderWithProviders(
|
|
238
|
-
<Select>
|
|
239
|
-
<SelectTrigger>
|
|
240
|
-
<SelectValue placeholder="Select an option" />
|
|
241
|
-
</SelectTrigger>
|
|
242
|
-
<SelectContent searchable>
|
|
243
|
-
<SelectItem value="apple">🍎 Apple</SelectItem>
|
|
244
|
-
<SelectItem value="banana">🍌 Banana</SelectItem>
|
|
245
|
-
<SelectItem value="cherry">🍒 Cherry</SelectItem>
|
|
246
|
-
<SelectItem value="grapes">🍇 Grapes</SelectItem>
|
|
247
|
-
</SelectContent>
|
|
248
|
-
</Select>
|
|
249
|
-
);
|
|
250
|
-
|
|
251
|
-
// Click trigger to open the select
|
|
252
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
253
|
-
await user.click(trigger);
|
|
254
|
-
|
|
255
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
256
|
-
|
|
257
|
-
// Test emoji search
|
|
258
|
-
fireEvent.change(searchInput, { target: { value: '🍎' } });
|
|
259
|
-
expect(searchInput).toHaveValue('🍎');
|
|
260
|
-
|
|
261
|
-
await user.clear(searchInput);
|
|
262
|
-
fireEvent.change(searchInput, { target: { value: 'Apple' } });
|
|
263
|
-
expect(searchInput).toHaveValue('Apple');
|
|
264
|
-
});
|
|
265
|
-
|
|
266
|
-
it('handles case-sensitive search correctly', async () => {
|
|
267
|
-
renderWithProviders(
|
|
268
|
-
<Select>
|
|
269
|
-
<SelectTrigger>
|
|
270
|
-
<SelectValue placeholder="Select an option" />
|
|
271
|
-
</SelectTrigger>
|
|
272
|
-
<SelectContent searchable caseSensitive>
|
|
273
|
-
<SelectItem value="Apple">Apple</SelectItem>
|
|
274
|
-
<SelectItem value="apple">apple</SelectItem>
|
|
275
|
-
<SelectItem value="APPLE">APPLE</SelectItem>
|
|
276
|
-
<SelectItem value="aPpLe">aPpLe</SelectItem>
|
|
277
|
-
</SelectContent>
|
|
278
|
-
</Select>
|
|
279
|
-
);
|
|
280
|
-
|
|
281
|
-
// Click trigger to open the select
|
|
282
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
283
|
-
await user.click(trigger);
|
|
284
|
-
|
|
285
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
286
|
-
|
|
287
|
-
fireEvent.change(searchInput, { target: { value: 'Apple' } });
|
|
288
|
-
expect(searchInput).toHaveValue('Apple');
|
|
289
|
-
|
|
290
|
-
await user.clear(searchInput);
|
|
291
|
-
fireEvent.change(searchInput, { target: { value: 'apple' } });
|
|
292
|
-
expect(searchInput).toHaveValue('apple');
|
|
293
|
-
});
|
|
294
|
-
|
|
295
|
-
it('handles fuzzy search with typos', async () => {
|
|
296
|
-
renderWithProviders(
|
|
297
|
-
<Select>
|
|
298
|
-
<SelectTrigger>
|
|
299
|
-
<SelectValue placeholder="Select an option" />
|
|
300
|
-
</SelectTrigger>
|
|
301
|
-
<SelectContent searchable searchStrategy="fuzzy">
|
|
302
|
-
<SelectItem value="javascript">JavaScript</SelectItem>
|
|
303
|
-
<SelectItem value="typescript">TypeScript</SelectItem>
|
|
304
|
-
<SelectItem value="python">Python</SelectItem>
|
|
305
|
-
<SelectItem value="react">React</SelectItem>
|
|
306
|
-
</SelectContent>
|
|
307
|
-
</Select>
|
|
308
|
-
);
|
|
309
|
-
|
|
310
|
-
// Click trigger to open the select
|
|
311
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
312
|
-
await user.click(trigger);
|
|
313
|
-
|
|
314
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
315
|
-
|
|
316
|
-
// Test fuzzy matching with typos
|
|
317
|
-
fireEvent.change(searchInput, { target: { value: 'javascrpt' } }); // Missing 'i'
|
|
318
|
-
expect(searchInput).toHaveValue('javascrpt');
|
|
319
|
-
|
|
320
|
-
await user.clear(searchInput);
|
|
321
|
-
fireEvent.change(searchInput, { target: { value: 'typescrpt' } }); // Missing 'i'
|
|
322
|
-
expect(searchInput).toHaveValue('typescrpt');
|
|
323
|
-
|
|
324
|
-
await user.clear(searchInput);
|
|
325
|
-
fireEvent.change(searchInput, { target: { value: 'pythn' } }); // Missing 'o'
|
|
326
|
-
expect(searchInput).toHaveValue('pythn');
|
|
327
|
-
});
|
|
328
|
-
|
|
329
|
-
it('handles exact match search correctly', async () => {
|
|
330
|
-
renderWithProviders(
|
|
331
|
-
<Select>
|
|
332
|
-
<SelectTrigger>
|
|
333
|
-
<SelectValue placeholder="Select an option" />
|
|
334
|
-
</SelectTrigger>
|
|
335
|
-
<SelectContent searchable searchStrategy="exact">
|
|
336
|
-
<SelectItem value="react">React</SelectItem>
|
|
337
|
-
<SelectItem value="react-native">React Native</SelectItem>
|
|
338
|
-
<SelectItem value="react-query">React Query</SelectItem>
|
|
339
|
-
</SelectContent>
|
|
340
|
-
</Select>
|
|
341
|
-
);
|
|
342
|
-
|
|
343
|
-
// Click trigger to open the select
|
|
344
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
345
|
-
await user.click(trigger);
|
|
346
|
-
|
|
347
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
348
|
-
|
|
349
|
-
fireEvent.change(searchInput, { target: { value: 'React' } });
|
|
350
|
-
expect(searchInput).toHaveValue('React');
|
|
351
|
-
|
|
352
|
-
await user.clear(searchInput);
|
|
353
|
-
fireEvent.change(searchInput, { target: { value: 'React Native' } });
|
|
354
|
-
expect(searchInput).toHaveValue('React Native');
|
|
355
|
-
});
|
|
356
|
-
|
|
357
|
-
it('handles startsWith search correctly', async () => {
|
|
358
|
-
renderWithProviders(
|
|
359
|
-
<Select>
|
|
360
|
-
<SelectTrigger>
|
|
361
|
-
<SelectValue placeholder="Select an option" />
|
|
362
|
-
</SelectTrigger>
|
|
363
|
-
<SelectContent searchable searchStrategy="startsWith">
|
|
364
|
-
<SelectItem value="react">React</SelectItem>
|
|
365
|
-
<SelectItem value="react-native">React Native</SelectItem>
|
|
366
|
-
<SelectItem value="react-query">React Query</SelectItem>
|
|
367
|
-
<SelectItem value="vue">Vue.js</SelectItem>
|
|
368
|
-
</SelectContent>
|
|
369
|
-
</Select>
|
|
370
|
-
);
|
|
371
|
-
|
|
372
|
-
// Click trigger to open the select
|
|
373
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
374
|
-
await user.click(trigger);
|
|
375
|
-
|
|
376
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
377
|
-
|
|
378
|
-
fireEvent.change(searchInput, { target: { value: 'React' } });
|
|
379
|
-
expect(searchInput).toHaveValue('React');
|
|
380
|
-
|
|
381
|
-
await user.clear(searchInput);
|
|
382
|
-
fireEvent.change(searchInput, { target: { value: 'Vue' } });
|
|
383
|
-
expect(searchInput).toHaveValue('Vue');
|
|
384
|
-
});
|
|
385
|
-
});
|
|
386
|
-
|
|
387
|
-
describe('Advanced Keyboard Navigation', () => {
|
|
388
|
-
it('handles complex keyboard shortcuts', async () => {
|
|
389
|
-
renderWithProviders(
|
|
390
|
-
<Select>
|
|
391
|
-
<SelectTrigger>
|
|
392
|
-
<SelectValue placeholder="Select an option" />
|
|
393
|
-
</SelectTrigger>
|
|
394
|
-
<SelectContent searchable>
|
|
395
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
396
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
397
|
-
<SelectItem value="option3">Option 3</SelectItem>
|
|
398
|
-
</SelectContent>
|
|
399
|
-
</Select>
|
|
400
|
-
);
|
|
401
|
-
|
|
402
|
-
// Click trigger to open the select
|
|
403
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
404
|
-
await user.click(trigger);
|
|
405
|
-
|
|
406
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
407
|
-
|
|
408
|
-
// Focus the search input first
|
|
409
|
-
searchInput.focus();
|
|
410
|
-
|
|
411
|
-
// Wait for focus to be set
|
|
412
|
-
await waitFor(() => {
|
|
413
|
-
expect(searchInput).toHaveFocus();
|
|
414
|
-
});
|
|
415
|
-
|
|
416
|
-
// Test that search input maintains focus
|
|
417
|
-
expect(searchInput).toHaveFocus();
|
|
418
|
-
|
|
419
|
-
fireEvent.change(searchInput, { target: { value: 'test' } });
|
|
420
|
-
await user.keyboard('{Escape}');
|
|
421
|
-
expect(searchInput).toHaveValue('');
|
|
422
|
-
});
|
|
423
|
-
|
|
424
|
-
it('handles rapid keyboard input', async () => {
|
|
425
|
-
renderWithProviders(
|
|
426
|
-
<Select>
|
|
427
|
-
<SelectTrigger>
|
|
428
|
-
<SelectValue placeholder="Select an option" />
|
|
429
|
-
</SelectTrigger>
|
|
430
|
-
<SelectContent searchable>
|
|
431
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
432
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
433
|
-
</SelectContent>
|
|
434
|
-
</Select>
|
|
435
|
-
);
|
|
436
|
-
|
|
437
|
-
// Click trigger to open the select
|
|
438
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
439
|
-
await user.click(trigger);
|
|
440
|
-
|
|
441
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
442
|
-
|
|
443
|
-
// Rapid typing
|
|
444
|
-
fireEvent.change(searchInput, { target: { value: 'abcdefghijklmnopqrstuvwxyz' } });
|
|
445
|
-
expect(searchInput).toHaveValue('abcdefghijklmnopqrstuvwxyz');
|
|
446
|
-
|
|
447
|
-
// Rapid clearing
|
|
448
|
-
await user.clear(searchInput);
|
|
449
|
-
expect(searchInput).toHaveValue('');
|
|
450
|
-
});
|
|
451
|
-
|
|
452
|
-
it('handles keyboard navigation in search results', async () => {
|
|
453
|
-
renderWithProviders(
|
|
454
|
-
<Select>
|
|
455
|
-
<SelectTrigger>
|
|
456
|
-
<SelectValue placeholder="Select an option" />
|
|
457
|
-
</SelectTrigger>
|
|
458
|
-
<SelectContent searchable>
|
|
459
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
460
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
461
|
-
<SelectItem value="option3">Option 3</SelectItem>
|
|
462
|
-
</SelectContent>
|
|
463
|
-
</Select>
|
|
464
|
-
);
|
|
465
|
-
|
|
466
|
-
// Click trigger to open the select
|
|
467
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
468
|
-
await user.click(trigger);
|
|
469
|
-
|
|
470
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
471
|
-
|
|
472
|
-
// Navigate with arrow keys
|
|
473
|
-
await user.keyboard('{ArrowDown}');
|
|
474
|
-
await user.keyboard('{ArrowUp}');
|
|
475
|
-
await user.keyboard('{ArrowLeft}');
|
|
476
|
-
await user.keyboard('{ArrowRight}');
|
|
477
|
-
|
|
478
|
-
// Test that no errors occur
|
|
479
|
-
expect(searchInput).toBeInTheDocument();
|
|
480
|
-
});
|
|
481
|
-
});
|
|
482
|
-
|
|
483
|
-
describe('Advanced Search Configuration', () => {
|
|
484
|
-
it('handles custom search fields configuration', async () => {
|
|
485
|
-
renderWithProviders(
|
|
486
|
-
<Select>
|
|
487
|
-
<SelectTrigger>
|
|
488
|
-
<SelectValue placeholder="Select an option" />
|
|
489
|
-
</SelectTrigger>
|
|
490
|
-
<SelectContent searchable searchFields={['value']}>
|
|
491
|
-
<SelectItem value="react">React Framework</SelectItem>
|
|
492
|
-
<SelectItem value="vue">Vue Framework</SelectItem>
|
|
493
|
-
<SelectItem value="angular">Angular Framework</SelectItem>
|
|
494
|
-
</SelectContent>
|
|
495
|
-
</Select>
|
|
496
|
-
);
|
|
497
|
-
|
|
498
|
-
// Click trigger to open the select
|
|
499
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
500
|
-
await user.click(trigger);
|
|
501
|
-
|
|
502
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
503
|
-
|
|
504
|
-
fireEvent.change(searchInput, { target: { value: 'react' } });
|
|
505
|
-
expect(searchInput).toHaveValue('react');
|
|
506
|
-
|
|
507
|
-
await user.clear(searchInput);
|
|
508
|
-
fireEvent.change(searchInput, { target: { value: 'vue' } });
|
|
509
|
-
expect(searchInput).toHaveValue('vue');
|
|
510
|
-
});
|
|
511
|
-
|
|
512
|
-
it('handles custom filter function', async () => {
|
|
513
|
-
const customFilter = vi.fn((options, searchTerm) =>
|
|
514
|
-
options.filter(option =>
|
|
515
|
-
option.label.toLowerCase().includes(searchTerm.toLowerCase()) ||
|
|
516
|
-
option.value.toLowerCase().includes(searchTerm.toLowerCase())
|
|
517
|
-
)
|
|
518
|
-
);
|
|
519
|
-
|
|
520
|
-
renderWithProviders(
|
|
521
|
-
<Select>
|
|
522
|
-
<SelectTrigger>
|
|
523
|
-
<SelectValue placeholder="Select an option" />
|
|
524
|
-
</SelectTrigger>
|
|
525
|
-
<SelectContent searchable filterFunction={customFilter}>
|
|
526
|
-
<SelectItem value="react">React</SelectItem>
|
|
527
|
-
<SelectItem value="vue">Vue</SelectItem>
|
|
528
|
-
<SelectItem value="angular">Angular</SelectItem>
|
|
529
|
-
</SelectContent>
|
|
530
|
-
</Select>
|
|
531
|
-
);
|
|
532
|
-
|
|
533
|
-
// Click trigger to open the select
|
|
534
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
535
|
-
await user.click(trigger);
|
|
536
|
-
|
|
537
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
538
|
-
|
|
539
|
-
fireEvent.change(searchInput, { target: { value: 'react' } });
|
|
540
|
-
expect(searchInput).toHaveValue('react');
|
|
541
|
-
// Note: The mock doesn't actually call the filter function, so we just test the input works
|
|
542
|
-
});
|
|
543
|
-
|
|
544
|
-
it('handles debounced search with different delays', async () => {
|
|
545
|
-
const { rerender } = renderWithProviders(
|
|
546
|
-
<Select>
|
|
547
|
-
<SelectTrigger>
|
|
548
|
-
<SelectValue placeholder="Select an option" />
|
|
549
|
-
</SelectTrigger>
|
|
550
|
-
<SelectContent searchable debounceMs={100}>
|
|
551
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
552
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
553
|
-
</SelectContent>
|
|
554
|
-
</Select>
|
|
555
|
-
);
|
|
556
|
-
|
|
557
|
-
// Click trigger to open the select
|
|
558
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
559
|
-
await user.click(trigger);
|
|
560
|
-
|
|
561
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
562
|
-
fireEvent.change(searchInput, { target: { value: 'test' } });
|
|
563
|
-
expect(searchInput).toHaveValue('test');
|
|
564
|
-
|
|
565
|
-
// Test with different debounce delay
|
|
566
|
-
rerenderWithProviders(
|
|
567
|
-
<Select>
|
|
568
|
-
<SelectTrigger>
|
|
569
|
-
<SelectValue placeholder="Select an option" />
|
|
570
|
-
</SelectTrigger>
|
|
571
|
-
<SelectContent searchable debounceMs={500}>
|
|
572
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
573
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
574
|
-
</SelectContent>
|
|
575
|
-
</Select>
|
|
576
|
-
);
|
|
577
|
-
|
|
578
|
-
const newSearchInput = screen.getByTestId('select-search-input');
|
|
579
|
-
await user.clear(newSearchInput);
|
|
580
|
-
fireEvent.change(newSearchInput, { target: { value: 'new' } });
|
|
581
|
-
expect(newSearchInput).toHaveValue('new');
|
|
582
|
-
});
|
|
583
|
-
});
|
|
584
|
-
|
|
585
|
-
describe('Advanced Component Behavior', () => {
|
|
586
|
-
it('handles dynamic option updates', async () => {
|
|
587
|
-
const { rerender } = renderWithProviders(
|
|
588
|
-
<Select>
|
|
589
|
-
<SelectTrigger>
|
|
590
|
-
<SelectValue placeholder="Select an option" />
|
|
591
|
-
</SelectTrigger>
|
|
592
|
-
<SelectContent>
|
|
593
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
594
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
595
|
-
</SelectContent>
|
|
596
|
-
</Select>
|
|
597
|
-
);
|
|
598
|
-
|
|
599
|
-
// Click trigger to open the select
|
|
600
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
601
|
-
await user.click(trigger);
|
|
602
|
-
|
|
603
|
-
expect(screen.getAllByTestId('select-item')).toHaveLength(2);
|
|
604
|
-
|
|
605
|
-
// Update options
|
|
606
|
-
rerenderWithProviders(
|
|
607
|
-
<Select>
|
|
608
|
-
<SelectTrigger>
|
|
609
|
-
<SelectValue placeholder="Select an option" />
|
|
610
|
-
</SelectTrigger>
|
|
611
|
-
<SelectContent>
|
|
612
|
-
<SelectItem value="option3">Option 3</SelectItem>
|
|
613
|
-
<SelectItem value="option4">Option 4</SelectItem>
|
|
614
|
-
<SelectItem value="option5">Option 5</SelectItem>
|
|
615
|
-
</SelectContent>
|
|
616
|
-
</Select>
|
|
617
|
-
);
|
|
618
|
-
|
|
619
|
-
// Open the select to render the new options
|
|
620
|
-
const newTriggers = screen.getAllByTestId('select-trigger');
|
|
621
|
-
const newTrigger = newTriggers[newTriggers.length - 1]; // Get the latest one
|
|
622
|
-
await user.click(newTrigger);
|
|
623
|
-
|
|
624
|
-
// Wait for the new options to be rendered
|
|
625
|
-
await waitFor(() => {
|
|
626
|
-
expect(screen.getAllByTestId('select-item')).toHaveLength(3);
|
|
627
|
-
});
|
|
628
|
-
});
|
|
629
|
-
|
|
630
|
-
it('handles controlled vs uncontrolled behavior', () => {
|
|
631
|
-
// Test uncontrolled
|
|
632
|
-
const { rerender } = renderWithProviders(
|
|
633
|
-
<Select>
|
|
634
|
-
<SelectTrigger>
|
|
635
|
-
<SelectValue placeholder="Select an option" />
|
|
636
|
-
</SelectTrigger>
|
|
637
|
-
<SelectContent>
|
|
638
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
639
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
640
|
-
</SelectContent>
|
|
641
|
-
</Select>
|
|
642
|
-
);
|
|
643
|
-
|
|
644
|
-
expect(screen.getByTestId('select-root')).toBeInTheDocument();
|
|
645
|
-
|
|
646
|
-
// Test controlled
|
|
647
|
-
rerenderWithProviders(
|
|
648
|
-
<Select value="option2">
|
|
649
|
-
<SelectTrigger>
|
|
650
|
-
<SelectValue placeholder="Select an option" />
|
|
651
|
-
</SelectTrigger>
|
|
652
|
-
<SelectContent>
|
|
653
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
654
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
655
|
-
</SelectContent>
|
|
656
|
-
</Select>
|
|
657
|
-
);
|
|
658
|
-
|
|
659
|
-
const selectRoots = screen.getAllByTestId('select-root');
|
|
660
|
-
expect(selectRoots[selectRoots.length - 1]).toHaveAttribute('data-value', 'option2');
|
|
661
|
-
});
|
|
662
|
-
|
|
663
|
-
it('handles disabled state propagation', async () => {
|
|
664
|
-
renderWithProviders(
|
|
665
|
-
<Select disabled>
|
|
666
|
-
<SelectTrigger disabled>
|
|
667
|
-
<SelectValue placeholder="Select an option" />
|
|
668
|
-
</SelectTrigger>
|
|
669
|
-
<SelectContent>
|
|
670
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
671
|
-
<SelectItem value="option2" disabled>Option 2</SelectItem>
|
|
672
|
-
</SelectContent>
|
|
673
|
-
</Select>
|
|
674
|
-
);
|
|
675
|
-
|
|
676
|
-
const root = screen.getByTestId('select-root');
|
|
677
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
678
|
-
|
|
679
|
-
// The disabled state is handled by the trigger being disabled, not a data attribute
|
|
680
|
-
// expect(root).toHaveAttribute('data-disabled', 'true');
|
|
681
|
-
expect(trigger).toBeDisabled();
|
|
682
|
-
|
|
683
|
-
// When disabled, the select should not open, so we just test the disabled state
|
|
684
|
-
expect(trigger).toBeDisabled();
|
|
685
|
-
expect(trigger).toHaveAttribute('aria-disabled', 'true');
|
|
686
|
-
});
|
|
687
|
-
|
|
688
|
-
it('handles custom positioning', async () => {
|
|
689
|
-
renderWithProviders(
|
|
690
|
-
<Select>
|
|
691
|
-
<SelectTrigger>
|
|
692
|
-
<SelectValue placeholder="Select an option" />
|
|
693
|
-
</SelectTrigger>
|
|
694
|
-
<SelectContent position="item-aligned">
|
|
695
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
696
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
697
|
-
</SelectContent>
|
|
698
|
-
</Select>
|
|
699
|
-
);
|
|
700
|
-
|
|
701
|
-
// Click trigger to open the select
|
|
702
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
703
|
-
await user.click(trigger);
|
|
704
|
-
|
|
705
|
-
const content = screen.getByTestId('select-content');
|
|
706
|
-
// The positioning is handled by CSS classes, not data attributes in our mock
|
|
707
|
-
expect(content).toBeInTheDocument();
|
|
708
|
-
});
|
|
709
|
-
});
|
|
710
|
-
|
|
711
|
-
describe('Advanced Search Scenarios', () => {
|
|
712
|
-
it('handles large dataset search performance', async () => {
|
|
713
|
-
const largeOptions = Array.from({ length: 1000 }, (_, i) => (
|
|
714
|
-
<SelectItem key={i} value={`option-${i}`}>
|
|
715
|
-
Option {i + 1} - {Math.random().toString(36).substring(7)}
|
|
716
|
-
</SelectItem>
|
|
717
|
-
));
|
|
718
|
-
|
|
719
|
-
renderWithProviders(
|
|
720
|
-
<Select>
|
|
721
|
-
<SelectTrigger>
|
|
722
|
-
<SelectValue placeholder="Select an option" />
|
|
723
|
-
</SelectTrigger>
|
|
724
|
-
<SelectContent searchable debounceMs={50}>
|
|
725
|
-
{largeOptions}
|
|
726
|
-
</SelectContent>
|
|
727
|
-
</Select>
|
|
728
|
-
);
|
|
729
|
-
|
|
730
|
-
// Click trigger to open the select
|
|
731
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
732
|
-
await user.click(trigger);
|
|
733
|
-
|
|
734
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
735
|
-
|
|
736
|
-
// Test search performance with large dataset
|
|
737
|
-
fireEvent.change(searchInput, { target: { value: 'Option 500' } });
|
|
738
|
-
expect(searchInput).toHaveValue('Option 500');
|
|
739
|
-
|
|
740
|
-
// Should not cause performance issues
|
|
741
|
-
expect(screen.getByTestId('select-search-input')).toBeInTheDocument();
|
|
742
|
-
});
|
|
743
|
-
|
|
744
|
-
it('handles search with grouped options', async () => {
|
|
745
|
-
renderWithProviders(
|
|
746
|
-
<Select>
|
|
747
|
-
<SelectTrigger>
|
|
748
|
-
<SelectValue placeholder="Select an option" />
|
|
749
|
-
</SelectTrigger>
|
|
750
|
-
<SelectContent searchable>
|
|
751
|
-
<SelectGroup>
|
|
752
|
-
<SelectLabel>Frontend</SelectLabel>
|
|
753
|
-
<SelectItem value="react">React</SelectItem>
|
|
754
|
-
<SelectItem value="vue">Vue</SelectItem>
|
|
755
|
-
<SelectItem value="angular">Angular</SelectItem>
|
|
756
|
-
</SelectGroup>
|
|
757
|
-
<SelectSeparator />
|
|
758
|
-
<SelectGroup>
|
|
759
|
-
<SelectLabel>Backend</SelectLabel>
|
|
760
|
-
<SelectItem value="node">Node.js</SelectItem>
|
|
761
|
-
<SelectItem value="python">Python</SelectItem>
|
|
762
|
-
<SelectItem value="java">Java</SelectItem>
|
|
763
|
-
</SelectGroup>
|
|
764
|
-
</SelectContent>
|
|
765
|
-
</Select>
|
|
766
|
-
);
|
|
767
|
-
|
|
768
|
-
// Click trigger to open the select
|
|
769
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
770
|
-
await user.click(trigger);
|
|
771
|
-
|
|
772
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
773
|
-
|
|
774
|
-
fireEvent.change(searchInput, { target: { value: 'React' } });
|
|
775
|
-
expect(searchInput).toHaveValue('React');
|
|
776
|
-
|
|
777
|
-
await user.clear(searchInput);
|
|
778
|
-
fireEvent.change(searchInput, { target: { value: 'Node' } });
|
|
779
|
-
expect(searchInput).toHaveValue('Node');
|
|
780
|
-
});
|
|
781
|
-
|
|
782
|
-
it('handles search highlighting with complex content', async () => {
|
|
783
|
-
renderWithProviders(
|
|
784
|
-
<Select>
|
|
785
|
-
<SelectTrigger>
|
|
786
|
-
<SelectValue placeholder="Select an option" />
|
|
787
|
-
</SelectTrigger>
|
|
788
|
-
<SelectContent searchable highlightMatches>
|
|
789
|
-
<SelectItem value="complex1">
|
|
790
|
-
<span>🚀</span>
|
|
791
|
-
<strong>React</strong>
|
|
792
|
-
<span>Framework</span>
|
|
793
|
-
</SelectItem>
|
|
794
|
-
<SelectItem value="complex2">
|
|
795
|
-
<span>⚡</span>
|
|
796
|
-
<strong>Vue</strong>
|
|
797
|
-
<span>Framework</span>
|
|
798
|
-
</SelectItem>
|
|
799
|
-
</SelectContent>
|
|
800
|
-
</Select>
|
|
801
|
-
);
|
|
802
|
-
|
|
803
|
-
// Click trigger to open the select
|
|
804
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
805
|
-
await user.click(trigger);
|
|
806
|
-
|
|
807
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
808
|
-
|
|
809
|
-
fireEvent.change(searchInput, { target: { value: 'React' } });
|
|
810
|
-
expect(searchInput).toHaveValue('React');
|
|
811
|
-
|
|
812
|
-
await user.clear(searchInput);
|
|
813
|
-
fireEvent.change(searchInput, { target: { value: 'Framework' } });
|
|
814
|
-
expect(searchInput).toHaveValue('Framework');
|
|
815
|
-
});
|
|
816
|
-
});
|
|
817
|
-
|
|
818
|
-
describe('Edge Cases and Error Handling', () => {
|
|
819
|
-
it('handles empty search results gracefully', async () => {
|
|
820
|
-
renderWithProviders(
|
|
821
|
-
<Select>
|
|
822
|
-
<SelectTrigger>
|
|
823
|
-
<SelectValue placeholder="Select an option" />
|
|
824
|
-
</SelectTrigger>
|
|
825
|
-
<SelectContent searchable noResultsText="No matching options found">
|
|
826
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
827
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
828
|
-
</SelectContent>
|
|
829
|
-
</Select>
|
|
830
|
-
);
|
|
831
|
-
|
|
832
|
-
// Click trigger to open the select
|
|
833
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
834
|
-
await user.click(trigger);
|
|
835
|
-
|
|
836
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
837
|
-
fireEvent.change(searchInput, { target: { value: 'xyz123' } });
|
|
838
|
-
|
|
839
|
-
// Should handle no results gracefully
|
|
840
|
-
expect(searchInput).toHaveValue('xyz123');
|
|
841
|
-
});
|
|
842
|
-
|
|
843
|
-
it('handles rapid search input changes', async () => {
|
|
844
|
-
renderWithProviders(
|
|
845
|
-
<Select>
|
|
846
|
-
<SelectTrigger>
|
|
847
|
-
<SelectValue placeholder="Select an option" />
|
|
848
|
-
</SelectTrigger>
|
|
849
|
-
<SelectContent searchable debounceMs={100}>
|
|
850
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
851
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
852
|
-
</SelectContent>
|
|
853
|
-
</Select>
|
|
854
|
-
);
|
|
855
|
-
|
|
856
|
-
// Click trigger to open the select
|
|
857
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
858
|
-
await user.click(trigger);
|
|
859
|
-
|
|
860
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
861
|
-
|
|
862
|
-
// Rapid typing and clearing
|
|
863
|
-
fireEvent.change(searchInput, { target: { value: 'a' } });
|
|
864
|
-
await user.clear(searchInput);
|
|
865
|
-
fireEvent.change(searchInput, { target: { value: 'b' } });
|
|
866
|
-
await user.clear(searchInput);
|
|
867
|
-
fireEvent.change(searchInput, { target: { value: 'c' } });
|
|
868
|
-
await user.clear(searchInput);
|
|
869
|
-
fireEvent.change(searchInput, { target: { value: 'd' } });
|
|
870
|
-
|
|
871
|
-
expect(searchInput).toHaveValue('d');
|
|
872
|
-
});
|
|
873
|
-
|
|
874
|
-
it('handles search with special regex characters', async () => {
|
|
875
|
-
renderWithProviders(
|
|
876
|
-
<Select>
|
|
877
|
-
<SelectTrigger>
|
|
878
|
-
<SelectValue placeholder="Select an option" />
|
|
879
|
-
</SelectTrigger>
|
|
880
|
-
<SelectContent searchable>
|
|
881
|
-
<SelectItem value="regex-test">Regex Test</SelectItem>
|
|
882
|
-
<SelectItem value="special-chars">Special Characters: .*+?^${}()|[\]\\</SelectItem>
|
|
883
|
-
</SelectContent>
|
|
884
|
-
</Select>
|
|
885
|
-
);
|
|
886
|
-
|
|
887
|
-
// Click trigger to open the select
|
|
888
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
889
|
-
await user.click(trigger);
|
|
890
|
-
|
|
891
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
892
|
-
|
|
893
|
-
// Test special regex characters (using fireEvent to avoid userEvent parsing issues)
|
|
894
|
-
fireEvent.change(searchInput, { target: { value: '.*+?^${}()|[\\]\\\\' } });
|
|
895
|
-
expect(searchInput).toHaveValue('.*+?^${}()|[\\]\\\\');
|
|
896
|
-
|
|
897
|
-
await user.clear(searchInput);
|
|
898
|
-
fireEvent.change(searchInput, { target: { value: 'Regex' } });
|
|
899
|
-
expect(searchInput).toHaveValue('Regex');
|
|
900
|
-
});
|
|
901
|
-
|
|
902
|
-
it('handles search with very long input', async () => {
|
|
903
|
-
renderWithProviders(
|
|
904
|
-
<Select>
|
|
905
|
-
<SelectTrigger>
|
|
906
|
-
<SelectValue placeholder="Select an option" />
|
|
907
|
-
</SelectTrigger>
|
|
908
|
-
<SelectContent searchable>
|
|
909
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
910
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
911
|
-
</SelectContent>
|
|
912
|
-
</Select>
|
|
913
|
-
);
|
|
914
|
-
|
|
915
|
-
// Click trigger to open the select
|
|
916
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
917
|
-
await user.click(trigger);
|
|
918
|
-
|
|
919
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
920
|
-
|
|
921
|
-
// Very long search term
|
|
922
|
-
const longSearchTerm = 'a'.repeat(1000);
|
|
923
|
-
fireEvent.change(searchInput, { target: { value: longSearchTerm } });
|
|
924
|
-
expect(searchInput).toHaveValue(longSearchTerm);
|
|
925
|
-
});
|
|
926
|
-
|
|
927
|
-
it('handles search with unicode normalization', async () => {
|
|
928
|
-
renderWithProviders(
|
|
929
|
-
<Select>
|
|
930
|
-
<SelectTrigger>
|
|
931
|
-
<SelectValue placeholder="Select an option" />
|
|
932
|
-
</SelectTrigger>
|
|
933
|
-
<SelectContent searchable>
|
|
934
|
-
<SelectItem value="cafe">café</SelectItem>
|
|
935
|
-
<SelectItem value="cafe-normal">café</SelectItem>
|
|
936
|
-
<SelectItem value="umlaut">über</SelectItem>
|
|
937
|
-
</SelectContent>
|
|
938
|
-
</Select>
|
|
939
|
-
);
|
|
940
|
-
|
|
941
|
-
// Click trigger to open the select
|
|
942
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
943
|
-
await user.click(trigger);
|
|
944
|
-
|
|
945
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
946
|
-
|
|
947
|
-
// Test unicode characters
|
|
948
|
-
fireEvent.change(searchInput, { target: { value: 'café' } });
|
|
949
|
-
expect(searchInput).toHaveValue('café');
|
|
950
|
-
|
|
951
|
-
await user.clear(searchInput);
|
|
952
|
-
fireEvent.change(searchInput, { target: { value: 'über' } });
|
|
953
|
-
expect(searchInput).toHaveValue('über');
|
|
954
|
-
});
|
|
955
|
-
});
|
|
956
|
-
|
|
957
|
-
describe('Performance and Memory Tests', () => {
|
|
958
|
-
it('handles memory cleanup on unmount', () => {
|
|
959
|
-
const { unmount } = renderWithProviders(
|
|
960
|
-
<Select>
|
|
961
|
-
<SelectTrigger>
|
|
962
|
-
<SelectValue placeholder="Select an option" />
|
|
963
|
-
</SelectTrigger>
|
|
964
|
-
<SelectContent searchable>
|
|
965
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
966
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
967
|
-
</SelectContent>
|
|
968
|
-
</Select>
|
|
969
|
-
);
|
|
970
|
-
|
|
971
|
-
// Should render without errors
|
|
972
|
-
expect(screen.getByTestId('select-trigger')).toBeInTheDocument();
|
|
973
|
-
|
|
974
|
-
// Should unmount without memory leaks
|
|
975
|
-
unmount();
|
|
976
|
-
});
|
|
977
|
-
|
|
978
|
-
it('handles rapid mount/unmount cycles', () => {
|
|
979
|
-
const { unmount } = renderWithProviders(
|
|
980
|
-
<Select>
|
|
981
|
-
<SelectTrigger>
|
|
982
|
-
<SelectValue placeholder="Select an option" />
|
|
983
|
-
</SelectTrigger>
|
|
984
|
-
<SelectContent searchable>
|
|
985
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
986
|
-
</SelectContent>
|
|
987
|
-
</Select>
|
|
988
|
-
);
|
|
989
|
-
|
|
990
|
-
// Test that component renders without errors
|
|
991
|
-
expect(screen.getByTestId('select-trigger')).toBeInTheDocument();
|
|
992
|
-
|
|
993
|
-
// Test unmount
|
|
994
|
-
unmount();
|
|
995
|
-
});
|
|
996
|
-
|
|
997
|
-
it('handles search with maximum height constraints', async () => {
|
|
998
|
-
renderWithProviders(
|
|
999
|
-
<Select>
|
|
1000
|
-
<SelectTrigger>
|
|
1001
|
-
<SelectValue placeholder="Select an option" />
|
|
1002
|
-
</SelectTrigger>
|
|
1003
|
-
<SelectContent searchable maxHeight="10rem">
|
|
1004
|
-
{Array.from({ length: 100 }, (_, i) => (
|
|
1005
|
-
<SelectItem key={i} value={`option-${i}`}>
|
|
1006
|
-
Option {i + 1}
|
|
1007
|
-
</SelectItem>
|
|
1008
|
-
))}
|
|
1009
|
-
</SelectContent>
|
|
1010
|
-
</Select>
|
|
1011
|
-
);
|
|
1012
|
-
|
|
1013
|
-
// Click trigger to open the select
|
|
1014
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
1015
|
-
await user.click(trigger);
|
|
1016
|
-
|
|
1017
|
-
const content = screen.getByTestId('select-content');
|
|
1018
|
-
expect(content).toBeInTheDocument();
|
|
1019
|
-
// The mock doesn't set data-searchable attribute, so we just test that it renders
|
|
1020
|
-
});
|
|
1021
|
-
});
|
|
1022
|
-
|
|
1023
|
-
describe('Accessibility Enhancements', () => {
|
|
1024
|
-
it('provides proper ARIA live regions for search', async () => {
|
|
1025
|
-
renderWithProviders(
|
|
1026
|
-
<Select>
|
|
1027
|
-
<SelectTrigger>
|
|
1028
|
-
<SelectValue placeholder="Select an option" />
|
|
1029
|
-
</SelectTrigger>
|
|
1030
|
-
<SelectContent searchable>
|
|
1031
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
1032
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
1033
|
-
</SelectContent>
|
|
1034
|
-
</Select>
|
|
1035
|
-
);
|
|
1036
|
-
|
|
1037
|
-
// Click trigger to open the select
|
|
1038
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
1039
|
-
await user.click(trigger);
|
|
1040
|
-
|
|
1041
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
1042
|
-
fireEvent.change(searchInput, { target: { value: 'test' } });
|
|
1043
|
-
|
|
1044
|
-
// Should have proper ARIA attributes
|
|
1045
|
-
expect(searchInput).toHaveAttribute('aria-label', 'Search options');
|
|
1046
|
-
expect(searchInput).toHaveAttribute('type', 'text');
|
|
1047
|
-
});
|
|
1048
|
-
|
|
1049
|
-
it('handles screen reader announcements correctly', async () => {
|
|
1050
|
-
renderWithProviders(
|
|
1051
|
-
<Select>
|
|
1052
|
-
<SelectTrigger>
|
|
1053
|
-
<SelectValue placeholder="Select an option" />
|
|
1054
|
-
</SelectTrigger>
|
|
1055
|
-
<SelectContent searchable>
|
|
1056
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
1057
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
1058
|
-
</SelectContent>
|
|
1059
|
-
</Select>
|
|
1060
|
-
);
|
|
1061
|
-
|
|
1062
|
-
// Click trigger to open the select
|
|
1063
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
1064
|
-
await user.click(trigger);
|
|
1065
|
-
|
|
1066
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
1067
|
-
fireEvent.change(searchInput, { target: { value: 'option' } });
|
|
1068
|
-
|
|
1069
|
-
// Should handle search announcements
|
|
1070
|
-
expect(searchInput).toHaveValue('option');
|
|
1071
|
-
});
|
|
1072
|
-
|
|
1073
|
-
it('provides proper focus management', async () => {
|
|
1074
|
-
renderWithProviders(
|
|
1075
|
-
<Select>
|
|
1076
|
-
<SelectTrigger>
|
|
1077
|
-
<SelectValue placeholder="Select an option" />
|
|
1078
|
-
</SelectTrigger>
|
|
1079
|
-
<SelectContent searchable>
|
|
1080
|
-
<SelectItem value="option1">Option 1</SelectItem>
|
|
1081
|
-
<SelectItem value="option2">Option 2</SelectItem>
|
|
1082
|
-
</SelectContent>
|
|
1083
|
-
</Select>
|
|
1084
|
-
);
|
|
1085
|
-
|
|
1086
|
-
const trigger = screen.getByTestId('select-trigger');
|
|
1087
|
-
|
|
1088
|
-
// Click trigger to open the select
|
|
1089
|
-
await user.click(trigger);
|
|
1090
|
-
|
|
1091
|
-
const searchInput = screen.getByTestId('select-search-input');
|
|
1092
|
-
|
|
1093
|
-
// Test focus management
|
|
1094
|
-
trigger.focus();
|
|
1095
|
-
expect(trigger).toHaveFocus();
|
|
1096
|
-
|
|
1097
|
-
searchInput.focus();
|
|
1098
|
-
expect(searchInput).toHaveFocus();
|
|
1099
|
-
});
|
|
1100
|
-
});
|
|
1101
|
-
});
|