@jmruthers/pace-core 0.6.9 → 0.6.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1182) hide show
  1. package/CHANGELOG.md +21 -0
  2. package/audit-tool/00-dependencies.cjs +46 -13
  3. package/audit-tool/audits/01-pace-core-compliance.cjs +96 -21
  4. package/audit-tool/audits/02-project-structure.cjs +74 -2
  5. package/audit-tool/audits/03-architecture.cjs +220 -20
  6. package/audit-tool/audits/04-code-quality.cjs +95 -3
  7. package/audit-tool/audits/05-styling.cjs +19 -7
  8. package/audit-tool/audits/06-security-rbac.cjs +214 -25
  9. package/audit-tool/audits/07-api-tech-stack.cjs +31 -15
  10. package/audit-tool/audits/08-testing-documentation.cjs +11 -3
  11. package/audit-tool/audits/09-operations.cjs +19 -7
  12. package/audit-tool/index.cjs +22 -11
  13. package/audit-tool/utils/report-utils.cjs +4 -0
  14. package/cursor-rules/01-pace-core-compliance.mdc +1 -0
  15. package/cursor-rules/02-project-structure.mdc +3 -26
  16. package/cursor-rules/03-architecture.mdc +3 -1
  17. package/cursor-rules/04-code-quality.mdc +1 -0
  18. package/cursor-rules/05-styling.mdc +120 -8
  19. package/cursor-rules/06-security-rbac.mdc +126 -2
  20. package/cursor-rules/07-api-tech-stack.mdc +1 -0
  21. package/cursor-rules/08-testing-documentation.mdc +1 -0
  22. package/cursor-rules/09-operations.mdc +1 -0
  23. package/dist/DataTable-EFYP2QLE.js +16 -0
  24. package/dist/InactivityServiceProvider-BbxwwDz1.d.ts +308 -0
  25. package/dist/UnifiedAuthProvider-Bkt_tzdS.d.ts +183 -0
  26. package/dist/api-BZR2CYXL.js +5 -0
  27. package/dist/api-result-USV1Czr-.d.ts +51 -0
  28. package/dist/assets/app-icons/admin_favicon.svg +462 -0
  29. package/dist/assets/app-icons/base_favicon.svg +85 -0
  30. package/dist/assets/app-icons/cake_favicon.svg +68 -0
  31. package/dist/assets/app-icons/core_favicon.svg +256 -0
  32. package/dist/assets/app-icons/gear_favicon.svg +91 -0
  33. package/dist/assets/app-icons/medi_favicon.svg +92 -0
  34. package/dist/assets/app-icons/mint_favicon.svg +83 -0
  35. package/dist/assets/app-icons/pace_favicon.svg +49 -0
  36. package/dist/assets/app-icons/pump_favicon.svg +68 -0
  37. package/dist/assets/app-icons/seed_favicon.svg +91 -0
  38. package/dist/assets/app-icons/team_favicon.svg +67 -0
  39. package/dist/assets/app-icons/trac_favicon.svg +112 -0
  40. package/dist/assets/app-icons/trip_favicon.svg +102 -0
  41. package/dist/audit-HI2DHUVU.js +4 -0
  42. package/dist/auth-JvdRVaud.d.ts +49 -0
  43. package/dist/chunk-2DL2WSOE.js +327 -0
  44. package/dist/chunk-2OEVOGGR.js +9598 -0
  45. package/dist/chunk-44CNXN4P.js +15 -0
  46. package/dist/chunk-4R3T5ENU.js +2943 -0
  47. package/dist/chunk-7A6IMHH2.js +2321 -0
  48. package/dist/chunk-BTHN5MKC.js +121 -0
  49. package/dist/chunk-CU2BU2MQ.js +2 -0
  50. package/dist/chunk-D6BMFMQZ.js +200 -0
  51. package/dist/chunk-DDMPHZ3D.js +58 -0
  52. package/dist/chunk-ENLXB7GP.js +721 -0
  53. package/dist/chunk-J2KQK6DG.js +2159 -0
  54. package/dist/chunk-KJXRL3XE.js +6434 -0
  55. package/dist/chunk-L5LFKKLJ.js +61 -0
  56. package/dist/chunk-PCSHBLPB.js +811 -0
  57. package/dist/chunk-QRYSEPHB.js +429 -0
  58. package/dist/chunk-RMLY6KB5.js +187 -0
  59. package/dist/chunk-SACF5YSM.js +31 -0
  60. package/dist/chunk-UZNAFKGW.js +125 -0
  61. package/dist/chunk-V7FTM2LU.js +1080 -0
  62. package/dist/chunk-WY6Y7KC3.js +264 -0
  63. package/dist/chunk-XOJME5T7.js +407 -0
  64. package/dist/chunk-XPFVT3GN.js +492 -0
  65. package/dist/chunk-YFTFFJIV.js +529 -0
  66. package/dist/chunk-YYTWKVHO.js +1334 -0
  67. package/dist/components.d.ts +12 -89
  68. package/dist/components.js +23 -55
  69. package/dist/database.generated-qkdoiVrJ.d.ts +9441 -0
  70. package/dist/eslint-rules/index.cjs +3 -0
  71. package/dist/eslint-rules/rules/03-architecture.cjs +74 -0
  72. package/dist/eslint-rules/rules/05-styling.cjs +507 -0
  73. package/dist/eslint-rules/rules/06-security-rbac.cjs +84 -0
  74. package/dist/event-BfCox3N2.d.ts +265 -0
  75. package/dist/file-reference-DU1hcawx.d.ts +164 -0
  76. package/dist/functions-DH45k8ec.d.ts +208 -0
  77. package/dist/hooks.d.ts +28 -14
  78. package/dist/hooks.js +90 -56
  79. package/dist/icons/index.d.ts +1 -0
  80. package/dist/icons/index.js +1 -0
  81. package/dist/index.d.ts +392 -155
  82. package/dist/index.js +337 -347
  83. package/dist/pagination-BW1mqywp.d.ts +201 -0
  84. package/dist/papaparseLoader-WG2UXQ22.js +7 -0
  85. package/dist/providers.d.ts +29 -14
  86. package/dist/providers.js +7 -5
  87. package/dist/rbac/eslint-rules.js +2 -2
  88. package/dist/rbac/index.d.ts +180 -351
  89. package/dist/rbac/index.js +13 -11
  90. package/dist/theming/runtime.d.ts +28 -5
  91. package/dist/theming/runtime.js +2 -2
  92. package/dist/timezone-BTWWXKVY.d.ts +696 -0
  93. package/dist/types-BE2sEHKd.d.ts +55 -0
  94. package/dist/types-CvOPXWWZ.d.ts +111 -0
  95. package/dist/types-Dr8sNhER.d.ts +50 -0
  96. package/dist/types.d.ts +20 -13
  97. package/dist/types.js +1 -0
  98. package/dist/usePublicPageContext-B91dGYW1.d.ts +4367 -0
  99. package/dist/usePublicRouteParams-BgV6VhMi.d.ts +946 -0
  100. package/dist/utils.d.ts +338 -156
  101. package/dist/utils.js +78 -60
  102. package/dist/validation-g5n0hDkh.d.ts +177 -0
  103. package/docs/api/modules.md +1226 -1094
  104. package/docs/api-reference/components.md +5 -5
  105. package/docs/api-reference/rpc-functions.md +12 -3
  106. package/docs/core-concepts/rbac-system.md +8 -0
  107. package/docs/getting-started/cursor-rules.md +17 -20
  108. package/docs/getting-started/dependencies.md +1 -1
  109. package/docs/getting-started/setup.md +235 -0
  110. package/docs/implementation-guides/authentication.md +27 -0
  111. package/docs/implementation-guides/data-tables.md +365 -10
  112. package/docs/migration/ApiResult-migration.md +25 -0
  113. package/docs/rbac/RBAC_CONTRACT.md +0 -12
  114. package/docs/rbac/api-reference.md +33 -31
  115. package/docs/standards/0-standards-overview.md +50 -15
  116. package/docs/standards/1-pace-core-compliance-standards.md +62 -57
  117. package/docs/standards/2-project-structure-standards.md +45 -90
  118. package/docs/standards/3-architecture-standards.md +41 -1
  119. package/docs/standards/4-code-quality-standards.md +26 -6
  120. package/docs/standards/5-styling-standards.md +35 -1
  121. package/docs/standards/6-security-rbac-standards.md +288 -7
  122. package/docs/standards/7-api-tech-stack-standards.md +116 -17
  123. package/docs/standards/8-testing-documentation-standards.md +31 -0
  124. package/docs/standards/9-operations-standards.md +19 -0
  125. package/docs/standards/README.md +20 -201
  126. package/docs/testing/README.md +10 -0
  127. package/docs/testing/test-setup-for-consumers.md +916 -0
  128. package/docs/troubleshooting/common-issues.md +17 -1
  129. package/docs/troubleshooting/organisation-context-setup.md +8 -0
  130. package/docs/troubleshooting/print-event-name-css-variable-analysis.md +217 -0
  131. package/eslint-config-pace-core.cjs +24 -0
  132. package/package.json +14 -20
  133. package/scripts/build-docs.js +180 -0
  134. package/scripts/setup.cjs +536 -0
  135. package/scripts/validate.cjs +480 -0
  136. package/src/__mocks__/lucide-react.ts +0 -2
  137. package/src/__tests__/helpers/component-test-utils.test.tsx +260 -0
  138. package/src/__tests__/helpers/optimized-test-setup.test.ts +224 -0
  139. package/src/__tests__/helpers/supabaseMock.test.ts +273 -0
  140. package/src/__tests__/helpers/test-providers.test.tsx +99 -0
  141. package/src/__tests__/helpers/test-providers.tsx +37 -39
  142. package/src/__tests__/helpers/test-utils.test.tsx +447 -0
  143. package/src/__tests__/helpers/timer-utils.test.ts +371 -0
  144. package/src/assets/app-icons/admin_favicon.svg +462 -0
  145. package/src/assets/app-icons/base_favicon.svg +85 -0
  146. package/src/assets/app-icons/cake_favicon.svg +68 -0
  147. package/src/assets/app-icons/core_favicon.svg +256 -0
  148. package/src/assets/app-icons/gear_favicon.svg +91 -0
  149. package/src/assets/app-icons/index.test.ts +304 -0
  150. package/src/assets/app-icons/index.ts +83 -0
  151. package/src/assets/app-icons/medi_favicon.svg +92 -0
  152. package/src/assets/app-icons/mint_favicon.svg +83 -0
  153. package/src/assets/app-icons/pace_favicon.svg +49 -0
  154. package/src/assets/app-icons/pump_favicon.svg +68 -0
  155. package/src/assets/app-icons/seed_favicon.svg +91 -0
  156. package/src/assets/app-icons/team_favicon.svg +67 -0
  157. package/src/assets/app-icons/trac_favicon.svg +112 -0
  158. package/src/assets/app-icons/trip_favicon.svg +102 -0
  159. package/src/components/AddressField/AddressField.test.tsx +379 -4
  160. package/src/components/AddressField/AddressField.tsx +239 -213
  161. package/src/components/AddressField/types.ts +2 -2
  162. package/src/components/Alert/Alert.test.tsx +35 -25
  163. package/src/components/Alert/Alert.tsx +8 -8
  164. package/src/components/AppSwitcher/AppSwitcher.test.tsx +1250 -0
  165. package/src/components/AppSwitcher/AppSwitcher.tsx +315 -0
  166. package/src/components/Avatar/Avatar.test.tsx +11 -1
  167. package/src/components/Avatar/Avatar.tsx +3 -2
  168. package/src/components/Badge/Badge.test.tsx +11 -1
  169. package/src/components/Button/Button.test.tsx +13 -3
  170. package/src/components/Button/Button.tsx +1 -1
  171. package/src/components/Calendar/Calendar.test.tsx +523 -131
  172. package/src/components/Calendar/Calendar.tsx +107 -488
  173. package/src/components/Card/Card.test.tsx +384 -258
  174. package/src/components/Card/Card.tsx +19 -10
  175. package/src/components/Checkbox/Checkbox.test.tsx +58 -174
  176. package/src/components/ContextSelector/ContextSelector.internals.tsx +204 -0
  177. package/src/components/ContextSelector/ContextSelector.test.tsx +360 -0
  178. package/src/components/ContextSelector/ContextSelector.tsx +66 -280
  179. package/src/components/ContextSelector/ContextSelector.types.ts +35 -0
  180. package/src/components/ContextSelector/useContextSelectorState.tsx +195 -0
  181. package/src/components/DataTable/AUDIT_REPORT.md +59 -44
  182. package/src/components/DataTable/DataTable.comprehensive.test.tsx +759 -0
  183. package/src/components/DataTable/DataTable.default-state.test.tsx +524 -0
  184. package/src/components/DataTable/DataTable.export.test.tsx +705 -0
  185. package/src/components/DataTable/DataTable.grouping-aggregation.test.tsx +658 -0
  186. package/src/components/DataTable/DataTable.hooks.test.tsx +192 -0
  187. package/src/components/DataTable/DataTable.select-label-display.test.tsx +485 -0
  188. package/src/components/DataTable/DataTable.test.tsx +787 -416
  189. package/src/components/DataTable/DataTable.tsx +14 -14
  190. package/src/components/DataTable/DataTableCore.integration.test.tsx +458 -0
  191. package/src/components/DataTable/DataTableCore.test-setup.ts +221 -0
  192. package/src/components/DataTable/DataTableCore.test.tsx +970 -0
  193. package/src/components/DataTable/README.md +155 -0
  194. package/src/components/DataTable/TESTING.md +101 -0
  195. package/src/components/DataTable/a11y.basic.test.tsx +788 -0
  196. package/src/components/DataTable/components/DataTableCore.tsx +126 -894
  197. package/src/components/DataTable/components/GroupingDropdown.test.tsx +621 -0
  198. package/src/components/DataTable/components/GroupingDropdown.tsx +2 -3
  199. package/src/components/DataTable/components/ImportModal.tsx +82 -408
  200. package/src/components/DataTable/components/ImportModalFileSection.tsx +148 -0
  201. package/src/components/DataTable/context/DataTableContext.test.tsx +328 -0
  202. package/src/components/DataTable/context/DataTableContext.tsx +13 -13
  203. package/src/components/DataTable/core/ColumnFactory.test.ts +403 -0
  204. package/src/components/DataTable/core/ColumnFactory.ts +3 -3
  205. package/src/components/DataTable/hooks/useColumnOrderPersistence.test.ts +516 -0
  206. package/src/components/DataTable/hooks/useColumnOrderPersistence.ts +12 -9
  207. package/src/components/DataTable/hooks/useColumnVisibilityPersistence.test.ts +256 -0
  208. package/src/components/DataTable/hooks/useColumnVisibilityPersistence.ts +12 -9
  209. package/src/components/DataTable/hooks/useDataTableConfiguration.test.ts +297 -0
  210. package/src/components/DataTable/hooks/useDataTableConfiguration.ts +15 -3
  211. package/src/components/DataTable/hooks/useDataTableDataPipeline.test.ts +270 -0
  212. package/src/components/DataTable/hooks/useDataTableDeletionBatching.test.ts +127 -0
  213. package/src/components/DataTable/hooks/useDataTableDeletionBatching.ts +106 -0
  214. package/src/components/DataTable/hooks/useDataTableEffectiveActions.test.ts +461 -0
  215. package/src/components/DataTable/hooks/useDataTableEffectiveActions.ts +238 -0
  216. package/src/components/DataTable/hooks/useDataTableLayoutHandlers.test.ts +296 -0
  217. package/src/components/DataTable/hooks/useDataTableLayoutHandlers.ts +175 -0
  218. package/src/components/DataTable/hooks/useDataTablePaginationSync.test.ts +203 -0
  219. package/src/components/DataTable/hooks/useDataTablePaginationSync.ts +109 -0
  220. package/src/components/DataTable/hooks/useDataTablePermissions.test.ts +280 -0
  221. package/src/components/DataTable/hooks/useDataTablePermissions.ts +81 -260
  222. package/src/components/DataTable/hooks/useDataTablePipeline.test.tsx +219 -0
  223. package/src/components/DataTable/hooks/useDataTablePipeline.tsx +239 -0
  224. package/src/components/DataTable/hooks/useDataTableRenderGuard.test.tsx +316 -0
  225. package/src/components/DataTable/hooks/useDataTableRenderGuard.tsx +195 -0
  226. package/src/components/DataTable/hooks/useDataTableScope.test.ts +110 -0
  227. package/src/components/DataTable/hooks/useDataTableScope.ts +123 -0
  228. package/src/components/DataTable/hooks/useDataTableState.test.ts +733 -0
  229. package/src/components/DataTable/hooks/useDataTableState.ts +161 -114
  230. package/src/components/DataTable/hooks/useDataTableStateAndPersistence.test.ts +277 -0
  231. package/src/components/DataTable/hooks/useDataTableStateAndPersistence.ts +222 -0
  232. package/src/components/DataTable/hooks/useDataTableSuperAdmin.test.ts +93 -0
  233. package/src/components/DataTable/hooks/useDataTableSuperAdmin.ts +86 -0
  234. package/src/components/DataTable/hooks/useDataTableTableInstance.test.ts +185 -0
  235. package/src/components/DataTable/hooks/useDataTableTableInstance.ts +178 -0
  236. package/src/components/DataTable/hooks/useEffectiveColumnOrder.test.ts +183 -0
  237. package/src/components/DataTable/hooks/useHierarchicalState.test.ts +294 -0
  238. package/src/components/DataTable/hooks/useImportModalFocus.test.ts +184 -0
  239. package/src/components/DataTable/hooks/useImportModalFocus.ts +53 -0
  240. package/src/components/DataTable/hooks/useImportModalState.test.ts +390 -0
  241. package/src/components/DataTable/hooks/useImportModalState.ts +345 -0
  242. package/src/components/DataTable/hooks/useKeyboardNavigation.test.ts +787 -0
  243. package/src/components/DataTable/hooks/useKeyboardNavigation.ts +311 -271
  244. package/src/components/DataTable/hooks/usePermissionTracking.test.ts +381 -0
  245. package/src/components/DataTable/hooks/usePermissionTracking.ts +122 -0
  246. package/src/components/DataTable/hooks/useServerSideDataEffect.test.ts +258 -0
  247. package/src/components/DataTable/hooks/useServerSideDataEffect.ts +27 -4
  248. package/src/components/DataTable/hooks/useTableColumns.test.ts +499 -0
  249. package/src/components/DataTable/hooks/useTableColumns.ts +15 -39
  250. package/src/components/DataTable/hooks/useTableHandlers.test.ts +461 -0
  251. package/src/components/DataTable/hooks/useTableHandlers.ts +13 -22
  252. package/src/components/DataTable/index.ts +28 -5
  253. package/src/components/DataTable/keyboard.test.tsx +734 -0
  254. package/src/components/DataTable/mocks/MockRBACProvider.tsx +66 -0
  255. package/src/components/DataTable/pagination.modes.test.tsx +728 -0
  256. package/src/components/DataTable/ssr.strict-mode.test.tsx +319 -0
  257. package/src/components/DataTable/styles.test.ts +379 -0
  258. package/src/components/DataTable/styles.ts +0 -1
  259. package/src/components/DataTable/test-utils/MockDataTableComponents.tsx +55 -0
  260. package/src/components/DataTable/test-utils/dataFactories.ts +103 -0
  261. package/src/components/DataTable/test-utils/featureConfig.ts +10 -0
  262. package/src/components/DataTable/test-utils/sharedTestUtils.ts +419 -0
  263. package/src/components/DataTable/test-utils.ts +94 -0
  264. package/src/components/DataTable/types/actions.ts +71 -0
  265. package/src/components/DataTable/types/base.ts +39 -0
  266. package/src/components/DataTable/types/columns.ts +125 -0
  267. package/src/components/DataTable/types/export.ts +32 -0
  268. package/src/components/DataTable/types/features.ts +81 -0
  269. package/src/components/DataTable/types/hierarchical.ts +44 -0
  270. package/src/components/DataTable/types/index.ts +43 -0
  271. package/src/components/DataTable/types/pagination.ts +85 -0
  272. package/src/components/DataTable/types/performance.ts +47 -0
  273. package/src/components/DataTable/types/props.ts +62 -0
  274. package/src/components/DataTable/types/rbac.ts +45 -0
  275. package/src/components/DataTable/ui/layout/DataTableCore.test.tsx +1194 -0
  276. package/src/components/DataTable/ui/layout/DataTableCore.tsx +345 -0
  277. package/src/components/DataTable/ui/layout/DataTableErrorBoundary.test.tsx +438 -0
  278. package/src/components/DataTable/ui/layout/DataTableErrorBoundary.tsx +225 -0
  279. package/src/components/DataTable/ui/layout/DataTableLayout.test.tsx +1352 -0
  280. package/src/components/DataTable/ui/layout/DataTableLayout.tsx +661 -0
  281. package/src/components/DataTable/ui/modals/BulkDeleteConfirmDialog.test.tsx +91 -0
  282. package/src/components/DataTable/ui/modals/BulkDeleteConfirmDialog.tsx +43 -0
  283. package/src/components/DataTable/ui/modals/DataTableModals.test.tsx +749 -0
  284. package/src/components/DataTable/ui/modals/DataTableModals.tsx +341 -0
  285. package/src/components/DataTable/ui/modals/ImportModal.test.tsx +1834 -0
  286. package/src/components/DataTable/ui/modals/ImportModal.tsx +197 -0
  287. package/src/components/DataTable/ui/modals/ImportModalFailedRowsSection.tsx +60 -0
  288. package/src/components/DataTable/ui/modals/ImportModalFileSection.tsx +148 -0
  289. package/src/components/DataTable/ui/modals/ImportModalPreviewSection.tsx +60 -0
  290. package/src/components/DataTable/ui/modals/ImportModalSummarySection.tsx +59 -0
  291. package/src/components/DataTable/ui/modals/importModalPersistence.ts +73 -0
  292. package/src/components/DataTable/ui/shared/AccessDeniedPage.test.tsx +245 -0
  293. package/src/components/DataTable/ui/shared/AccessDeniedPage.tsx +159 -0
  294. package/src/components/DataTable/ui/shared/ActionButtons.test.tsx +921 -0
  295. package/src/components/DataTable/ui/shared/ActionButtons.tsx +195 -0
  296. package/src/components/DataTable/ui/shared/ColumnFilter.test.tsx +497 -0
  297. package/src/components/DataTable/ui/shared/ColumnFilter.tsx +113 -0
  298. package/src/components/DataTable/ui/shared/PaginationControls.test.tsx +451 -0
  299. package/src/components/DataTable/ui/shared/PaginationControls.tsx +291 -0
  300. package/src/components/DataTable/ui/shared/SortIndicator.test.tsx +135 -0
  301. package/src/components/DataTable/ui/shared/SortIndicator.tsx +50 -0
  302. package/src/components/DataTable/ui/table/EditFields.test.tsx +526 -0
  303. package/src/components/DataTable/ui/table/EditFields.tsx +355 -0
  304. package/src/components/DataTable/ui/table/EditableRow.test.tsx +1003 -0
  305. package/src/components/DataTable/ui/table/EditableRow.tsx +444 -0
  306. package/src/components/DataTable/ui/table/EmptyState.test.tsx +360 -0
  307. package/src/components/DataTable/ui/table/EmptyState.tsx +74 -0
  308. package/src/components/DataTable/ui/table/FilterRow.test.tsx +416 -0
  309. package/src/components/DataTable/ui/table/FilterRow.tsx +148 -0
  310. package/src/components/DataTable/ui/table/LoadingState.test.tsx +77 -0
  311. package/src/components/DataTable/ui/table/LoadingState.tsx +17 -0
  312. package/src/components/DataTable/ui/table/RowComponent.test.tsx +1024 -0
  313. package/src/components/DataTable/ui/table/RowComponent.tsx +429 -0
  314. package/src/components/DataTable/ui/table/UnifiedTableBody.test.tsx +1273 -0
  315. package/src/components/DataTable/ui/table/UnifiedTableBody.tsx +440 -0
  316. package/src/components/DataTable/ui/table/cellValueUtils.test.ts +453 -0
  317. package/src/components/DataTable/ui/table/cellValueUtils.ts +40 -0
  318. package/src/components/DataTable/ui/toolbar/BulkOperationsDropdown.test.tsx +551 -0
  319. package/src/components/DataTable/ui/toolbar/BulkOperationsDropdown.tsx +160 -0
  320. package/src/components/DataTable/ui/toolbar/ColumnVisibilityDropdown.test.tsx +751 -0
  321. package/src/components/DataTable/ui/toolbar/ColumnVisibilityDropdown.tsx +114 -0
  322. package/src/components/DataTable/ui/toolbar/DataTableToolbar.test.tsx +629 -0
  323. package/src/components/DataTable/ui/toolbar/DataTableToolbar.tsx +271 -0
  324. package/src/components/DataTable/ui/toolbar/GroupingDropdown.test.tsx +621 -0
  325. package/src/components/DataTable/ui/toolbar/GroupingDropdown.tsx +107 -0
  326. package/src/components/DataTable/utils/a11yUtils.test.ts +548 -0
  327. package/src/components/DataTable/utils/a11yUtils.ts +1 -1
  328. package/src/components/DataTable/utils/aggregationUtils.test.ts +288 -0
  329. package/src/components/DataTable/utils/aggregationUtils.ts +5 -5
  330. package/src/components/DataTable/utils/columnUtils.test.ts +94 -0
  331. package/src/components/DataTable/utils/csvParse.test.ts +74 -0
  332. package/src/components/DataTable/utils/csvParse.ts +65 -0
  333. package/src/components/DataTable/utils/errorHandling.test.ts +209 -0
  334. package/src/components/DataTable/utils/errorHandling.ts +3 -1
  335. package/src/components/DataTable/utils/exportUtils.test.ts +954 -0
  336. package/src/components/DataTable/utils/exportUtils.ts +1 -1
  337. package/src/components/DataTable/utils/flexibleImport.test.ts +573 -0
  338. package/src/components/DataTable/utils/flexibleImport.ts +3 -186
  339. package/src/components/DataTable/utils/hierarchicalSorting.test.ts +235 -0
  340. package/src/components/DataTable/utils/hierarchicalSorting.ts +3 -3
  341. package/src/components/DataTable/utils/hierarchicalUtils.test.ts +586 -0
  342. package/src/components/DataTable/utils/importDateParser.test.ts +162 -0
  343. package/src/components/DataTable/utils/importDateParser.ts +114 -0
  344. package/src/components/DataTable/utils/importValueParser.test.ts +138 -0
  345. package/src/components/DataTable/utils/importValueParser.ts +91 -0
  346. package/src/components/DataTable/utils/paginationUtils.test.ts +593 -0
  347. package/src/components/DataTable/utils/paginationUtils.ts +7 -4
  348. package/src/components/DataTable/utils/performanceUtils.test.ts +470 -0
  349. package/src/components/DataTable/utils/performanceUtils.ts +1 -1
  350. package/src/components/DataTable/utils/rowUtils.test.ts +235 -0
  351. package/src/components/DataTable/utils/selectFieldUtils.test.ts +271 -0
  352. package/src/components/DataTable/utils/selectFieldUtils.ts +97 -67
  353. package/src/components/DatePickerWithTimezone/DatePickerWithTimezone.test.tsx +18 -25
  354. package/src/components/DatePickerWithTimezone/DatePickerWithTimezone.tsx +1 -1
  355. package/src/components/DateTimeField/DateTimeField.test.tsx +3 -16
  356. package/src/components/DateTimeField/DateTimeField.tsx +1 -1
  357. package/src/components/Dialog/Dialog.test-utils.ts +49 -0
  358. package/src/components/Dialog/Dialog.test.tsx +2865 -458
  359. package/src/components/Dialog/Dialog.tsx +183 -986
  360. package/src/components/Dialog/dialogLock.test.ts +238 -0
  361. package/src/components/Dialog/dialogLock.ts +98 -0
  362. package/src/components/Dialog/index.ts +2 -0
  363. package/src/components/Dialog/useDialogDimensions.test.ts +163 -0
  364. package/src/components/Dialog/useDialogDimensions.ts +140 -0
  365. package/src/components/Dialog/useDialogLifecycle.test.ts +358 -0
  366. package/src/components/Dialog/useDialogLifecycle.ts +135 -0
  367. package/src/components/Dialog/useDialogPersistence.test.ts +381 -0
  368. package/src/components/Dialog/useDialogPersistence.ts +357 -0
  369. package/src/components/ErrorBoundary/ErrorBoundary.test.tsx +2 -62
  370. package/src/components/ErrorBoundary/ErrorBoundaryContext.context.ts +17 -0
  371. package/src/components/ErrorBoundary/ErrorBoundaryContext.tsx +2 -45
  372. package/src/components/ErrorBoundary/ErrorBoundaryContext.types.ts +41 -0
  373. package/src/components/ErrorBoundary/index.ts +3 -4
  374. package/src/components/ErrorBoundary/useErrorBoundaryContext.ts +20 -0
  375. package/src/components/FileDisplay/FileDisplay.test.tsx +479 -247
  376. package/src/components/FileDisplay/FileDisplay.tsx +29 -659
  377. package/src/components/FileDisplay/FileDisplayContent.test.tsx +395 -0
  378. package/src/components/FileDisplay/FileDisplayContent.tsx +242 -0
  379. package/src/components/FileDisplay/FileDisplayDeleteConfirmDialog.test.tsx +74 -0
  380. package/src/components/FileDisplay/FileDisplayDeleteConfirmDialog.tsx +38 -0
  381. package/src/components/FileDisplay/FileDisplayEmptyView.test.tsx +33 -0
  382. package/src/components/FileDisplay/FileDisplayEmptyView.tsx +33 -0
  383. package/src/components/FileDisplay/FileDisplayErrorView.test.tsx +71 -0
  384. package/src/components/FileDisplay/FileDisplayErrorView.tsx +50 -0
  385. package/src/components/FileDisplay/FileDisplayLoadingFallbackView.test.tsx +22 -0
  386. package/src/components/FileDisplay/FileDisplayLoadingFallbackView.tsx +22 -0
  387. package/src/components/FileDisplay/FileDisplayLoadingView.test.tsx +21 -0
  388. package/src/components/FileDisplay/FileDisplayLoadingView.tsx +23 -0
  389. package/src/components/FileDisplay/FileDisplayMultipleFilesView.test.tsx +101 -0
  390. package/src/components/FileDisplay/FileDisplayMultipleFilesView.tsx +109 -0
  391. package/src/components/FileDisplay/FileDisplaySingleDocumentLinkView.test.tsx +58 -0
  392. package/src/components/FileDisplay/FileDisplaySingleDocumentLinkView.tsx +48 -0
  393. package/src/components/FileDisplay/FileDisplaySingleFileWithActionsView.test.tsx +111 -0
  394. package/src/components/FileDisplay/FileDisplaySingleFileWithActionsView.tsx +270 -0
  395. package/src/components/FileDisplay/FileDisplaySingleImageView.test.tsx +78 -0
  396. package/src/components/FileDisplay/FileDisplaySingleImageView.tsx +67 -0
  397. package/src/components/FileDisplay/fallbackUtils.test.ts +50 -0
  398. package/src/components/FileDisplay/fallbackUtils.ts +44 -0
  399. package/src/components/FileDisplay/fetchFileDisplayData.ts +24 -0
  400. package/src/components/FileDisplay/fetchFileDisplayData.unit.test.ts +183 -0
  401. package/src/components/FileDisplay/fileDisplayUtils.test.ts +58 -0
  402. package/src/components/FileDisplay/fileDisplayUtils.ts +24 -0
  403. package/src/components/FileDisplay/index.tsx +1 -1
  404. package/src/components/FileDisplay/useFileDisplay.test.ts +538 -0
  405. package/src/components/FileDisplay/useFileDisplay.ts +515 -0
  406. package/src/components/FileDisplay/useFileDisplay.unit.test.ts +1438 -0
  407. package/src/components/FileDisplay/useFileDisplayData.ts +126 -0
  408. package/src/components/FileDisplay/usePublicFileDisplay.test.ts +729 -0
  409. package/src/components/FileDisplay/usePublicFileDisplay.ts +579 -0
  410. package/src/components/FileUpload/FileUpload.test.tsx +69 -27
  411. package/src/components/FileUpload/FileUpload.tsx +112 -527
  412. package/src/components/FileUpload/FileUploadDropZone.tsx +112 -0
  413. package/src/components/FileUpload/FileUploadProgressItem.tsx +86 -0
  414. package/src/components/FileUpload/FileUploadProgressList.tsx +40 -0
  415. package/src/components/FileUpload/index.tsx +1 -1
  416. package/src/components/FileUpload/useFileUploadManager.test.ts +308 -0
  417. package/src/components/FileUpload/useFileUploadManager.ts +454 -0
  418. package/src/components/FileUpload/useResolvedAppId.test.ts +102 -0
  419. package/src/components/FileUpload/useResolvedAppId.ts +77 -0
  420. package/src/components/Footer/Footer.test.tsx +15 -382
  421. package/src/components/Footer/Footer.tsx +8 -125
  422. package/src/components/Form/Form.test.tsx +425 -88
  423. package/src/components/Form/Form.tsx +91 -299
  424. package/src/components/Form/useFormPersistence.ts +257 -0
  425. package/src/components/Header/Header.test.tsx +653 -163
  426. package/src/components/Header/Header.tsx +62 -44
  427. package/src/components/InactivityWarningModal/InactivityWarningModal.test.tsx +35 -76
  428. package/src/components/Input/Input.test.tsx +34 -120
  429. package/src/components/Input/Input.tsx +1 -1
  430. package/src/components/Label/Label.test.tsx +46 -45
  431. package/src/components/LoadingSpinner/LoadingSpinner.test.tsx +8 -11
  432. package/src/components/LoginForm/LoginForm.test.tsx +0 -1
  433. package/src/components/NavigationMenu/HierarchicalNavItem.tsx +104 -0
  434. package/src/components/NavigationMenu/NavigationMenu.test.tsx +2422 -102
  435. package/src/components/NavigationMenu/NavigationMenu.tsx +62 -362
  436. package/src/components/NavigationMenu/index.ts +6 -1
  437. package/src/components/NavigationMenu/navigationPermissionHelper.ts +188 -0
  438. package/src/components/NavigationMenu/useNavigationFiltering.test.ts +1949 -0
  439. package/src/components/NavigationMenu/useNavigationFiltering.ts +199 -308
  440. package/src/components/NavigationMenu/useNavigationScope.ts +125 -0
  441. package/src/components/PaceAppLayout/PaceAppLayout.edge-cases.test.tsx +1322 -0
  442. package/src/components/PaceAppLayout/PaceAppLayout.integration.test.tsx +50 -49
  443. package/src/components/PaceAppLayout/PaceAppLayout.performance.test.tsx +81 -38
  444. package/src/components/PaceAppLayout/PaceAppLayout.security.test.tsx +103 -85
  445. package/src/components/PaceAppLayout/PaceAppLayout.test.tsx +774 -44
  446. package/src/components/PaceAppLayout/PaceAppLayout.tsx +282 -764
  447. package/src/components/PaceAppLayout/README.md +0 -9
  448. package/src/components/PaceAppLayout/test-setup.tsx +15 -9
  449. package/src/components/PaceAppLayout/useFilteredNavItems.ts +304 -0
  450. package/src/components/PaceAppLayout/usePaceAppLayoutConfig.ts +142 -0
  451. package/src/components/PaceAppLayout/usePaceAppLayoutGate.tsx +150 -0
  452. package/src/components/PaceAppLayout/usePaceAppLayoutPermissions.ts +162 -0
  453. package/src/components/PaceAppLayout/usePaceAppLayoutScope.ts +79 -0
  454. package/src/components/PaceAppLayout/useRoleBasedRouteAccess.ts +157 -0
  455. package/src/components/PaceAppLayout/useSuperAdminFallback.ts +58 -0
  456. package/src/components/PaceLoginPage/PaceLoginPage.test.tsx +782 -20
  457. package/src/components/PaceLoginPage/PaceLoginPage.tsx +33 -125
  458. package/src/components/PaceLoginPage/useLoginAppAccess.ts +153 -0
  459. package/src/components/PasswordChange/PasswordChangeForm.test.tsx +1 -1
  460. package/src/components/Progress/Progress.test.tsx +127 -1
  461. package/src/components/Progress/Progress.tsx +1 -2
  462. package/src/components/ProtectedRoute/ProtectedRoute.test.tsx +1196 -4
  463. package/src/components/ProtectedRoute/ProtectedRoute.tsx +29 -217
  464. package/src/components/ProtectedRoute/useProtectedRouteState.ts +128 -0
  465. package/src/components/ProtectedRoute/useVisibilityRedirectGrace.ts +89 -0
  466. package/src/components/PublicLayout/PublicLayout.test.tsx +1640 -38
  467. package/src/components/PublicLayout/PublicPageContext.ts +28 -0
  468. package/src/components/PublicLayout/PublicPageLayout.tsx +134 -75
  469. package/src/components/PublicLayout/PublicPageProvider.tsx +7 -42
  470. package/src/components/PublicLayout/usePublicPageContext.ts +36 -0
  471. package/src/components/Select/Select.test.tsx +45 -8
  472. package/src/components/Select/Select.tsx +57 -40
  473. package/src/components/Select/context.test.tsx +56 -0
  474. package/src/components/Select/text.test.tsx +104 -0
  475. package/src/components/Select/text.ts +26 -0
  476. package/src/components/Select/types.ts +3 -0
  477. package/src/components/Select/useSelectEvents.test.ts +279 -0
  478. package/src/components/Select/useSelectEvents.ts +87 -0
  479. package/src/components/Select/useSelectSearch.test.tsx +295 -0
  480. package/src/components/Select/useSelectSearch.ts +91 -0
  481. package/src/components/Select/useSelectState.test.ts +268 -0
  482. package/src/components/Select/useSelectState.ts +104 -0
  483. package/src/components/SessionRestorationLoader/SessionRestorationLoader.test.tsx +28 -112
  484. package/src/components/Switch/Switch.test.tsx +57 -153
  485. package/src/components/Table/Table.test.tsx +395 -317
  486. package/src/components/Tabs/Tabs.test.tsx +270 -0
  487. package/src/components/Tabs/Tabs.tsx +4 -4
  488. package/src/components/Textarea/Textarea.test.tsx +11 -38
  489. package/src/components/Toast/Toast.test.tsx +425 -496
  490. package/src/components/Tooltip/Tooltip.test.tsx +4 -21
  491. package/src/components/UserMenu/UserMenu.test.tsx +1 -21
  492. package/src/components/UserMenu/UserMenu.tsx +0 -1
  493. package/src/components/index.test.ts +346 -0
  494. package/src/components/index.ts +12 -1
  495. package/src/constants/performance.test.ts +91 -0
  496. package/src/hooks/ServiceHooks.test.tsx +725 -0
  497. package/src/hooks/hooks.integration.test.tsx +608 -0
  498. package/src/hooks/index.ts +18 -3
  499. package/src/hooks/index.unit.test.ts +220 -0
  500. package/src/hooks/public/usePublicEvent.test.ts +304 -0
  501. package/src/hooks/public/usePublicEvent.ts +11 -11
  502. package/src/hooks/public/usePublicEventLogo.test.ts +655 -120
  503. package/src/hooks/public/usePublicEventLogo.ts +2 -2
  504. package/src/hooks/public/usePublicRouteParams.test.ts +595 -0
  505. package/src/hooks/public/usePublicRouteParams.ts +2 -2
  506. package/src/hooks/services/useAuth.ts +9 -7
  507. package/src/hooks/services/useAuthService.ts +1 -1
  508. package/src/hooks/services/useEventService.ts +1 -1
  509. package/src/hooks/useAccessibleApps.test.ts +400 -0
  510. package/src/hooks/useAccessibleApps.ts +264 -0
  511. package/src/hooks/useAddressAutocomplete.test.ts +170 -47
  512. package/src/hooks/useAddressAutocomplete.ts +109 -81
  513. package/src/hooks/useApiFetch.unit.test.ts +111 -0
  514. package/src/hooks/useAppConfig.ts +13 -3
  515. package/src/hooks/useAppConfig.unit.test.ts +712 -0
  516. package/src/hooks/useComponentPerformance.unit.test.tsx +314 -0
  517. package/src/hooks/useDataTablePerformance.ts +111 -130
  518. package/src/hooks/useDataTablePerformance.unit.test.ts +720 -0
  519. package/src/hooks/useDataTableState.test.ts +170 -0
  520. package/src/hooks/useDataTableState.ts +5 -5
  521. package/src/hooks/useDebounce.unit.test.ts +157 -0
  522. package/src/hooks/useEventTheme.test.ts +70 -18
  523. package/src/hooks/useEventTheme.ts +50 -22
  524. package/src/hooks/useEvents.ts +49 -2
  525. package/src/hooks/useEvents.unit.test.ts +227 -0
  526. package/src/hooks/useFileReference.test.ts +388 -107
  527. package/src/hooks/useFileReference.ts +184 -179
  528. package/src/hooks/useFileUrl.ts +1 -1
  529. package/src/hooks/useFileUrl.unit.test.ts +686 -0
  530. package/src/hooks/useFileUrlCache.test.ts +319 -0
  531. package/src/hooks/useFileUrlCache.ts +5 -2
  532. package/src/hooks/useFocusManagement.unit.test.ts +604 -0
  533. package/src/hooks/useFocusTrap.unit.test.tsx +613 -0
  534. package/src/hooks/useFormDialog.test.ts +307 -0
  535. package/src/hooks/useFormDialog.ts +2 -2
  536. package/src/hooks/useInactivityTracker.ts +141 -134
  537. package/src/hooks/useInactivityTracker.unit.test.ts +446 -0
  538. package/src/hooks/useIsMobile.unit.test.ts +317 -0
  539. package/src/hooks/useIsPrint.ts +62 -0
  540. package/src/hooks/useIsPrint.unit.test.ts +545 -0
  541. package/src/hooks/useKeyboardShortcuts.unit.test.ts +907 -0
  542. package/src/hooks/useOrganisationPermissions.test.ts +1 -2
  543. package/src/hooks/useOrganisationPermissions.ts +1 -4
  544. package/src/hooks/useOrganisationPermissions.unit.test.tsx +293 -0
  545. package/src/hooks/useOrganisationSecurity.test.ts +4 -33
  546. package/src/hooks/useOrganisationSecurity.ts +192 -203
  547. package/src/hooks/useOrganisationSecurity.unit.test.tsx +959 -0
  548. package/src/hooks/useOrganisations.ts +1 -1
  549. package/src/hooks/useOrganisations.unit.test.ts +369 -0
  550. package/src/hooks/usePerformanceMonitor.ts +1 -1
  551. package/src/hooks/usePerformanceMonitor.unit.test.ts +693 -0
  552. package/src/hooks/usePermissionCache.test.ts +298 -329
  553. package/src/hooks/usePermissionCache.ts +277 -276
  554. package/src/hooks/usePreventTabReload.test.ts +307 -0
  555. package/src/hooks/usePublicEvent.simple.test.ts +794 -0
  556. package/src/hooks/usePublicEvent.test.ts +670 -0
  557. package/src/hooks/usePublicEvent.unit.test.ts +638 -0
  558. package/src/hooks/usePublicFileDisplay.test.ts +948 -0
  559. package/src/hooks/usePublicRouteParams.unit.test.ts +442 -0
  560. package/src/hooks/useQueryCache.test.ts +391 -0
  561. package/src/hooks/useQueryCache.ts +7 -9
  562. package/src/hooks/useRBAC.unit.test.ts +253 -0
  563. package/src/hooks/useSessionDraft.test.ts +556 -0
  564. package/src/hooks/useSessionDraft.ts +14 -11
  565. package/src/hooks/useSessionRestoration.ts +1 -1
  566. package/src/hooks/useSessionRestoration.unit.test.tsx +381 -0
  567. package/src/hooks/useStorage.ts +94 -54
  568. package/src/hooks/useStorage.unit.test.ts +684 -0
  569. package/src/hooks/useToast.test.ts +413 -0
  570. package/src/hooks/useToast.ts +2 -2
  571. package/src/hooks/useToast.unit.test.tsx +481 -0
  572. package/src/hooks/useZodForm.ts +3 -3
  573. package/src/hooks/useZodForm.unit.test.tsx +191 -0
  574. package/src/icons/index.test.ts +133 -0
  575. package/src/icons/index.ts +3 -1
  576. package/src/index.test.ts +528 -0
  577. package/src/index.ts +56 -9
  578. package/src/providers/AuthProvider.test.tsx +218 -0
  579. package/src/providers/EventProvider.test.tsx +487 -0
  580. package/src/providers/InactivityProvider.test-helper.tsx +40 -0
  581. package/src/providers/InactivityProvider.test.tsx +421 -0
  582. package/src/providers/ProviderLifecycle.test.tsx +308 -0
  583. package/src/providers/UnifiedAuthProvider.smoke.test.tsx +7 -12
  584. package/src/providers/UnifiedAuthProvider.test.tsx +503 -0
  585. package/src/providers/index.test.ts +138 -0
  586. package/src/providers/services/AuthServiceContext.ts +27 -0
  587. package/src/providers/services/AuthServiceProvider.integration.test.tsx +229 -0
  588. package/src/providers/services/AuthServiceProvider.test.tsx +638 -0
  589. package/src/providers/services/AuthServiceProvider.tsx +81 -20
  590. package/src/providers/services/EventServiceContext.ts +25 -0
  591. package/src/providers/services/EventServiceProvider.test.tsx +839 -0
  592. package/src/providers/services/EventServiceProvider.tsx +11 -20
  593. package/src/providers/services/InactivityServiceContext.ts +25 -0
  594. package/src/providers/services/InactivityServiceProvider.test.tsx +662 -0
  595. package/src/providers/services/InactivityServiceProvider.tsx +7 -17
  596. package/src/providers/services/OrganisationServiceContext.ts +25 -0
  597. package/src/providers/services/OrganisationServiceProvider.test.tsx +440 -0
  598. package/src/providers/services/OrganisationServiceProvider.tsx +7 -17
  599. package/src/providers/services/UnifiedAuthContext.ts +102 -0
  600. package/src/providers/services/UnifiedAuthProvider.advanced.test.tsx +434 -0
  601. package/src/providers/services/UnifiedAuthProvider.appId.test.tsx +408 -0
  602. package/src/providers/services/UnifiedAuthProvider.integration.test.tsx +304 -0
  603. package/src/providers/services/UnifiedAuthProvider.test.tsx +212 -0
  604. package/src/providers/services/UnifiedAuthProvider.tsx +147 -497
  605. package/src/providers/services/contexts.test.tsx +281 -0
  606. package/src/providers/services/useUnifiedAuth.test.tsx +251 -0
  607. package/src/providers/services/useUnifiedAuth.ts +29 -0
  608. package/src/providers/services/useUnifiedAuthContextValue.ts +279 -0
  609. package/src/providers/useInactivity.test-helper.ts +27 -0
  610. package/src/rbac/README.md +5 -5
  611. package/src/rbac/adapters.comprehensive.test.tsx +429 -0
  612. package/src/rbac/adapters.test.tsx +654 -0
  613. package/src/rbac/adapters.tsx +53 -38
  614. package/src/rbac/api.test.ts +986 -259
  615. package/src/rbac/api.ts +260 -216
  616. package/src/rbac/audit-batched.test.ts +550 -0
  617. package/src/rbac/audit-batched.ts +5 -4
  618. package/src/rbac/audit.test.ts +225 -28
  619. package/src/rbac/audit.ts +26 -18
  620. package/src/rbac/auth-rbac-security.integration.test.tsx +300 -0
  621. package/src/rbac/auth-rbac.e2e.test.tsx +510 -0
  622. package/src/rbac/cache-invalidation.test.ts +715 -0
  623. package/src/rbac/cache-invalidation.ts +18 -15
  624. package/src/rbac/cache.test.ts +123 -63
  625. package/src/rbac/cache.ts +3 -4
  626. package/src/rbac/components/AccessDenied.test.tsx +324 -0
  627. package/src/rbac/components/AccessDenied.tsx +20 -18
  628. package/src/rbac/components/NavigationGuard.test.tsx +1148 -0
  629. package/src/rbac/components/NavigationGuard.tsx +10 -8
  630. package/src/rbac/components/PagePermissionGuard.guard.test.tsx +236 -0
  631. package/src/rbac/components/PagePermissionGuard.performance.test.tsx +252 -0
  632. package/src/rbac/components/PagePermissionGuard.race-condition.test.tsx +243 -0
  633. package/src/rbac/components/PagePermissionGuard.test.tsx +1430 -0
  634. package/src/rbac/components/PagePermissionGuard.tsx +188 -381
  635. package/src/rbac/components/PagePermissionGuard.verification.test.tsx +185 -0
  636. package/src/rbac/config.test.ts +131 -48
  637. package/src/rbac/config.ts +69 -26
  638. package/src/rbac/docs/event-based-apps.md +26 -13
  639. package/src/rbac/engine.comprehensive.test.ts +808 -0
  640. package/src/rbac/engine.test.ts +974 -130
  641. package/src/rbac/engine.ts +53 -13
  642. package/src/rbac/errors.test.ts +99 -87
  643. package/src/rbac/errors.ts +89 -55
  644. package/src/rbac/eslint-rules.js +2 -2
  645. package/src/rbac/hooks/permissions/runPermissionCheck.ts +77 -0
  646. package/src/rbac/hooks/permissions/useAccessLevel.test.ts +622 -0
  647. package/src/rbac/hooks/permissions/useAccessLevel.ts +23 -14
  648. package/src/rbac/hooks/permissions/useCan.test.ts +798 -0
  649. package/src/rbac/hooks/permissions/useCan.ts +173 -253
  650. package/src/rbac/hooks/permissions/useMultiplePermissions.test.ts +843 -0
  651. package/src/rbac/hooks/permissions/useMultiplePermissions.ts +63 -10
  652. package/src/rbac/hooks/permissions/usePermissions.test.ts +543 -0
  653. package/src/rbac/hooks/permissions/usePermissions.ts +50 -78
  654. package/src/rbac/hooks/useCan.test.ts +348 -32
  655. package/src/rbac/hooks/usePageAccessLogging.ts +160 -0
  656. package/src/rbac/hooks/usePageGuardScope.ts +117 -0
  657. package/src/rbac/hooks/usePagePermissionCheck.ts +67 -0
  658. package/src/rbac/hooks/usePermissions.integration.test.ts +427 -0
  659. package/src/rbac/hooks/usePermissions.stability.test.ts +268 -0
  660. package/src/rbac/hooks/usePermissions.test.ts +459 -33
  661. package/src/rbac/hooks/usePermissions.ts +5 -7
  662. package/src/rbac/hooks/useRBAC.test.ts +1784 -21
  663. package/src/rbac/hooks/useRBAC.ts +148 -88
  664. package/src/rbac/hooks/useResolvedScope.test.ts +442 -5
  665. package/src/rbac/hooks/useResolvedScope.ts +4 -1
  666. package/src/rbac/hooks/useResourcePermissions.test.ts +561 -24
  667. package/src/rbac/hooks/useResourcePermissions.ts +76 -140
  668. package/src/rbac/hooks/useResourcePermissionsSuperAdmin.ts +67 -0
  669. package/src/rbac/hooks/useRoleManagement.test.ts +634 -61
  670. package/src/rbac/hooks/useRoleManagement.ts +158 -586
  671. package/src/rbac/hooks/useSecureSupabase.test.ts +1179 -0
  672. package/src/rbac/hooks/useSecureSupabase.ts +21 -14
  673. package/src/rbac/hooks/useSuperAdminCheck.ts +80 -0
  674. package/src/rbac/index.test.ts +107 -0
  675. package/src/rbac/index.ts +32 -32
  676. package/src/rbac/performance.test.ts +451 -0
  677. package/src/rbac/permissions.test.ts +149 -68
  678. package/src/rbac/permissions.ts +0 -3
  679. package/src/rbac/rbac-core.test.tsx +276 -0
  680. package/src/rbac/rbac-engine-core-logic.test.ts +387 -0
  681. package/src/rbac/rbac-engine-simplified.test.ts +252 -0
  682. package/src/rbac/rbac-functions.test.ts +703 -0
  683. package/src/rbac/rbac-integration.test.ts +523 -0
  684. package/src/rbac/rbac-role-isolation.test.ts +456 -0
  685. package/src/rbac/request-deduplication.test.ts +352 -0
  686. package/src/rbac/request-deduplication.ts +5 -4
  687. package/src/rbac/scenarios.user-role.test.tsx +271 -0
  688. package/src/rbac/secureClient.test.ts +499 -115
  689. package/src/rbac/secureClient.ts +54 -28
  690. package/src/rbac/security.test.ts +448 -44
  691. package/src/rbac/security.ts +7 -6
  692. package/src/rbac/types/roleManagement.ts +66 -0
  693. package/src/rbac/types.test.ts +236 -0
  694. package/src/rbac/types.ts +7 -5
  695. package/src/rbac/utils/clientSecurity.test.ts +192 -0
  696. package/src/rbac/utils/clientSecurity.ts +6 -4
  697. package/src/rbac/utils/contextValidator.test.ts +126 -0
  698. package/src/rbac/utils/contextValidator.ts +6 -3
  699. package/src/rbac/utils/deep-equal.test.ts +76 -0
  700. package/src/rbac/utils/eventContext.test.ts +401 -0
  701. package/src/rbac/utils/eventContext.ts +38 -34
  702. package/src/rbac/utils/fetchPermissionMap.ts +13 -0
  703. package/src/rbac/utils/permissionMapHelpers.ts +34 -0
  704. package/src/rbac/utils/roleManagementRpc.ts +303 -0
  705. package/src/services/AuthService.edge-cases.test.ts +746 -0
  706. package/src/services/AuthService.restoreSession.test.ts +59 -0
  707. package/src/services/AuthService.test.ts +1362 -0
  708. package/src/services/AuthService.ts +197 -216
  709. package/src/services/BaseService.edge-cases.test.ts +506 -0
  710. package/src/services/BaseService.test.ts +363 -0
  711. package/src/services/EventService.edge-cases.test.ts +636 -0
  712. package/src/services/EventService.eventColours.test.ts +64 -0
  713. package/src/services/EventService.test.ts +1250 -0
  714. package/src/services/EventService.ts +244 -315
  715. package/src/services/InactivityService.edge-cases.test.ts +492 -0
  716. package/src/services/InactivityService.lifecycle.test.ts +406 -0
  717. package/src/services/InactivityService.test.ts +829 -0
  718. package/src/services/InactivityService.ts +172 -213
  719. package/src/services/OrganisationService.edge-cases.test.ts +633 -0
  720. package/src/services/OrganisationService.pagination.test.ts +409 -0
  721. package/src/services/OrganisationService.test.ts +1579 -0
  722. package/src/services/OrganisationService.ts +186 -257
  723. package/src/services/base/BaseService.test.ts +214 -0
  724. package/src/services/interfaces/IAuthService.test.ts +184 -0
  725. package/src/services/interfaces/IAuthService.ts +10 -9
  726. package/src/services/interfaces/IEventService.test.ts +176 -0
  727. package/src/services/interfaces/IInactivityService.test.ts +183 -0
  728. package/src/services/interfaces/IOrganisationService.test.ts +207 -0
  729. package/src/services/interfaces/IOrganisationService.ts +0 -1
  730. package/src/styles/core.css +244 -12
  731. package/src/theming/parseEventColours.test.ts +321 -0
  732. package/src/theming/parseEventColours.ts +18 -9
  733. package/src/theming/runtime.test.ts +495 -0
  734. package/src/theming/runtime.ts +72 -7
  735. package/src/types/api-result.ts +53 -0
  736. package/src/types/auth.ts +0 -1
  737. package/src/types/core.test.ts +397 -0
  738. package/src/types/database-generated.test.ts +78 -0
  739. package/src/types/database.generated.ts +45 -10
  740. package/src/types/event.ts +39 -19
  741. package/src/types/file-reference.test.ts +351 -0
  742. package/src/types/file-reference.ts +37 -12
  743. package/src/types/guards.test.ts +246 -0
  744. package/src/types/index.test.ts +265 -0
  745. package/src/types/index.ts +3 -0
  746. package/src/types/organisation.roles.test.ts +55 -0
  747. package/src/types/organisation.test.ts +1105 -0
  748. package/src/types/organisation.ts +15 -15
  749. package/src/types/rpc-responses.ts +33 -0
  750. package/src/types/supabase.ts +14 -6
  751. package/src/types/theme.test.ts +830 -0
  752. package/src/types/type-validation.test.ts +526 -0
  753. package/src/types/validation.test.ts +729 -0
  754. package/src/types/vitest-globals.d.ts +1 -1
  755. package/src/utils/app/appConfig.test.ts +235 -0
  756. package/src/utils/app/appIdResolver.test.ts +252 -57
  757. package/src/utils/app/appIdResolver.ts +31 -20
  758. package/src/utils/app/appNameResolver.test.ts +18 -10
  759. package/src/utils/app/appNameResolver.ts +11 -9
  760. package/src/utils/app/appPortMap.test.ts +125 -0
  761. package/src/utils/app/appPortMap.ts +51 -0
  762. package/src/utils/app/buildAppUrl.test.ts +273 -0
  763. package/src/utils/app/buildAppUrl.ts +114 -0
  764. package/src/utils/appConfig.unit.test.ts +55 -0
  765. package/src/utils/audit/audit.test.ts +354 -39
  766. package/src/utils/audit.unit.test.ts +69 -0
  767. package/src/utils/auth-utils.unit.test.ts +69 -0
  768. package/src/utils/bundleAnalysis.unit.test.ts +326 -0
  769. package/src/utils/cn.unit.test.ts +34 -0
  770. package/src/utils/context/organisationContext.test.ts +115 -95
  771. package/src/utils/context/organisationContext.ts +32 -43
  772. package/src/utils/context/sessionTracking.test.ts +354 -0
  773. package/src/utils/core/cn.test.ts +66 -0
  774. package/src/utils/core/debugLogger.test.ts +113 -0
  775. package/src/utils/core/debugLogger.ts +15 -8
  776. package/src/utils/core/logger.test.ts +217 -0
  777. package/src/utils/core/logger.ts +20 -16
  778. package/src/utils/core/mergeRefs.ts +24 -0
  779. package/src/utils/debugLogger.test.ts +417 -0
  780. package/src/utils/device/deviceFingerprint.test.ts +8 -5
  781. package/src/utils/device/deviceFingerprint.ts +3 -3
  782. package/src/utils/deviceFingerprint.unit.test.ts +818 -0
  783. package/src/utils/dynamic/createLazyComponent.tsx +46 -0
  784. package/src/utils/dynamic/dynamicUtils.test.ts +185 -0
  785. package/src/utils/dynamic/dynamicUtils.ts +6 -6
  786. package/src/utils/dynamic/lazyLoad.test.tsx +156 -0
  787. package/src/utils/dynamic/lazyLoad.tsx +8 -36
  788. package/src/utils/dynamic/papaparseLoader.ts +7 -0
  789. package/src/utils/dynamicUtils.unit.test.ts +331 -0
  790. package/src/utils/file-reference/file-reference.test.ts +1238 -0
  791. package/src/utils/file-reference/index.ts +330 -348
  792. package/src/utils/formatDate.unit.test.ts +109 -0
  793. package/src/utils/formatting/formatDate.test.ts +22 -148
  794. package/src/utils/formatting/formatDateTime.test.ts +41 -119
  795. package/src/utils/formatting/formatDateTimeTimezone.test.ts +41 -85
  796. package/src/utils/formatting/formatNumber.test.ts +259 -0
  797. package/src/utils/formatting/formatTime.test.ts +36 -128
  798. package/src/utils/formatting/formatting.ts +1 -1
  799. package/src/utils/formatting.unit.test.ts +99 -0
  800. package/src/utils/google-places/googlePlacesUtils.test.ts +127 -36
  801. package/src/utils/google-places/googlePlacesUtils.ts +67 -86
  802. package/src/utils/google-places/loadGoogleMapsScript.test.ts +68 -8
  803. package/src/utils/google-places/loadGoogleMapsScript.ts +140 -118
  804. package/src/utils/index.ts +52 -11
  805. package/src/utils/index.unit.test.ts +251 -0
  806. package/src/utils/lazyLoad.unit.test.tsx +319 -0
  807. package/src/utils/location/location.test.ts +19 -116
  808. package/src/utils/logger.unit.test.ts +398 -0
  809. package/src/utils/organisationContext.unit.test.ts +180 -0
  810. package/src/utils/performance/bundleAnalysis.test.ts +148 -0
  811. package/src/utils/performance/bundleAnalysis.ts +16 -22
  812. package/src/utils/performance/performanceBenchmark.test.ts +251 -0
  813. package/src/utils/performance/performanceBenchmark.ts +12 -4
  814. package/src/utils/performance/performanceBudgets.test.ts +241 -0
  815. package/src/utils/performance/performanceBudgets.ts +9 -6
  816. package/src/utils/performanceBenchmark.test.ts +174 -0
  817. package/src/utils/performanceBudgets.unit.test.ts +288 -0
  818. package/src/utils/permissionTypes.unit.test.ts +250 -0
  819. package/src/utils/permissionUtils.unit.test.ts +362 -0
  820. package/src/utils/permissions/permissionTypes.test.ts +149 -0
  821. package/src/utils/permissions/permissionUtils.test.ts +20 -42
  822. package/src/utils/persistence/keyDerivation.test.ts +306 -0
  823. package/src/utils/persistence/sensitiveFieldDetection.test.ts +271 -0
  824. package/src/utils/persistence/sensitiveFieldDetection.ts +2 -2
  825. package/src/utils/request-deduplication.test.ts +349 -0
  826. package/src/utils/request-deduplication.ts +6 -4
  827. package/src/utils/sanitization.unit.test.ts +346 -0
  828. package/src/utils/schemaUtils.unit.test.ts +441 -0
  829. package/src/utils/secureDataAccess.unit.test.ts +334 -0
  830. package/src/utils/secureErrors.unit.test.ts +390 -0
  831. package/src/utils/secureStorage.unit.test.ts +289 -0
  832. package/src/utils/security/auth-utils.ts +38 -27
  833. package/src/utils/security/secureDataAccess.test.ts +22 -191
  834. package/src/utils/security/secureDataAccess.ts +241 -281
  835. package/src/utils/security/secureErrors.test.ts +163 -0
  836. package/src/utils/security/secureStorage.test.ts +156 -0
  837. package/src/utils/security/secureStorage.ts +1 -1
  838. package/src/utils/security/security.test.ts +212 -0
  839. package/src/utils/security/security.ts +15 -18
  840. package/src/utils/security/securityMonitor.test.ts +90 -0
  841. package/src/utils/security/securityMonitor.ts +1 -1
  842. package/src/utils/security.unit.test.ts +155 -0
  843. package/src/utils/securityMonitor.unit.test.ts +276 -0
  844. package/src/utils/sessionTracking.unit.test.ts +218 -0
  845. package/src/utils/storage/config.unit.test.ts +239 -0
  846. package/src/utils/storage/helpers.test.ts +769 -456
  847. package/src/utils/storage/helpers.ts +174 -253
  848. package/src/utils/storage/index.unit.test.ts +68 -0
  849. package/src/utils/storage/storageUtils.ts +32 -0
  850. package/src/utils/storage/types.ts +9 -2
  851. package/src/utils/supabase/createBaseClient.test.ts +201 -0
  852. package/src/utils/supabase/createBaseClient.ts +2 -1
  853. package/src/utils/timezone/timezone.test.ts +26 -44
  854. package/src/utils/timezone.test.ts +345 -0
  855. package/src/utils/validation/common.test.ts +115 -0
  856. package/src/utils/validation/csrf.test.ts +198 -0
  857. package/src/utils/validation/csrf.ts +42 -41
  858. package/src/utils/validation/htmlSanitization.ts +27 -31
  859. package/src/utils/validation/htmlSanitization.unit.test.ts +618 -0
  860. package/src/utils/validation/passwordSchema.test.ts +164 -0
  861. package/src/utils/validation/schema.test.ts +127 -0
  862. package/src/utils/validation/schema.ts +6 -3
  863. package/src/utils/validation/sqlInjectionProtection.test.ts +165 -0
  864. package/src/utils/validation/sqlInjectionProtection.ts +2 -2
  865. package/src/utils/validation/user.test.ts +173 -0
  866. package/src/utils/validation/validation.test.ts +197 -0
  867. package/src/utils/validation/validationUtils.test.ts +294 -0
  868. package/src/utils/validation.unit.test.ts +307 -0
  869. package/src/utils/validationUtils.unit.test.ts +558 -0
  870. package/src/vite-env.d.ts +6 -0
  871. package/dist/AuthService-DmfO5rGS.d.ts +0 -524
  872. package/dist/DataTable-DRUIgtUH.d.ts +0 -166
  873. package/dist/DataTable-SOAFXIWY.js +0 -15
  874. package/dist/PublicPageProvider-CIGSujI2.d.ts +0 -4147
  875. package/dist/UnifiedAuthProvider-7SNDOWYD.js +0 -7
  876. package/dist/UnifiedAuthProvider-CKvHP1MK.d.ts +0 -139
  877. package/dist/api-7P7DI652.js +0 -4
  878. package/dist/audit-MYQXYZFU.js +0 -3
  879. package/dist/auth-BZOJqrdd.d.ts +0 -49
  880. package/dist/chunk-4DDCYDQ3.js +0 -544
  881. package/dist/chunk-5HNSDQWH.js +0 -5046
  882. package/dist/chunk-5W2A3DRC.js +0 -164
  883. package/dist/chunk-6GLLNA6U.js +0 -31
  884. package/dist/chunk-7ILTDCL2.js +0 -80
  885. package/dist/chunk-A3W6LW53.js +0 -70
  886. package/dist/chunk-AHU7G2R5.js +0 -423
  887. package/dist/chunk-C7ZQ5O4C.js +0 -481
  888. package/dist/chunk-EF2UGZWY.js +0 -611
  889. package/dist/chunk-FEJLJNWA.js +0 -181
  890. package/dist/chunk-FYHN4DD5.js +0 -415
  891. package/dist/chunk-GS5672WG.js +0 -2003
  892. package/dist/chunk-HF6O3O37.js +0 -187
  893. package/dist/chunk-J2U36LHD.js +0 -8517
  894. package/dist/chunk-LX6U42O3.js +0 -2177
  895. package/dist/chunk-MPBLMWVR.js +0 -2161
  896. package/dist/chunk-OJ4SKRSV.js +0 -105
  897. package/dist/chunk-S6ZQKDY6.js +0 -62
  898. package/dist/chunk-S7DKJPLT.js +0 -699
  899. package/dist/chunk-T5CVK4R3.js +0 -2816
  900. package/dist/chunk-TTRFSOKR.js +0 -121
  901. package/dist/chunk-Z2FNRKF3.js +0 -994
  902. package/dist/database.generated-DT8JTZiP.d.ts +0 -9406
  903. package/dist/event-CW5YB_2p.d.ts +0 -239
  904. package/dist/file-reference-BavO2eQj.d.ts +0 -148
  905. package/dist/functions-lBy5L2ry.d.ts +0 -208
  906. package/dist/timezone-0AyangqX.d.ts +0 -697
  907. package/dist/types-BeoeWV5I.d.ts +0 -110
  908. package/dist/types-DXstZpNI.d.ts +0 -614
  909. package/dist/types-t9H8qKRw.d.ts +0 -55
  910. package/dist/usePublicRouteParams-DQLrDqDb.d.ts +0 -876
  911. package/dist/useToast-AyaT-x7p.d.ts +0 -68
  912. package/dist/validation-643vUDZW.d.ts +0 -177
  913. package/scripts/build-docs-incremental.js +0 -179
  914. package/scripts/eslint-audit.cjs +0 -123
  915. package/scripts/generate-docs.js +0 -157
  916. package/scripts/install-cursor-rules.cjs +0 -255
  917. package/scripts/install-eslint-config.cjs +0 -349
  918. package/scripts/setup-build-cache.js +0 -73
  919. package/scripts/validate-pre-publish.js +0 -145
  920. package/src/__tests__/helpers/__tests__/component-test-utils.test.tsx +0 -260
  921. package/src/__tests__/helpers/__tests__/optimized-test-setup.test.ts +0 -224
  922. package/src/__tests__/helpers/__tests__/supabaseMock.test.ts +0 -273
  923. package/src/__tests__/helpers/__tests__/test-providers.test.tsx +0 -99
  924. package/src/__tests__/helpers/__tests__/test-utils.test.tsx +0 -448
  925. package/src/__tests__/helpers/__tests__/timer-utils.test.ts +0 -371
  926. package/src/__tests__/hooks/usePermissions.test.ts +0 -268
  927. package/src/__tests__/integration/UserProfile.test.tsx +0 -124
  928. package/src/__tests__/public-recipe-view.test.ts +0 -228
  929. package/src/__tests__/rbac/PagePermissionGuard.test.tsx +0 -220
  930. package/src/__tests__/rls-policies.test.ts +0 -471
  931. package/src/components/DataTable/__tests__/DataTable.comprehensive.test.tsx +0 -759
  932. package/src/components/DataTable/__tests__/DataTable.default-state.test.tsx +0 -524
  933. package/src/components/DataTable/__tests__/DataTable.export.test.tsx +0 -705
  934. package/src/components/DataTable/__tests__/DataTable.grouping-aggregation.test.tsx +0 -658
  935. package/src/components/DataTable/__tests__/DataTable.hooks.test.tsx +0 -192
  936. package/src/components/DataTable/__tests__/DataTable.select-label-display.test.tsx +0 -483
  937. package/src/components/DataTable/__tests__/DataTable.test.tsx +0 -876
  938. package/src/components/DataTable/__tests__/DataTableCore.test-setup.ts +0 -220
  939. package/src/components/DataTable/__tests__/DataTableCore.test.tsx +0 -1474
  940. package/src/components/DataTable/__tests__/README.md +0 -145
  941. package/src/components/DataTable/__tests__/a11y.basic.test.tsx +0 -788
  942. package/src/components/DataTable/__tests__/keyboard.test.tsx +0 -756
  943. package/src/components/DataTable/__tests__/mocks/MockRBACProvider.tsx +0 -66
  944. package/src/components/DataTable/__tests__/pagination.modes.test.tsx +0 -730
  945. package/src/components/DataTable/__tests__/ssr.strict-mode.test.tsx +0 -325
  946. package/src/components/DataTable/__tests__/styles.test.ts +0 -382
  947. package/src/components/DataTable/__tests__/test-utils/dataFactories.ts +0 -103
  948. package/src/components/DataTable/__tests__/test-utils/sharedTestUtils.tsx +0 -380
  949. package/src/components/DataTable/__tests__/test-utils.ts +0 -94
  950. package/src/components/DataTable/components/AccessDeniedPage.tsx +0 -159
  951. package/src/components/DataTable/components/ActionButtons.tsx +0 -190
  952. package/src/components/DataTable/components/BulkOperationsDropdown.tsx +0 -160
  953. package/src/components/DataTable/components/ColumnFilter.tsx +0 -118
  954. package/src/components/DataTable/components/ColumnVisibilityDropdown.tsx +0 -114
  955. package/src/components/DataTable/components/DataTableErrorBoundary.tsx +0 -225
  956. package/src/components/DataTable/components/DataTableLayout.tsx +0 -573
  957. package/src/components/DataTable/components/DataTableModals.tsx +0 -245
  958. package/src/components/DataTable/components/DataTableToolbar.tsx +0 -271
  959. package/src/components/DataTable/components/EditFields.tsx +0 -327
  960. package/src/components/DataTable/components/EditableRow.tsx +0 -462
  961. package/src/components/DataTable/components/EmptyState.tsx +0 -79
  962. package/src/components/DataTable/components/FilterRow.tsx +0 -141
  963. package/src/components/DataTable/components/LoadingState.tsx +0 -17
  964. package/src/components/DataTable/components/PaginationControls.tsx +0 -289
  965. package/src/components/DataTable/components/RowComponent.tsx +0 -403
  966. package/src/components/DataTable/components/SortIndicator.tsx +0 -50
  967. package/src/components/DataTable/components/UnifiedTableBody.tsx +0 -355
  968. package/src/components/DataTable/components/__tests__/AccessDeniedPage.test.tsx +0 -657
  969. package/src/components/DataTable/components/__tests__/ActionButtons.test.tsx +0 -913
  970. package/src/components/DataTable/components/__tests__/BulkOperationsDropdown.test.tsx +0 -572
  971. package/src/components/DataTable/components/__tests__/ColumnFilter.test.tsx +0 -612
  972. package/src/components/DataTable/components/__tests__/ColumnVisibilityDropdown.test.tsx +0 -708
  973. package/src/components/DataTable/components/__tests__/DataTableErrorBoundary.test.tsx +0 -479
  974. package/src/components/DataTable/components/__tests__/DataTableModals.test.tsx +0 -475
  975. package/src/components/DataTable/components/__tests__/DataTableToolbar.test.tsx +0 -157
  976. package/src/components/DataTable/components/__tests__/EditableRow.test.tsx +0 -1061
  977. package/src/components/DataTable/components/__tests__/EmptyState.test.tsx +0 -437
  978. package/src/components/DataTable/components/__tests__/FilterRow.test.tsx +0 -474
  979. package/src/components/DataTable/components/__tests__/GroupingDropdown.test.tsx +0 -617
  980. package/src/components/DataTable/components/__tests__/ImportModal.test.tsx +0 -1093
  981. package/src/components/DataTable/components/__tests__/LoadingState.test.tsx +0 -139
  982. package/src/components/DataTable/components/__tests__/PaginationControls.test.tsx +0 -519
  983. package/src/components/DataTable/components/__tests__/UnifiedTableBody.test.tsx +0 -1004
  984. package/src/components/DataTable/components/cellValueUtils.ts +0 -40
  985. package/src/components/DataTable/components/hooks/useImportModalFocus.ts +0 -53
  986. package/src/components/DataTable/components/hooks/usePermissionTracking.ts +0 -122
  987. package/src/components/DataTable/components/index.ts +0 -16
  988. package/src/components/DataTable/context/__tests__/DataTableContext.test.tsx +0 -342
  989. package/src/components/DataTable/core/ActionManager.ts +0 -235
  990. package/src/components/DataTable/core/ColumnManager.ts +0 -205
  991. package/src/components/DataTable/core/DataManager.ts +0 -188
  992. package/src/components/DataTable/core/LocalDataAdapter.ts +0 -274
  993. package/src/components/DataTable/core/PluginRegistry.ts +0 -229
  994. package/src/components/DataTable/core/StateManager.ts +0 -312
  995. package/src/components/DataTable/core/__tests__/ActionManager.test.ts +0 -123
  996. package/src/components/DataTable/core/__tests__/ColumnFactory.test.ts +0 -305
  997. package/src/components/DataTable/core/__tests__/ColumnManager.test.ts +0 -84
  998. package/src/components/DataTable/core/__tests__/DataManager.test.ts +0 -115
  999. package/src/components/DataTable/core/__tests__/LocalDataAdapter.test.ts +0 -100
  1000. package/src/components/DataTable/core/__tests__/PluginRegistry.test.ts +0 -120
  1001. package/src/components/DataTable/core/__tests__/StateManager.test.ts +0 -104
  1002. package/src/components/DataTable/core/index.ts +0 -1
  1003. package/src/components/DataTable/core/interfaces.ts +0 -338
  1004. package/src/components/DataTable/hooks/__tests__/useColumnOrderPersistence.test.ts +0 -521
  1005. package/src/components/DataTable/hooks/__tests__/useColumnVisibilityPersistence.test.ts +0 -167
  1006. package/src/components/DataTable/hooks/__tests__/useDataTableConfiguration.test.ts +0 -124
  1007. package/src/components/DataTable/hooks/__tests__/useDataTableDataPipeline.test.ts +0 -117
  1008. package/src/components/DataTable/hooks/__tests__/useDataTablePermissions.test.ts +0 -102
  1009. package/src/components/DataTable/hooks/__tests__/useDataTableState.test.ts +0 -596
  1010. package/src/components/DataTable/hooks/__tests__/useEffectiveColumnOrder.test.ts +0 -53
  1011. package/src/components/DataTable/hooks/__tests__/useHierarchicalState.test.ts +0 -214
  1012. package/src/components/DataTable/hooks/__tests__/useTableColumns.test.ts +0 -448
  1013. package/src/components/DataTable/hooks/index.ts +0 -13
  1014. package/src/components/DataTable/types.ts +0 -761
  1015. package/src/components/DataTable/utils/__tests__/a11yUtils.test.ts +0 -612
  1016. package/src/components/DataTable/utils/__tests__/columnUtils.test.ts +0 -94
  1017. package/src/components/DataTable/utils/__tests__/errorHandling.test.ts +0 -266
  1018. package/src/components/DataTable/utils/__tests__/exportUtils.test.ts +0 -954
  1019. package/src/components/DataTable/utils/__tests__/flexibleImport.test.ts +0 -573
  1020. package/src/components/DataTable/utils/__tests__/hierarchicalSorting.test.ts +0 -247
  1021. package/src/components/DataTable/utils/__tests__/hierarchicalUtils.test.ts +0 -570
  1022. package/src/components/DataTable/utils/__tests__/performanceUtils.test.ts +0 -470
  1023. package/src/components/DataTable/utils/__tests__/rowUtils.test.ts +0 -251
  1024. package/src/components/DataTable/utils/__tests__/selectFieldUtils.test.ts +0 -207
  1025. package/src/components/DataTable/utils/index.ts +0 -10
  1026. package/src/components/PublicLayout/index.ts +0 -32
  1027. package/src/components/Select/hooks/useSelectEvents.ts +0 -87
  1028. package/src/components/Select/hooks/useSelectSearch.ts +0 -91
  1029. package/src/components/Select/hooks/useSelectState.ts +0 -104
  1030. package/src/components/Select/utils/text.ts +0 -26
  1031. package/src/hooks/__tests__/ServiceHooks.test.tsx +0 -615
  1032. package/src/hooks/__tests__/hooks.integration.test.tsx +0 -607
  1033. package/src/hooks/__tests__/index.unit.test.ts +0 -220
  1034. package/src/hooks/__tests__/useApiFetch.unit.test.ts +0 -111
  1035. package/src/hooks/__tests__/useAppConfig.unit.test.ts +0 -347
  1036. package/src/hooks/__tests__/useComponentPerformance.unit.test.tsx +0 -144
  1037. package/src/hooks/__tests__/useDataTablePerformance.unit.test.ts +0 -776
  1038. package/src/hooks/__tests__/useDataTableState.test.ts +0 -76
  1039. package/src/hooks/__tests__/useDebounce.unit.test.ts +0 -82
  1040. package/src/hooks/__tests__/useEvents.unit.test.ts +0 -252
  1041. package/src/hooks/__tests__/useFileDisplay.unit.test.ts +0 -1112
  1042. package/src/hooks/__tests__/useFileUrl.unit.test.ts +0 -916
  1043. package/src/hooks/__tests__/useFileUrlCache.test.ts +0 -129
  1044. package/src/hooks/__tests__/useFocusManagement.unit.test.ts +0 -230
  1045. package/src/hooks/__tests__/useFocusTrap.unit.test.tsx +0 -828
  1046. package/src/hooks/__tests__/useFormDialog.test.ts +0 -478
  1047. package/src/hooks/__tests__/useInactivityTracker.unit.test.ts +0 -446
  1048. package/src/hooks/__tests__/useIsMobile.unit.test.ts +0 -317
  1049. package/src/hooks/__tests__/useKeyboardShortcuts.unit.test.ts +0 -910
  1050. package/src/hooks/__tests__/useOrganisationPermissions.unit.test.tsx +0 -294
  1051. package/src/hooks/__tests__/useOrganisationSecurity.unit.test.tsx +0 -961
  1052. package/src/hooks/__tests__/useOrganisations.unit.test.ts +0 -369
  1053. package/src/hooks/__tests__/usePerformanceMonitor.unit.test.ts +0 -694
  1054. package/src/hooks/__tests__/usePermissionCache.simple.test.ts +0 -192
  1055. package/src/hooks/__tests__/usePermissionCache.unit.test.ts +0 -741
  1056. package/src/hooks/__tests__/usePreventTabReload.test.ts +0 -88
  1057. package/src/hooks/__tests__/usePublicEvent.simple.test.ts +0 -785
  1058. package/src/hooks/__tests__/usePublicEvent.test.ts +0 -678
  1059. package/src/hooks/__tests__/usePublicEvent.unit.test.ts +0 -630
  1060. package/src/hooks/__tests__/usePublicFileDisplay.test.ts +0 -951
  1061. package/src/hooks/__tests__/usePublicRouteParams.unit.test.ts +0 -443
  1062. package/src/hooks/__tests__/useQueryCache.test.ts +0 -144
  1063. package/src/hooks/__tests__/useRBAC.unit.test.ts +0 -236
  1064. package/src/hooks/__tests__/useSessionDraft.test.ts +0 -163
  1065. package/src/hooks/__tests__/useSessionRestoration.unit.test.tsx +0 -390
  1066. package/src/hooks/__tests__/useStorage.unit.test.ts +0 -751
  1067. package/src/hooks/__tests__/useToast.unit.test.tsx +0 -481
  1068. package/src/hooks/__tests__/useZodForm.unit.test.tsx +0 -37
  1069. package/src/hooks/public/index.ts +0 -36
  1070. package/src/hooks/public/usePublicFileDisplay.ts +0 -504
  1071. package/src/hooks/useFileDisplay.ts +0 -715
  1072. package/src/providers/OrganisationProvider.tsx +0 -92
  1073. package/src/providers/__tests__/AuthProvider.test.tsx +0 -287
  1074. package/src/providers/__tests__/EventProvider.test.tsx +0 -551
  1075. package/src/providers/__tests__/InactivityProvider.test-helper.tsx +0 -65
  1076. package/src/providers/__tests__/InactivityProvider.test.tsx +0 -572
  1077. package/src/providers/__tests__/OrganisationProvider.test.tsx +0 -617
  1078. package/src/providers/__tests__/ProviderLifecycle.test.tsx +0 -424
  1079. package/src/providers/__tests__/UnifiedAuthProvider.test.tsx +0 -596
  1080. package/src/providers/services/__tests__/AuthServiceProvider.integration.test.tsx +0 -263
  1081. package/src/providers/services/__tests__/UnifiedAuthProvider.integration.test.tsx +0 -294
  1082. package/src/rbac/__tests__/adapters.comprehensive.test.tsx +0 -434
  1083. package/src/rbac/__tests__/auth-rbac-security.integration.test.tsx +0 -313
  1084. package/src/rbac/__tests__/auth-rbac.e2e.test.tsx +0 -486
  1085. package/src/rbac/__tests__/cache-invalidation.test.ts +0 -399
  1086. package/src/rbac/__tests__/engine.comprehensive.test.ts +0 -813
  1087. package/src/rbac/__tests__/isSuperAdmin.real.test.ts +0 -82
  1088. package/src/rbac/__tests__/rbac-core.test.tsx +0 -276
  1089. package/src/rbac/__tests__/rbac-engine-core-logic.test.ts +0 -392
  1090. package/src/rbac/__tests__/rbac-engine-simplified.test.ts +0 -258
  1091. package/src/rbac/__tests__/rbac-functions.test.ts +0 -647
  1092. package/src/rbac/__tests__/rbac-integration.test.ts +0 -524
  1093. package/src/rbac/__tests__/rbac-role-isolation.test.ts +0 -456
  1094. package/src/rbac/__tests__/scenarios.user-role.test.tsx +0 -282
  1095. package/src/rbac/audit-enhanced.ts +0 -384
  1096. package/src/rbac/compliance/database-validator.ts +0 -165
  1097. package/src/rbac/compliance/index.ts +0 -48
  1098. package/src/rbac/compliance/pattern-detector.ts +0 -553
  1099. package/src/rbac/compliance/quick-fix-suggestions.ts +0 -209
  1100. package/src/rbac/compliance/runtime-compliance.ts +0 -99
  1101. package/src/rbac/compliance/setup-validator.ts +0 -131
  1102. package/src/rbac/components/__tests__/NavigationGuard.test.tsx +0 -975
  1103. package/src/rbac/components/__tests__/PagePermissionGuard.performance.test.tsx +0 -248
  1104. package/src/rbac/components/__tests__/PagePermissionGuard.race-condition.test.tsx +0 -242
  1105. package/src/rbac/components/__tests__/PagePermissionGuard.test.tsx +0 -1107
  1106. package/src/rbac/components/__tests__/PagePermissionGuard.verification.test.tsx +0 -184
  1107. package/src/rbac/components/index.ts +0 -26
  1108. package/src/rbac/hooks/__tests__/usePermissions.integration.test.ts +0 -432
  1109. package/src/rbac/hooks/__tests__/useSecureSupabase.test.ts +0 -579
  1110. package/src/rbac/hooks/index.ts +0 -34
  1111. package/src/rbac/hooks/permissions/index.ts +0 -4
  1112. package/src/rbac/hooks/useRBAC.simple.test.ts +0 -95
  1113. package/src/rbac/utils/__tests__/contextValidator.test.ts +0 -128
  1114. package/src/rbac/utils/__tests__/deep-equal.test.ts +0 -53
  1115. package/src/rbac/utils/__tests__/eventContext.test.ts +0 -433
  1116. package/src/rbac/utils/__tests__/eventContext.unit.test.ts +0 -490
  1117. package/src/services/__tests__/AuthService.restoreSession.test.ts +0 -39
  1118. package/src/services/__tests__/AuthService.test.ts +0 -1332
  1119. package/src/services/__tests__/BaseService.test.ts +0 -314
  1120. package/src/services/__tests__/EventService.eventColours.test.ts +0 -76
  1121. package/src/services/__tests__/EventService.test.ts +0 -1025
  1122. package/src/services/__tests__/InactivityService.lifecycle.test.ts +0 -411
  1123. package/src/services/__tests__/InactivityService.test.ts +0 -654
  1124. package/src/services/__tests__/OrganisationService.pagination.test.ts +0 -409
  1125. package/src/services/__tests__/OrganisationService.test.ts +0 -1176
  1126. package/src/theming/__tests__/parseEventColours.test.ts +0 -321
  1127. package/src/theming/__tests__/runtime.test.ts +0 -569
  1128. package/src/types/__tests__/file-reference.test.ts +0 -447
  1129. package/src/types/__tests__/guards.test.ts +0 -246
  1130. package/src/types/__tests__/organisation.roles.test.ts +0 -55
  1131. package/src/types/__tests__/organisation.test.ts +0 -1133
  1132. package/src/types/__tests__/theme.test.ts +0 -830
  1133. package/src/types/__tests__/type-validation.test.ts +0 -526
  1134. package/src/types/__tests__/validation.test.ts +0 -731
  1135. package/src/utils/__tests__/appConfig.unit.test.ts +0 -55
  1136. package/src/utils/__tests__/audit.unit.test.ts +0 -69
  1137. package/src/utils/__tests__/auth-utils.unit.test.ts +0 -70
  1138. package/src/utils/__tests__/bundleAnalysis.unit.test.ts +0 -339
  1139. package/src/utils/__tests__/cn.unit.test.ts +0 -34
  1140. package/src/utils/__tests__/debugLogger.test.ts +0 -417
  1141. package/src/utils/__tests__/deviceFingerprint.unit.test.ts +0 -818
  1142. package/src/utils/__tests__/dynamicUtils.unit.test.ts +0 -318
  1143. package/src/utils/__tests__/formatDate.unit.test.ts +0 -109
  1144. package/src/utils/__tests__/formatting.unit.test.ts +0 -99
  1145. package/src/utils/__tests__/index.unit.test.ts +0 -251
  1146. package/src/utils/__tests__/lazyLoad.unit.test.tsx +0 -321
  1147. package/src/utils/__tests__/logger.unit.test.ts +0 -398
  1148. package/src/utils/__tests__/organisationContext.unit.test.ts +0 -191
  1149. package/src/utils/__tests__/performanceBenchmark.test.ts +0 -175
  1150. package/src/utils/__tests__/performanceBudgets.unit.test.ts +0 -253
  1151. package/src/utils/__tests__/permissionTypes.unit.test.ts +0 -250
  1152. package/src/utils/__tests__/permissionUtils.unit.test.ts +0 -362
  1153. package/src/utils/__tests__/sanitization.unit.test.ts +0 -346
  1154. package/src/utils/__tests__/schemaUtils.unit.test.ts +0 -441
  1155. package/src/utils/__tests__/secureDataAccess.unit.test.ts +0 -335
  1156. package/src/utils/__tests__/secureErrors.unit.test.ts +0 -390
  1157. package/src/utils/__tests__/secureStorage.unit.test.ts +0 -289
  1158. package/src/utils/__tests__/security.unit.test.ts +0 -149
  1159. package/src/utils/__tests__/securityMonitor.unit.test.ts +0 -276
  1160. package/src/utils/__tests__/sessionTracking.unit.test.ts +0 -218
  1161. package/src/utils/__tests__/timezone.test.ts +0 -345
  1162. package/src/utils/__tests__/validation.unit.test.ts +0 -308
  1163. package/src/utils/__tests__/validationUtils.unit.test.ts +0 -555
  1164. package/src/utils/app/appNameResolver.simple.test.ts +0 -212
  1165. package/src/utils/file-reference/__tests__/file-reference.test.ts +0 -875
  1166. package/src/utils/google-places/index.ts +0 -26
  1167. package/src/utils/location/index.ts +0 -16
  1168. package/src/utils/persistence/__tests__/keyDerivation.test.ts +0 -135
  1169. package/src/utils/persistence/__tests__/sensitiveFieldDetection.test.ts +0 -123
  1170. package/src/utils/storage/__tests__/helpers.unit.test.ts +0 -332
  1171. package/src/utils/storage/__tests__/index.unit.test.ts +0 -16
  1172. package/src/utils/storage/index.ts +0 -67
  1173. package/src/utils/timezone/index.ts +0 -17
  1174. package/src/utils/validation/__tests__/csrf.test.ts +0 -105
  1175. package/src/utils/validation/__tests__/htmlSanitization.unit.test.ts +0 -598
  1176. package/src/utils/validation/__tests__/sqlInjectionProtection.test.ts +0 -92
  1177. package/src/utils/validation/__tests__/validationUtils.test.ts +0 -72
  1178. package/src/utils/validation/index.ts +0 -73
  1179. /package/src/components/DataTable/{components/__tests__ → ui}/COVERAGE_NOTE.md +0 -0
  1180. /package/src/components/DataTable/utils/{__tests__/COVERAGE_NOTE.md → COVERAGE_NOTE.md} +0 -0
  1181. /package/src/providers/{__tests__/README.md → README.md} +0 -0
  1182. /package/src/types/{__tests__/README.md → README.md} +0 -0
@@ -0,0 +1,1834 @@
1
+ /**
2
+ * @file Import Modal Component Tests
3
+ * @package @jmruthers/pace-core
4
+ * @module Components/DataTable/Components
5
+ * @since 0.4.0
6
+ *
7
+ * Comprehensive test suite for ImportModal component following testing guidelines.
8
+ * Tests cover all major functionality, edge cases, and user interactions.
9
+ */
10
+
11
+ import React from 'react';
12
+ import { render, screen, waitFor, cleanup } from '@testing-library/react';
13
+ import userEvent from '@testing-library/user-event';
14
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
15
+ import { ImportModal } from './ImportModal';
16
+
17
+ // Helper function to wait for dialog to be accessible
18
+ // Native dialog elements are only accessible after showModal() completes
19
+ // In test environments, we use querySelector as fallback since getByRole may not work
20
+ // Note: In test environments (jsdom), dialog.open may not be set even when dialog is rendered
21
+ // Also note: Dialog uses requestAnimationFrame before showModal(), so we need to wait for content
22
+ const waitForDialog = async (): Promise<HTMLElement> => {
23
+ return await waitFor(
24
+ () => {
25
+ // Try getByRole first (works in browsers with full dialog support)
26
+ try {
27
+ const dialog = screen.getByRole('dialog');
28
+ expect(dialog).toBeInTheDocument();
29
+ return dialog;
30
+ } catch (_e) {
31
+ // Fallback: use querySelector for test environments that don't fully support dialog accessibility
32
+ const dialog = document.querySelector('dialog[role="dialog"]') as HTMLDialogElement;
33
+ if (!dialog) {
34
+ throw new Error('Dialog not found in DOM');
35
+ }
36
+ // In test environments, dialog.open may not be set even when dialog is rendered
37
+ // Just check that dialog exists in DOM - that's sufficient for testing
38
+ return dialog;
39
+ }
40
+ },
41
+ { timeout: 5000 }
42
+ );
43
+ };
44
+
45
+ // Helper function to find buttons in dialogs (more reliable than getByRole in test environments)
46
+ const findButtonByText = (text: string | RegExp): HTMLButtonElement | null => {
47
+ // Try getByRole first
48
+ try {
49
+ const button = screen.getByRole('button', { name: text });
50
+ return button as HTMLButtonElement;
51
+ } catch (_e) {
52
+ // Fallback: search all buttons by text content
53
+ const buttons = Array.from(document.querySelectorAll('button'));
54
+ const regex = typeof text === 'string' ? new RegExp(text, 'i') : text;
55
+ return buttons.find(btn => regex.test(btn.textContent || '')) as HTMLButtonElement || null;
56
+ }
57
+ };
58
+
59
+ // Helper function to ensure fresh modal state (clears any persistent state from previous tests)
60
+ // This works by rendering a modal, opening it, clicking close button to trigger handleClose,
61
+ // then closing it programmatically to ensure cleanup
62
+ const ensureFreshModalState = async () => {
63
+ const onClose = vi.fn();
64
+ const user = userEvent.setup();
65
+
66
+ // Render a modal and open it
67
+ const { rerender, unmount } = render(
68
+ <ImportModal
69
+ isOpen={true}
70
+ onClose={onClose}
71
+ onImport={vi.fn()}
72
+ />
73
+ );
74
+
75
+ // Wait for modal to initialize
76
+ try {
77
+ await waitFor(() => {
78
+ const dialog = document.querySelector('dialog[role="dialog"]');
79
+ if (!dialog) throw new Error('Dialog not found');
80
+ }, { timeout: 1000 });
81
+ } catch (_e) {
82
+ // If modal didn't render, that's okay, we'll still close it
83
+ }
84
+
85
+ // Check if there's a summary - if so, click close button to trigger handleClose
86
+ await waitForDialog();
87
+ const hasSummary = screen.queryByText(/import completed/i);
88
+ if (hasSummary) {
89
+ const closeButton = findButtonByText(/^close$/i) || findButtonByText(/cancel/i);
90
+ if (closeButton) {
91
+ await user.click(closeButton);
92
+ await new Promise(resolve => setTimeout(resolve, 100));
93
+ }
94
+ }
95
+
96
+ // Close the modal by setting isOpen to false
97
+ // This triggers the useEffect that clears persistent storage when there's no importSummary
98
+ rerender(
99
+ <ImportModal
100
+ isOpen={false}
101
+ onClose={onClose}
102
+ onImport={vi.fn() as (data: Array<Record<string, unknown>>) => void | Promise<void>}
103
+ />
104
+ );
105
+
106
+ // Wait for React to process the state change and run cleanup effects
107
+ // The component's useEffect (lines 152-173) clears persistent storage when isOpen becomes false
108
+ // We need to wait for this effect to complete
109
+ await new Promise(resolve => setTimeout(resolve, 500));
110
+
111
+ // Unmount to ensure cleanup
112
+ unmount();
113
+ cleanup();
114
+
115
+ // Additional wait to ensure module-level state is cleared
116
+ await new Promise(resolve => setTimeout(resolve, 200));
117
+ };
118
+
119
+ // Helper function to wait for file input to be available
120
+ // This function handles the case where persistent state might have leaked from previous tests
121
+ const waitForFileInput = async (
122
+ rerender?: (element: React.ReactElement) => void,
123
+ props?: { isOpen?: boolean; onClose: () => void; onImport: (data: Array<Record<string, unknown>>) => void | Promise<void>; config?: any }
124
+ ): Promise<HTMLInputElement> => {
125
+ // First wait for dialog to be accessible
126
+ await waitForDialog();
127
+
128
+ // Then wait for dialog content to be rendered (check for title)
129
+ await waitFor(() => {
130
+ try {
131
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
132
+ } catch (_e) {
133
+ // Fallback: check if any element with "Import Data" exists
134
+ const elements = screen.getAllByText('Import Data');
135
+ expect(elements.length).toBeGreaterThan(0);
136
+ }
137
+ }, { timeout: 5000 });
138
+
139
+ // Check if a file is already selected OR if there's a summary (from persistent state leakage)
140
+ const selectedFileText = screen.queryByText(/selected:/i);
141
+ const hasSummary = screen.queryByText(/import completed/i);
142
+
143
+ if ((selectedFileText || hasSummary) && rerender) {
144
+ // Persistent state leaked from a previous test - clear it by closing and reopening the modal
145
+ const closeProps = props || getBaseProps();
146
+
147
+ // Strategy: Always click cancel/close button to trigger handleClose which clears everything
148
+ // This is more reliable than relying on useEffect cleanup
149
+ const closeButton = findButtonByText(/^close$/i) || findButtonByText(/cancel/i);
150
+ if (closeButton) {
151
+ const user = userEvent.setup();
152
+ await user.click(closeButton);
153
+ // Wait for handleClose to execute (clears all persistent storage)
154
+ await new Promise(resolve => setTimeout(resolve, 150));
155
+ }
156
+
157
+ // Close the modal programmatically (onClose is just a mock, so we need to set isOpen=false)
158
+ rerender(<ImportModal {...closeProps} isOpen={false} />);
159
+
160
+ // Wait for modal to close
161
+ await waitFor(() => {
162
+ expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
163
+ }, { timeout: 2000 });
164
+
165
+ // Wait for cleanup effect to run (clears persistent storage)
166
+ // The useEffect at line 152-173 clears persistentFile when isOpen=false and !importSummary
167
+ // Since handleClose already cleared persistentSummary, the cleanup should run
168
+ await new Promise(resolve => setTimeout(resolve, 500));
169
+
170
+ // Reopen the modal with fresh state
171
+ const reopenProps = props || getBaseProps();
172
+ rerender(<ImportModal {...reopenProps} isOpen={true} />);
173
+
174
+ // Wait for dialog to be accessible again
175
+ await waitForDialog();
176
+
177
+ // Wait for dialog content to be rendered
178
+ await waitFor(() => {
179
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
180
+ }, { timeout: 5000 });
181
+
182
+ // Wait for any state restoration to complete (useEffect at line 134-149)
183
+ // But since we cleared persistent storage, nothing should be restored
184
+ await new Promise(resolve => setTimeout(resolve, 300));
185
+
186
+ // Verify state was cleared - check multiple times
187
+ for (let i = 0; i < 5; i++) {
188
+ const stillSelected = screen.queryByText(/selected:/i);
189
+ const stillHasSummary = screen.queryByText(/import completed/i);
190
+ if (!stillSelected && !stillHasSummary) {
191
+ break; // State is cleared
192
+ }
193
+ if (i === 4) {
194
+ // Last attempt - if still not cleared, we have a problem
195
+ throw new Error('Persistent state was not cleared after close/reopen. Still seeing selected file or summary.');
196
+ }
197
+ // Wait a bit more and check again
198
+ await new Promise(resolve => setTimeout(resolve, 200));
199
+ // Re-render to force state refresh
200
+ rerender(<ImportModal {...reopenProps} isOpen={true} />);
201
+ await waitForDialog();
202
+ await new Promise(resolve => setTimeout(resolve, 200));
203
+ }
204
+ } else if (selectedFileText || hasSummary) {
205
+ // Can't clear it without rerender function
206
+ throw new Error('File input not available because a file is already selected or summary exists. The test needs to pass rerender (and optionally props) to waitForFileInput to clear persistent state.');
207
+ }
208
+
209
+ // Wait for file input to be in the DOM (it's only rendered when !file)
210
+ // After clearing state, wait a bit more to ensure component is fully initialized
211
+ if ((selectedFileText || hasSummary) && rerender) {
212
+ await new Promise(resolve => setTimeout(resolve, 200));
213
+ }
214
+
215
+ return await waitFor(() => {
216
+ const fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
217
+ if (!fileInput) {
218
+ // Double-check if file is still selected
219
+ const stillSelected = screen.queryByText(/selected:/i);
220
+ if (stillSelected) {
221
+ throw new Error('File input not found because a file is already selected. Persistent state may not have been cleared.');
222
+ }
223
+ throw new Error('File input not found in DOM. The dialog may not be fully rendered yet.');
224
+ }
225
+ // Ensure file input is enabled and ready
226
+ if (fileInput.disabled) {
227
+ throw new Error('File input is disabled. Component may not be fully initialized.');
228
+ }
229
+ return fileInput;
230
+ }, { timeout: 5000 });
231
+ };
232
+
233
+ // Helper function to upload a file and wait for preview to appear
234
+ const uploadFileAndWaitForPreview = async (user: ReturnType<typeof userEvent.setup>, file: File) => {
235
+ // Wait for dialog and file input
236
+ const fileInput = await waitForFileInput();
237
+
238
+ // Upload the file
239
+ await user.upload(fileInput, file);
240
+
241
+ // Wait for preview table to appear (file processing is async)
242
+ // Try multiple ways to find the table since queryByRole may not work in all test environments
243
+ await waitFor(() => {
244
+ const table = screen.queryByRole('table') ||
245
+ document.querySelector('table') ||
246
+ document.querySelector('table.min-w-full');
247
+ expect(table).toBeInTheDocument();
248
+ }, { timeout: 10000 });
249
+ };
250
+
251
+ // Mock Button component (Dialog uses IconButton from same module - must export both)
252
+ vi.mock('../../Button/Button', () => ({
253
+ Button: ({ children, onClick, variant, size, disabled, className }: any) => (
254
+ <button
255
+ onClick={onClick}
256
+ disabled={disabled}
257
+ data-variant={variant}
258
+ data-size={size}
259
+ className={className}
260
+ >
261
+ {children}
262
+ </button>
263
+ ),
264
+ IconButton: React.forwardRef(({ onClick, icon, 'aria-label': ariaLabel, className, ...props }: any, ref: any) => (
265
+ <button ref={ref} onClick={onClick} aria-label={ariaLabel} className={className} {...props}>
266
+ {icon}
267
+ </button>
268
+ )),
269
+ }));
270
+
271
+ // Mock Input component
272
+ vi.mock('../../Input/Input', () => ({
273
+ Input: React.forwardRef(({ type, accept, onChange, className, ...props }: any, ref: any) => (
274
+ <input
275
+ ref={ref}
276
+ type={type}
277
+ accept={accept}
278
+ onChange={onChange}
279
+ className={className}
280
+ {...props}
281
+ />
282
+ )),
283
+ }));
284
+
285
+ // Mock lucide-react icons
286
+ vi.mock('lucide-react', () => ({
287
+ Upload: ({ className }: { className?: string }) => (
288
+ <span data-testid="upload-icon" className={className}>Upload</span>
289
+ ),
290
+ FileText: ({ className }: { className?: string }) => (
291
+ <span data-testid="file-text-icon" className={className}>File</span>
292
+ ),
293
+ AlertCircle: ({ className }: { className?: string }) => (
294
+ <span data-testid="alert-circle-icon" className={className}>Alert</span>
295
+ ),
296
+ CheckCircle2: ({ className }: { className?: string }) => (
297
+ <span data-testid="check-circle-icon" className={className}>Check</span>
298
+ ),
299
+ X: ({ className }: { className?: string }) => (
300
+ <span data-testid="x-icon" className={className}>X</span>
301
+ ),
302
+ }));
303
+
304
+ // Mock logger
305
+ vi.mock('../../../utils/core/logger', () => ({
306
+ createLogger: () => ({
307
+ debug: vi.fn(),
308
+ info: vi.fn(),
309
+ warn: vi.fn(),
310
+ error: vi.fn(),
311
+ }),
312
+ }));
313
+
314
+ // Base props for tests - defined outside describe so helpers can access it
315
+ const getBaseProps = () => ({
316
+ isOpen: true,
317
+ onClose: vi.fn(),
318
+ onImport: vi.fn(),
319
+ });
320
+
321
+ describe('[component] ImportModal', () => {
322
+ const baseProps = getBaseProps();
323
+
324
+ const createCSVFile = (content: string, filename = 'test.csv'): File => {
325
+ const blob = new Blob([content], { type: 'text/csv' });
326
+ const file = new File([blob], filename, { type: 'text/csv' });
327
+ // Store content for File.text() mock to access
328
+ (file as any)._content = content;
329
+ return file;
330
+ };
331
+
332
+ beforeEach(async () => {
333
+ vi.clearAllMocks();
334
+
335
+ // Clear any rendered components from previous tests
336
+ cleanup();
337
+
338
+ // Mock showModal for dialog elements (needed for test environments)
339
+ // MUST be set up before any components are rendered
340
+ HTMLDialogElement.prototype.showModal = vi.fn(function(this: HTMLDialogElement) {
341
+ this.setAttribute('open', '');
342
+ this.dispatchEvent(new Event('show', { bubbles: true }));
343
+ });
344
+ HTMLDialogElement.prototype.close = vi.fn(function(this: HTMLDialogElement) {
345
+ this.removeAttribute('open');
346
+ this.dispatchEvent(new Event('close', { bubbles: true }));
347
+ });
348
+
349
+ // Mock File.text() method for jsdom compatibility
350
+ // File.text() reads the file content asynchronously
351
+ // In tests, we read directly from the Blob content synchronously
352
+ if (!File.prototype.text) {
353
+ Object.defineProperty(File.prototype, 'text', {
354
+ writable: true,
355
+ configurable: true,
356
+ value: async function(this: File) {
357
+ // For test files, read from the stored content or from the Blob
358
+ const file = this as any;
359
+ if (file._content) {
360
+ // Use stored content if available
361
+ return Promise.resolve(file._content);
362
+ }
363
+ // Otherwise, try to read from Blob using FileReader
364
+ return new Promise((resolve, reject) => {
365
+ const reader = new FileReader();
366
+ reader.onload = (e) => {
367
+ resolve(e.target?.result as string);
368
+ };
369
+ reader.onerror = () => {
370
+ reject(new Error('Failed to read file'));
371
+ };
372
+ // Read the file as text
373
+ reader.readAsText(this);
374
+ });
375
+ },
376
+ });
377
+ }
378
+
379
+ // Clear ImportModal's persistent state by opening and closing a modal
380
+ // The ImportModal uses module-level persistent storage that can leak between tests
381
+ // We need to ensure it's cleared by closing a modal (which calls handleClose and clears persistent storage)
382
+ await ensureFreshModalState();
383
+ });
384
+
385
+ afterEach(() => {
386
+ // Clean up all rendered components to ensure state doesn't leak
387
+ cleanup();
388
+ vi.clearAllMocks();
389
+ });
390
+
391
+ describe('Rendering', () => {
392
+ it('returns null when modal is closed', () => {
393
+ const { container } = render(
394
+ <ImportModal {...baseProps} isOpen={false} />
395
+ );
396
+ expect(container.firstChild).toBeNull();
397
+ });
398
+
399
+ it('renders modal when open', async () => {
400
+ render(<ImportModal {...baseProps} />);
401
+
402
+ // Wait for dialog to be accessible
403
+ await waitForDialog();
404
+ // Check for content instead of testids
405
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
406
+ });
407
+
408
+ it('renders default title', async () => {
409
+ render(<ImportModal {...baseProps} />);
410
+
411
+ // Wait for dialog content to be rendered (showModal is async via requestAnimationFrame)
412
+ await waitFor(() => {
413
+ // Try by role first, fallback to text content
414
+ try {
415
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
416
+ } catch (_e) {
417
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
418
+ }
419
+ }, { timeout: 5000 });
420
+ });
421
+
422
+ it('renders custom title from config', async () => {
423
+ render(
424
+ <ImportModal
425
+ {...baseProps}
426
+ config={{ title: 'Custom Import Title' }}
427
+ />
428
+ );
429
+
430
+ // Wait for dialog content to be rendered (showModal is async via requestAnimationFrame)
431
+ await waitFor(() => {
432
+ // Try by role first, fallback to text content
433
+ try {
434
+ expect(screen.getByRole('heading', { name: 'Custom Import Title' })).toBeInTheDocument();
435
+ } catch (_e) {
436
+ expect(screen.getByText('Custom Import Title')).toBeInTheDocument();
437
+ }
438
+ }, { timeout: 5000 });
439
+ });
440
+
441
+ it('renders default description', async () => {
442
+ render(<ImportModal {...baseProps} />);
443
+
444
+ // Wait for dialog to be accessible
445
+ await waitForDialog();
446
+ // Description is rendered as p in DialogHeader
447
+ expect(screen.getByText('Upload a CSV file to import multiple records at once.')).toBeInTheDocument();
448
+ });
449
+
450
+ it('renders custom description from config', async () => {
451
+ render(
452
+ <ImportModal
453
+ {...baseProps}
454
+ config={{ description: 'Custom description' }}
455
+ />
456
+ );
457
+
458
+ // Wait for dialog to be accessible
459
+ await waitForDialog();
460
+ // Description is rendered as p in DialogHeader
461
+ expect(screen.getByText('Custom description')).toBeInTheDocument();
462
+ });
463
+
464
+ it('renders file upload area', async () => {
465
+ render(<ImportModal {...baseProps} />);
466
+
467
+ // Wait for dialog title first (most reliable indicator dialog is rendered)
468
+ // Use getByRole for heading to avoid multiple matches (h2 and button both have "Import Data")
469
+ await waitFor(() => {
470
+ try {
471
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
472
+ } catch (_e) {
473
+ // Fallback: check if any element with "Import Data" exists
474
+ const elements = screen.getAllByText('Import Data');
475
+ expect(elements.length).toBeGreaterThan(0);
476
+ }
477
+ }, { timeout: 5000 });
478
+
479
+ // Then check for upload area text and button (use querySelector as fallback for buttons in dialogs)
480
+ await waitFor(() => {
481
+ expect(screen.getByText(/choose a csv file/i)).toBeInTheDocument();
482
+ const selectFileButton = findButtonByText(/select file/i);
483
+ expect(selectFileButton).toBeInTheDocument();
484
+ }, { timeout: 5000 });
485
+ });
486
+
487
+ it('renders cancel button', async () => {
488
+ render(<ImportModal {...baseProps} />);
489
+
490
+ // Wait for dialog title first (most reliable indicator dialog is rendered)
491
+ // Use getByRole for heading to avoid multiple matches (h2 and button both have "Import Data")
492
+ await waitFor(() => {
493
+ try {
494
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
495
+ } catch (_e) {
496
+ // Fallback: check if any element with "Import Data" exists
497
+ const elements = screen.getAllByText('Import Data');
498
+ expect(elements.length).toBeGreaterThan(0);
499
+ }
500
+ }, { timeout: 5000 });
501
+
502
+ // Then check for cancel button (use querySelector as fallback for buttons in dialogs)
503
+ await waitFor(() => {
504
+ const cancelButton = screen.queryByRole('button', { name: /cancel/i })
505
+ || Array.from(document.querySelectorAll('button')).find(btn => btn.textContent?.match(/cancel/i));
506
+ expect(cancelButton).toBeInTheDocument();
507
+ }, { timeout: 5000 });
508
+ });
509
+
510
+ it('renders import button', async () => {
511
+ render(<ImportModal {...baseProps} />);
512
+
513
+ // Wait for dialog title first (most reliable indicator dialog is rendered)
514
+ // Use getByRole for heading to avoid multiple matches (h2 and button both have "Import Data")
515
+ await waitFor(() => {
516
+ try {
517
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
518
+ } catch (_e) {
519
+ // Fallback: check if any element with "Import Data" exists
520
+ const elements = screen.getAllByText('Import Data');
521
+ expect(elements.length).toBeGreaterThan(0);
522
+ }
523
+ }, { timeout: 5000 });
524
+
525
+ // Then check for import button (use querySelector as fallback for buttons in dialogs)
526
+ await waitFor(() => {
527
+ const importButton = screen.queryByRole('button', { name: /import/i })
528
+ || Array.from(document.querySelectorAll('button')).find(btn => btn.textContent?.match(/^import$/i));
529
+ expect(importButton).toBeInTheDocument();
530
+ }, { timeout: 5000 });
531
+ });
532
+ });
533
+
534
+ describe('File Selection', () => {
535
+ it('displays selected file name', async () => {
536
+ const user = userEvent.setup();
537
+ const csvContent = 'name,email\nJohn,john@example.com';
538
+ const file = createCSVFile(csvContent);
539
+
540
+ render(<ImportModal {...baseProps} />);
541
+
542
+ // Wait for file input to be available
543
+ const fileInput = await waitForFileInput();
544
+ await user.upload(fileInput, file);
545
+
546
+ await waitFor(() => {
547
+ // Text is split across elements with <br/>, so use getAllByText and check first match
548
+ const elements = screen.getAllByText((content, element) => {
549
+ return element?.textContent?.includes(`Selected: ${file.name}`) ?? false;
550
+ });
551
+ expect(elements.length).toBeGreaterThan(0);
552
+ });
553
+ });
554
+
555
+ it('resets file when modal closes and reopens', async () => {
556
+ const user = userEvent.setup();
557
+ const csvContent = 'name,email\nJohn,john@example.com';
558
+ const file = createCSVFile(csvContent);
559
+
560
+ const { rerender } = render(<ImportModal {...baseProps} />);
561
+
562
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
563
+ const fileInput = await waitForFileInput(rerender, baseProps);
564
+ await user.upload(fileInput, file);
565
+
566
+ await waitFor(() => {
567
+ // Text is split across elements with <br/>, so use getAllByText and check first match
568
+ const elements = screen.getAllByText((content, element) => {
569
+ return element?.textContent?.includes(`Selected: ${file.name}`) ?? false;
570
+ });
571
+ expect(elements.length).toBeGreaterThan(0);
572
+ });
573
+
574
+ rerender(<ImportModal {...baseProps} isOpen={false} />);
575
+
576
+ // Wait for dialog to close
577
+ await waitFor(() => {
578
+ const dialog = document.querySelector('dialog[role="dialog"]');
579
+ expect(dialog).not.toBeInTheDocument();
580
+ });
581
+
582
+ rerender(<ImportModal {...baseProps} isOpen={true} />);
583
+
584
+ // Wait for dialog to reopen
585
+ await waitForDialog();
586
+
587
+ // File should be reset - use queryAllByText since text may be split
588
+ const elements = screen.queryAllByText((content, element) => {
589
+ return element?.textContent?.includes(`Selected: ${file.name}`) ?? false;
590
+ });
591
+ expect(elements.length).toBe(0);
592
+ });
593
+ });
594
+
595
+ describe('CSV Parsing', () => {
596
+ it('parses valid CSV and shows preview with data and row count', async () => {
597
+ const user = userEvent.setup();
598
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com\nBob,bob@example.com';
599
+ const file = createCSVFile(csvContent);
600
+
601
+ render(<ImportModal {...baseProps} />);
602
+
603
+ await uploadFileAndWaitForPreview(user, file);
604
+
605
+ await waitFor(() => {
606
+ expect(screen.getByText(/name/i)).toBeInTheDocument();
607
+ expect(screen.getByText(/email/i)).toBeInTheDocument();
608
+ expect(screen.getByText('John')).toBeInTheDocument();
609
+ expect(screen.getByText('john@example.com')).toBeInTheDocument();
610
+ expect(screen.getByText(/total rows to import: 3/i)).toBeInTheDocument();
611
+ }, { timeout: 5000 });
612
+ });
613
+
614
+ it('handles CSV with quoted values and commas in values', async () => {
615
+ const user = userEvent.setup();
616
+ const csvContent = 'name,description\n"John Doe","Description, with comma"\n"Jane Smith","Another, description"';
617
+ const file = createCSVFile(csvContent);
618
+
619
+ render(<ImportModal {...baseProps} />);
620
+
621
+ await waitForDialog();
622
+ await waitFor(() => {
623
+ const fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
624
+ expect(fileInput).toBeInTheDocument();
625
+ }, { timeout: 5000 });
626
+
627
+ const fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
628
+ await user.upload(fileInput, file);
629
+
630
+ await waitFor(() => {
631
+ const table = screen.queryByRole('table') ||
632
+ document.querySelector('table') ||
633
+ document.querySelector('table.min-w-full');
634
+ expect(table).toBeInTheDocument();
635
+ }, { timeout: 10000 });
636
+
637
+ await waitFor(() => {
638
+ expect(screen.getByText('John Doe')).toBeInTheDocument();
639
+ expect(screen.getByText(/description, with comma/i)).toBeInTheDocument();
640
+ }, { timeout: 2000 });
641
+ });
642
+ });
643
+
644
+ describe('Error Handling', () => {
645
+ it('displays error for invalid CSV files (empty, only header, whitespace)', async () => {
646
+ // Clear any persistent state from previous tests first
647
+ await ensureFreshModalState();
648
+
649
+ const user = userEvent.setup();
650
+ const onClose = vi.fn();
651
+
652
+ // Test empty file
653
+ const emptyFile = createCSVFile('');
654
+ const testProps = { ...baseProps, onClose };
655
+ const { rerender } = render(<ImportModal {...testProps} />);
656
+
657
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
658
+ let fileInput = await waitForFileInput(rerender, testProps);
659
+
660
+ // Ensure we have a fresh file input reference (in case it was recreated after state clear)
661
+ fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
662
+ expect(fileInput).toBeInTheDocument();
663
+ expect(fileInput.value).toBe(''); // Ensure file input is cleared
664
+
665
+ await user.upload(fileInput, emptyFile);
666
+
667
+ // When a file is selected but invalid, the error is set but file is also set
668
+ // The error should be displayed - wait for it to appear
669
+ // Note: The error might be in the CardContent section even when file is selected
670
+ await waitFor(() => {
671
+ // Check for error text or alert icon - error should be displayed somewhere
672
+ const errorText = screen.queryByText(/failed to preview file|CSV must have at least/i);
673
+ const alertIcon = screen.queryByTestId('alert-circle-icon');
674
+ if (!errorText && !alertIcon) {
675
+ throw new Error('Error message or alert icon not found after uploading invalid file');
676
+ }
677
+ }, { timeout: 3000 });
678
+
679
+ // Test CSV with only header
680
+ // Clear the file input value first to allow new file selection
681
+ fileInput.value = '';
682
+ const headerOnlyFile = createCSVFile('name,email');
683
+ await user.upload(fileInput, headerOnlyFile);
684
+
685
+ await waitFor(() => {
686
+ expect(screen.getByTestId('alert-circle-icon')).toBeInTheDocument();
687
+ }, { timeout: 3000 });
688
+
689
+ // Test CSV with only whitespace
690
+ fileInput.value = '';
691
+ const whitespaceFile = createCSVFile(' \n \n ');
692
+ await user.upload(fileInput, whitespaceFile);
693
+
694
+ await waitFor(() => {
695
+ expect(screen.getByTestId('alert-circle-icon')).toBeInTheDocument();
696
+ }, { timeout: 3000 });
697
+ });
698
+
699
+ it('clears error when new file is selected', async () => {
700
+ // Clear any persistent state from previous tests first
701
+ await ensureFreshModalState();
702
+
703
+ const user = userEvent.setup();
704
+ const onClose = vi.fn();
705
+ const invalidFile = createCSVFile('invalid');
706
+ const validFile = createCSVFile('name,email\nJohn,john@example.com');
707
+
708
+ const testProps = { ...baseProps, onClose };
709
+ const { rerender } = render(<ImportModal {...testProps} />);
710
+
711
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
712
+ let fileInput = await waitForFileInput(rerender, testProps);
713
+
714
+ // Ensure we have a fresh file input reference (in case it was recreated after state clear)
715
+ fileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
716
+ expect(fileInput).toBeInTheDocument();
717
+ expect(fileInput.value).toBe(''); // Ensure file input is cleared
718
+
719
+ await user.upload(fileInput, invalidFile);
720
+
721
+ await waitFor(() => {
722
+ expect(screen.getByTestId('alert-circle-icon')).toBeInTheDocument();
723
+ }, { timeout: 3000 });
724
+
725
+ // Clear the file input value first to allow new file selection
726
+ fileInput.value = '';
727
+
728
+ // Get a fresh file input reference (it might have been recreated)
729
+ await waitFor(() => {
730
+ const freshInput = document.querySelector('input[type="file"]') as HTMLInputElement;
731
+ expect(freshInput).toBeInTheDocument();
732
+ expect(freshInput.value).toBe('');
733
+ }, { timeout: 1000 });
734
+
735
+ const freshFileInput = document.querySelector('input[type="file"]') as HTMLInputElement;
736
+ await user.upload(freshFileInput, validFile);
737
+
738
+ // Wait for error to be cleared and preview to appear
739
+ await waitFor(() => {
740
+ expect(screen.queryByTestId('alert-circle-icon')).not.toBeInTheDocument();
741
+ // Also verify that preview appears (indicating valid file was processed)
742
+ const table = screen.queryByRole('table') || document.querySelector('table');
743
+ expect(table).toBeInTheDocument();
744
+ }, { timeout: 3000 });
745
+ });
746
+ });
747
+
748
+ describe('Import Action', () => {
749
+ it('calls onImport with parsed data when import button is clicked', async () => {
750
+ const user = userEvent.setup();
751
+ const onImport = vi.fn();
752
+ const csvContent = 'name,email\nJohn,john@example.com';
753
+ const file = createCSVFile(csvContent);
754
+
755
+ render(<ImportModal {...baseProps} onImport={onImport} />);
756
+
757
+ // Wait for file input to be available
758
+ const fileInput = await waitForFileInput();
759
+ await user.upload(fileInput, file);
760
+
761
+ // Wait for preview to appear
762
+ await waitFor(() => {
763
+ const table = screen.queryByRole('table') ||
764
+ document.querySelector('table') ||
765
+ document.querySelector('table.min-w-full');
766
+ expect(table).toBeInTheDocument();
767
+ }, { timeout: 10000 });
768
+
769
+ await waitFor(() => {
770
+ expect(screen.getByText('John')).toBeInTheDocument();
771
+ }, { timeout: 2000 });
772
+
773
+ const importButton = findButtonByText(/^import$/i);
774
+ expect(importButton).toBeInTheDocument();
775
+ if (importButton) {
776
+ await user.click(importButton);
777
+ }
778
+
779
+ await waitFor(() => {
780
+ expect(onImport).toHaveBeenCalledTimes(1);
781
+ expect(onImport).toHaveBeenCalledWith(
782
+ expect.arrayContaining([
783
+ expect.objectContaining({
784
+ name: 'John',
785
+ email: 'john@example.com',
786
+ }),
787
+ ])
788
+ );
789
+ }, { timeout: 3000 });
790
+ });
791
+
792
+ it('shows app-returned ImportSummary counts in summary section', async () => {
793
+ const user = userEvent.setup();
794
+ const onImport = vi.fn().mockResolvedValue({
795
+ successCount: 2,
796
+ totalCount: 3,
797
+ failedCount: 1,
798
+ failedRows: [{ row: 3, reason: 'Invalid meal type' }],
799
+ });
800
+ const csvContent = 'name,email\nAlice,a@x.com\nBob,b@x.com\nCarol,c@x.com';
801
+ const file = createCSVFile(csvContent);
802
+
803
+ render(<ImportModal {...baseProps} onImport={onImport} />);
804
+
805
+ const fileInput = await waitForFileInput();
806
+ await user.upload(fileInput, file);
807
+
808
+ await waitFor(() => {
809
+ const table = screen.queryByRole('table') || document.querySelector('table');
810
+ expect(table).toBeInTheDocument();
811
+ }, { timeout: 10000 });
812
+
813
+ const importButton = findButtonByText(/^import$/i);
814
+ expect(importButton).toBeInTheDocument();
815
+ if (importButton) {
816
+ await user.click(importButton);
817
+ }
818
+
819
+ await waitFor(() => {
820
+ expect(screen.getByText(/2 of 3.*imported successfully/i)).toBeInTheDocument();
821
+ }, { timeout: 5000 });
822
+ await waitFor(() => {
823
+ const failedTexts = screen.getAllByText(/1.*row.*failed to import/i);
824
+ expect(failedTexts.length).toBeGreaterThanOrEqual(1);
825
+ }, { timeout: 2000 });
826
+ });
827
+
828
+ it('disables import button when no file is selected', async () => {
829
+ const { rerender } = render(<ImportModal {...baseProps} />);
830
+
831
+ // Check if there's persistent state (file selected or summary) - if so, clear it first
832
+ await waitForDialog();
833
+ const selectedFileText = screen.queryByText(/selected:/i);
834
+ const hasSummary = screen.queryByText(/import completed/i);
835
+
836
+ if (selectedFileText || hasSummary) {
837
+ // Clear persistent state by closing and reopening
838
+ if (hasSummary) {
839
+ const closeButton = findButtonByText(/^close$/i);
840
+ if (closeButton) {
841
+ const user = userEvent.setup();
842
+ await user.click(closeButton);
843
+ await new Promise(resolve => setTimeout(resolve, 100));
844
+ }
845
+ }
846
+ // Close programmatically (onClose is just a mock, so we need to set isOpen=false)
847
+ rerender(<ImportModal {...baseProps} isOpen={false} />);
848
+ await waitFor(() => {
849
+ expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
850
+ }, { timeout: 2000 });
851
+
852
+ await new Promise(resolve => setTimeout(resolve, 500));
853
+ rerender(<ImportModal {...baseProps} isOpen={true} />);
854
+ await waitForDialog();
855
+ // Verify state was cleared
856
+ await waitFor(() => {
857
+ const stillSelected = screen.queryByText(/selected:/i);
858
+ const stillHasSummary = screen.queryByText(/import completed/i);
859
+ if (stillSelected || stillHasSummary) {
860
+ throw new Error('State was not cleared');
861
+ }
862
+ }, { timeout: 2000 });
863
+ }
864
+
865
+ // Wait for dialog title first (most reliable indicator dialog is rendered)
866
+ // Use getByRole for heading to avoid multiple matches (h2 and button both have "Import Data")
867
+ await waitFor(() => {
868
+ try {
869
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
870
+ } catch (_e) {
871
+ // Fallback: check if any element with "Import Data" exists
872
+ const elements = screen.getAllByText('Import Data');
873
+ expect(elements.length).toBeGreaterThan(0);
874
+ }
875
+ }, { timeout: 5000 });
876
+
877
+ // Then check for import button and its disabled state
878
+ await waitFor(() => {
879
+ const importButton = findButtonByText(/^import$/i);
880
+ expect(importButton).toBeInTheDocument();
881
+ expect(importButton).toBeDisabled();
882
+ }, { timeout: 5000 });
883
+ });
884
+
885
+ it('disables import button while processing', async () => {
886
+ const user = userEvent.setup();
887
+ const onImport = vi.fn(async () => {
888
+ await new Promise<void>(resolve => setTimeout(resolve, 100));
889
+ });
890
+ const csvContent = 'name,email\nJohn,john@example.com';
891
+ const file = createCSVFile(csvContent);
892
+
893
+ const testProps = { ...baseProps, onImport };
894
+ const { rerender } = render(<ImportModal {...testProps} />);
895
+
896
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
897
+ const fileInput = await waitForFileInput(rerender, testProps);
898
+ await user.upload(fileInput, file);
899
+
900
+ // Wait for preview to appear
901
+ await waitFor(() => {
902
+ const table = screen.queryByRole('table') ||
903
+ document.querySelector('table') ||
904
+ document.querySelector('table.min-w-full');
905
+ expect(table).toBeInTheDocument();
906
+ }, { timeout: 10000 });
907
+
908
+ await waitFor(() => {
909
+ expect(screen.getByText('John')).toBeInTheDocument();
910
+ }, { timeout: 2000 });
911
+
912
+ const importButton = findButtonByText(/^import$/i);
913
+ expect(importButton).toBeInTheDocument();
914
+ if (importButton) {
915
+ await user.click(importButton);
916
+ }
917
+
918
+ await waitFor(() => {
919
+ expect(importButton).toBeDisabled();
920
+ }, { timeout: 1000 });
921
+ });
922
+
923
+ it('shows processing text while importing', async () => {
924
+ const user = userEvent.setup();
925
+ const onImport = vi.fn(async () => {
926
+ await new Promise<void>(resolve => setTimeout(resolve, 100));
927
+ });
928
+ const csvContent = 'name,email\nJohn,john@example.com';
929
+ const file = createCSVFile(csvContent);
930
+
931
+ const testProps = { ...baseProps, onImport };
932
+ const { rerender } = render(<ImportModal {...testProps} />);
933
+
934
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
935
+ const fileInput = await waitForFileInput(rerender, testProps);
936
+ await user.upload(fileInput, file);
937
+
938
+ // Wait for preview to appear
939
+ await waitFor(() => {
940
+ const table = screen.queryByRole('table') ||
941
+ document.querySelector('table') ||
942
+ document.querySelector('table.min-w-full');
943
+ expect(table).toBeInTheDocument();
944
+ }, { timeout: 10000 });
945
+
946
+ await waitFor(() => {
947
+ expect(screen.getByText('John')).toBeInTheDocument();
948
+ }, { timeout: 2000 });
949
+
950
+ const importButton = findButtonByText(/^import$/i);
951
+ expect(importButton).toBeInTheDocument();
952
+ if (importButton) {
953
+ await user.click(importButton);
954
+ }
955
+
956
+ // Button text changes to "Processing..." when isProcessing is true
957
+ await waitFor(() => {
958
+ const processingButton = findButtonByText(/processing/i);
959
+ expect(processingButton).toBeInTheDocument();
960
+ }, { timeout: 1000 });
961
+ });
962
+
963
+ it('calls onClose after successful import', async () => {
964
+ const user = userEvent.setup();
965
+ const onClose = vi.fn();
966
+ const csvContent = 'name,email\nJohn,john@example.com';
967
+ const file = createCSVFile(csvContent);
968
+ const onImport = vi.fn().mockResolvedValue({ successCount: 1, totalCount: 1, failedCount: 0 });
969
+
970
+ const testProps = { ...baseProps, onClose, onImport };
971
+ const { rerender } = render(<ImportModal {...testProps} />);
972
+
973
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
974
+ const fileInput = await waitForFileInput(rerender, testProps);
975
+ await user.upload(fileInput, file);
976
+
977
+ // Wait for preview to appear
978
+ await waitFor(() => {
979
+ const table = screen.queryByRole('table') ||
980
+ document.querySelector('table') ||
981
+ document.querySelector('table.min-w-full');
982
+ expect(table).toBeInTheDocument();
983
+ }, { timeout: 10000 });
984
+
985
+ await waitFor(() => {
986
+ expect(screen.getByText('John')).toBeInTheDocument();
987
+ }, { timeout: 2000 });
988
+
989
+ const importButton = findButtonByText(/^import$/i);
990
+ expect(importButton).toBeInTheDocument();
991
+ if (importButton) {
992
+ await user.click(importButton);
993
+ }
994
+
995
+ // Wait for import to complete and summary to appear
996
+ await waitFor(() => {
997
+ expect(screen.getByText(/import completed successfully/i)).toBeInTheDocument();
998
+ }, { timeout: 5000 });
999
+
1000
+ // Click the Close button to trigger onClose
1001
+ const closeButton = findButtonByText(/^close$/i);
1002
+ expect(closeButton).toBeInTheDocument();
1003
+ if (closeButton) {
1004
+ await user.click(closeButton);
1005
+ }
1006
+
1007
+ // Verify onClose was called
1008
+ expect(onClose).toHaveBeenCalled();
1009
+ });
1010
+ });
1011
+
1012
+ describe('Close Action', () => {
1013
+ it('calls onClose when cancel button is clicked', async () => {
1014
+ const user = userEvent.setup();
1015
+ const onClose = vi.fn();
1016
+ const { rerender } = render(<ImportModal {...baseProps} onClose={onClose} />);
1017
+
1018
+ // Wait for dialog title first (most reliable indicator dialog is rendered)
1019
+ await waitFor(() => {
1020
+ try {
1021
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
1022
+ } catch (_e) {
1023
+ const elements = screen.getAllByText('Import Data');
1024
+ expect(elements.length).toBeGreaterThan(0);
1025
+ }
1026
+ }, { timeout: 5000 });
1027
+
1028
+ // Check if there's persistent state (summary or selected file) - if so, clear it first
1029
+ const hasSummary = screen.queryByText(/import completed/i);
1030
+ const hasSelectedFile = screen.queryByText(/selected:/i);
1031
+ if (hasSummary || hasSelectedFile) {
1032
+ // If there's a summary, click close button to trigger handleClose
1033
+ if (hasSummary) {
1034
+ const closeButton = findButtonByText(/^close$/i);
1035
+ if (closeButton) {
1036
+ await user.click(closeButton);
1037
+ await new Promise(resolve => setTimeout(resolve, 100));
1038
+ }
1039
+ }
1040
+ // Close and reopen to clear state
1041
+ rerender(<ImportModal {...baseProps} isOpen={false} onClose={onClose} />);
1042
+ await waitFor(() => {
1043
+ expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
1044
+ }, { timeout: 2000 });
1045
+ await new Promise(resolve => setTimeout(resolve, 500));
1046
+ // Reset the mock call count since we called onClose when clearing state
1047
+ onClose.mockClear();
1048
+ rerender(<ImportModal {...baseProps} isOpen={true} onClose={onClose} />);
1049
+ await waitForDialog();
1050
+ await waitFor(() => {
1051
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
1052
+ }, { timeout: 5000 });
1053
+ // Verify state was cleared
1054
+ await waitFor(() => {
1055
+ const stillHasSummary = screen.queryByText(/import completed/i);
1056
+ const stillHasSelected = screen.queryByText(/selected:/i);
1057
+ if (stillHasSummary || stillHasSelected) {
1058
+ throw new Error('State was not cleared');
1059
+ }
1060
+ }, { timeout: 2000 });
1061
+ }
1062
+
1063
+ // Then wait for cancel button (should exist when no summary)
1064
+ await waitFor(() => {
1065
+ const cancelButton = findButtonByText(/cancel/i);
1066
+ expect(cancelButton).toBeInTheDocument();
1067
+ }, { timeout: 5000 });
1068
+
1069
+ const cancelButton = findButtonByText(/cancel/i);
1070
+ expect(cancelButton).toBeInTheDocument();
1071
+ if (cancelButton) {
1072
+ await user.click(cancelButton);
1073
+ }
1074
+
1075
+ expect(onClose).toHaveBeenCalledTimes(1);
1076
+ });
1077
+
1078
+ it('resets state when modal closes', async () => {
1079
+ const user = userEvent.setup();
1080
+ const csvContent = 'name,email\nJohn,john@example.com';
1081
+ const file = createCSVFile(csvContent);
1082
+
1083
+ const { rerender } = render(<ImportModal {...baseProps} />);
1084
+
1085
+ // Wait for file input to be available (pass rerender to handle persistent state)
1086
+ const fileInput = await waitForFileInput(rerender, baseProps);
1087
+ await user.upload(fileInput, file);
1088
+
1089
+ // Wait for preview to appear
1090
+ await waitFor(() => {
1091
+ const table = screen.queryByRole('table') ||
1092
+ document.querySelector('table') ||
1093
+ document.querySelector('table.min-w-full');
1094
+ expect(table).toBeInTheDocument();
1095
+ }, { timeout: 10000 });
1096
+
1097
+ await waitFor(() => {
1098
+ expect(screen.getByText('John')).toBeInTheDocument();
1099
+ }, { timeout: 2000 });
1100
+
1101
+ rerender(<ImportModal {...baseProps} isOpen={false} />);
1102
+ rerender(<ImportModal {...baseProps} isOpen={true} />);
1103
+
1104
+ // State should be reset - wait for modal to reopen
1105
+ await waitFor(() => {
1106
+ expect(screen.queryByText('John')).not.toBeInTheDocument();
1107
+ }, { timeout: 1000 });
1108
+ });
1109
+ });
1110
+
1111
+ describe('Custom Configuration', () => {
1112
+ it('uses custom button texts from config', async () => {
1113
+ const testProps = {
1114
+ ...baseProps,
1115
+ config: {
1116
+ selectFileButtonText: 'Browse Files',
1117
+ importButtonText: 'Import Data',
1118
+ cancelButtonText: 'Close',
1119
+ }
1120
+ };
1121
+ const { rerender } = render(<ImportModal {...testProps} />);
1122
+
1123
+ // Wait for dialog title first
1124
+ await waitFor(() => {
1125
+ try {
1126
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
1127
+ } catch (_e) {
1128
+ const elements = screen.getAllByText('Import Data');
1129
+ expect(elements.length).toBeGreaterThan(0);
1130
+ }
1131
+ }, { timeout: 5000 });
1132
+
1133
+ // Check if there's a summary - if so, clear it first
1134
+ const hasSummary = screen.queryByText(/import completed/i);
1135
+ if (hasSummary) {
1136
+ rerender(
1137
+ <ImportModal
1138
+ {...baseProps}
1139
+ isOpen={false}
1140
+ config={{
1141
+ selectFileButtonText: 'Browse Files',
1142
+ importButtonText: 'Import Data',
1143
+ cancelButtonText: 'Close',
1144
+ }}
1145
+ />
1146
+ );
1147
+ await waitFor(() => {
1148
+ expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
1149
+ }, { timeout: 2000 });
1150
+ await new Promise(resolve => setTimeout(resolve, 300));
1151
+ rerender(
1152
+ <ImportModal
1153
+ {...baseProps}
1154
+ isOpen={true}
1155
+ config={{
1156
+ selectFileButtonText: 'Browse Files',
1157
+ importButtonText: 'Import Data',
1158
+ cancelButtonText: 'Close',
1159
+ }}
1160
+ />
1161
+ );
1162
+ await waitForDialog();
1163
+ await waitFor(() => {
1164
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
1165
+ }, { timeout: 5000 });
1166
+ }
1167
+
1168
+ // Then wait for buttons (should exist when no summary)
1169
+ await waitFor(() => {
1170
+ const browseButton = findButtonByText(/browse files/i);
1171
+ expect(browseButton).toBeInTheDocument();
1172
+ }, { timeout: 5000 });
1173
+
1174
+ await waitFor(() => {
1175
+ const importDataButton = findButtonByText(/import data/i);
1176
+ expect(importDataButton).toBeInTheDocument();
1177
+ }, { timeout: 5000 });
1178
+ // Dialog has a close button too, so find the Cancel one by text
1179
+ const cancelButton = findButtonByText(/^close$/i);
1180
+ expect(cancelButton).toBeInTheDocument();
1181
+ });
1182
+
1183
+ it('uses custom preview header text from config', async () => {
1184
+ const user = userEvent.setup();
1185
+ const csvContent = 'name,email\nJohn,john@example.com';
1186
+ const file = createCSVFile(csvContent);
1187
+
1188
+ const testProps = { ...baseProps, config: { previewHeaderText: 'Data Preview' } };
1189
+ const { rerender } = render(<ImportModal {...testProps} />);
1190
+
1191
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
1192
+ const fileInput = await waitForFileInput(rerender, testProps);
1193
+ await user.upload(fileInput, file);
1194
+
1195
+ // Wait for preview to appear
1196
+ await waitFor(() => {
1197
+ expect(screen.getByText('Data Preview')).toBeInTheDocument();
1198
+ }, { timeout: 5000 });
1199
+ });
1200
+
1201
+ it('uses custom total rows text from config', async () => {
1202
+ const user = userEvent.setup();
1203
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1204
+ const file = createCSVFile(csvContent);
1205
+
1206
+ const testProps = { ...baseProps, config: { totalRowsText: 'Found {count} records' } };
1207
+ const { rerender } = render(<ImportModal {...testProps} />);
1208
+
1209
+ // Wait for file input to be available (pass rerender and props to handle persistent state)
1210
+ const fileInput = await waitForFileInput(rerender, testProps);
1211
+ await user.upload(fileInput, file);
1212
+
1213
+ // Wait for preview to appear
1214
+ await waitFor(() => {
1215
+ const table = screen.queryByRole('table') ||
1216
+ document.querySelector('table') ||
1217
+ document.querySelector('table.min-w-full');
1218
+ expect(table).toBeInTheDocument();
1219
+ }, { timeout: 10000 });
1220
+
1221
+ await waitFor(() => {
1222
+ expect(screen.getByText(/found 2 records/i)).toBeInTheDocument();
1223
+ }, { timeout: 2000 });
1224
+ });
1225
+ });
1226
+
1227
+ describe('Import Progress Tracking', () => {
1228
+ it('shows parsing progress during CSV parsing', async () => {
1229
+ const user = userEvent.setup();
1230
+ const headers = 'name,email\n';
1231
+ const rows = Array.from({ length: 200 }, (_, i) => `User${i},user${i}@example.com`).join('\n');
1232
+ const csvContent = headers + rows;
1233
+ const file = createCSVFile(csvContent);
1234
+ const onImport = vi.fn(async () => {
1235
+ await new Promise(resolve => setTimeout(resolve, 100));
1236
+ });
1237
+
1238
+ const testProps = { ...baseProps, onImport };
1239
+ const { rerender } = render(<ImportModal {...testProps} />);
1240
+
1241
+ const fileInput = await waitForFileInput(rerender, testProps);
1242
+ await user.upload(fileInput, file);
1243
+
1244
+ await waitFor(() => {
1245
+ const table = screen.queryByRole('table') || document.querySelector('table');
1246
+ expect(table).toBeInTheDocument();
1247
+ }, { timeout: 10000 });
1248
+
1249
+ const importButton = findButtonByText(/^import$/i);
1250
+ expect(importButton).toBeInTheDocument();
1251
+ if (importButton && !importButton.disabled) {
1252
+ await user.click(importButton);
1253
+ }
1254
+
1255
+ // Should show parsing progress (may be very brief)
1256
+ await waitFor(() => {
1257
+ const parsingText = screen.queryByText(/parsing csv file|importing data/i);
1258
+ if (parsingText) {
1259
+ expect(parsingText).toBeInTheDocument();
1260
+ }
1261
+ }, { timeout: 5000 });
1262
+ });
1263
+
1264
+ it('shows importing progress during data import', async () => {
1265
+ const user = userEvent.setup();
1266
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1267
+ const file = createCSVFile(csvContent);
1268
+ const onImport = vi.fn(async () => {
1269
+ await new Promise(resolve => setTimeout(resolve, 100));
1270
+ });
1271
+
1272
+ const testProps = { ...baseProps, onImport };
1273
+ const { rerender } = render(<ImportModal {...testProps} />);
1274
+
1275
+ const fileInput = await waitForFileInput(rerender, testProps);
1276
+ await user.upload(fileInput, file);
1277
+
1278
+ await waitFor(() => {
1279
+ const table = screen.queryByRole('table') || document.querySelector('table');
1280
+ expect(table).toBeInTheDocument();
1281
+ }, { timeout: 10000 });
1282
+
1283
+ const importButton = findButtonByText(/^import$/i);
1284
+ if (importButton) {
1285
+ await user.click(importButton);
1286
+ }
1287
+
1288
+ // Should show importing progress
1289
+ await waitFor(() => {
1290
+ expect(screen.getByText(/importing data/i)).toBeInTheDocument();
1291
+ }, { timeout: 2000 });
1292
+ });
1293
+
1294
+ it('shows chunk progress when processing large datasets', async () => {
1295
+ const user = userEvent.setup();
1296
+ const headers = 'name,email\n';
1297
+ const rows = Array.from({ length: 250 }, (_, i) => `User${i},user${i}@example.com`).join('\n');
1298
+ const csvContent = headers + rows;
1299
+ const file = createCSVFile(csvContent);
1300
+ const onImport = vi.fn(async () => {
1301
+ await new Promise(resolve => setTimeout(resolve, 50));
1302
+ });
1303
+
1304
+ const testProps = { ...baseProps, onImport };
1305
+ const { rerender } = render(<ImportModal {...testProps} />);
1306
+
1307
+ const fileInput = await waitForFileInput(rerender, testProps);
1308
+ await user.upload(fileInput, file);
1309
+
1310
+ await waitFor(() => {
1311
+ const table = screen.queryByRole('table') || document.querySelector('table');
1312
+ expect(table).toBeInTheDocument();
1313
+ }, { timeout: 10000 });
1314
+
1315
+ const importButton = findButtonByText(/^import$/i);
1316
+ if (importButton) {
1317
+ await user.click(importButton);
1318
+ }
1319
+
1320
+ // Should show chunk progress
1321
+ await waitFor(() => {
1322
+ const progressText = screen.queryByText(/chunk \d+ of \d+/i);
1323
+ if (progressText) {
1324
+ expect(progressText).toBeInTheDocument();
1325
+ }
1326
+ }, { timeout: 3000 });
1327
+ });
1328
+ });
1329
+
1330
+ describe('Import Summary', () => {
1331
+ it('displays success summary when all rows imported successfully', async () => {
1332
+ const user = userEvent.setup();
1333
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1334
+ const file = createCSVFile(csvContent);
1335
+ const onImport = vi.fn().mockResolvedValue({ successCount: 2, totalCount: 2, failedCount: 0 });
1336
+
1337
+ const testProps = { ...baseProps, onImport };
1338
+ const { rerender } = render(<ImportModal {...testProps} />);
1339
+
1340
+ const fileInput = await waitForFileInput(rerender, testProps);
1341
+ await user.upload(fileInput, file);
1342
+
1343
+ await waitFor(() => {
1344
+ const table = screen.queryByRole('table') || document.querySelector('table');
1345
+ expect(table).toBeInTheDocument();
1346
+ }, { timeout: 10000 });
1347
+
1348
+ const importButton = findButtonByText(/^import$/i);
1349
+ if (importButton) {
1350
+ await user.click(importButton);
1351
+ }
1352
+
1353
+ await waitFor(() => {
1354
+ expect(screen.getByText(/import completed successfully/i)).toBeInTheDocument();
1355
+ expect(screen.getByText(/2 of 2.*imported/i)).toBeInTheDocument();
1356
+ }, { timeout: 5000 });
1357
+ });
1358
+
1359
+ it('displays error summary when some rows fail', async () => {
1360
+ const user = userEvent.setup();
1361
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1362
+ const file = createCSVFile(csvContent);
1363
+ let callCount = 0;
1364
+ const onImport = vi.fn(async () => {
1365
+ callCount++;
1366
+ if (callCount === 1) {
1367
+ throw new Error('Import failed for chunk');
1368
+ }
1369
+ });
1370
+
1371
+ const testProps = { ...baseProps, onImport };
1372
+ const { rerender } = render(<ImportModal {...testProps} />);
1373
+
1374
+ const fileInput = await waitForFileInput(rerender, testProps);
1375
+ await user.upload(fileInput, file);
1376
+
1377
+ await waitFor(() => {
1378
+ const table = screen.queryByRole('table') || document.querySelector('table');
1379
+ expect(table).toBeInTheDocument();
1380
+ }, { timeout: 10000 });
1381
+
1382
+ const importButton = findButtonByText(/^import$/i);
1383
+ if (importButton) {
1384
+ await user.click(importButton);
1385
+ }
1386
+
1387
+ await waitFor(() => {
1388
+ expect(screen.getByText(/import completed with errors/i)).toBeInTheDocument();
1389
+ }, { timeout: 5000 });
1390
+ });
1391
+
1392
+ it('displays failed rows table when failures occur', async () => {
1393
+ const user = userEvent.setup();
1394
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1395
+ const file = createCSVFile(csvContent);
1396
+ const onImport = vi.fn().mockRejectedValue(new Error('Import failed'));
1397
+
1398
+ const testProps = { ...baseProps, onImport };
1399
+ const { rerender } = render(<ImportModal {...testProps} />);
1400
+
1401
+ const fileInput = await waitForFileInput(rerender, testProps);
1402
+ await user.upload(fileInput, file);
1403
+
1404
+ await waitFor(() => {
1405
+ const table = screen.queryByRole('table') || document.querySelector('table');
1406
+ expect(table).toBeInTheDocument();
1407
+ }, { timeout: 10000 });
1408
+
1409
+ const importButton = findButtonByText(/^import$/i);
1410
+ if (importButton) {
1411
+ await user.click(importButton);
1412
+ }
1413
+
1414
+ await waitFor(() => {
1415
+ const failedRowsTable = screen.queryByText(/failed rows/i);
1416
+ if (failedRowsTable) {
1417
+ expect(failedRowsTable).toBeInTheDocument();
1418
+ }
1419
+ }, { timeout: 5000 });
1420
+ });
1421
+
1422
+ it('limits failed rows display to 50 rows', async () => {
1423
+ const user = userEvent.setup();
1424
+ const headers = 'name,email\n';
1425
+ const rows = Array.from({ length: 100 }, (_, i) => `User${i},user${i}@example.com`).join('\n');
1426
+ const csvContent = headers + rows;
1427
+ const file = createCSVFile(csvContent);
1428
+ const onImport = vi.fn().mockRejectedValue(new Error('Import failed'));
1429
+
1430
+ const testProps = { ...baseProps, onImport };
1431
+ const { rerender } = render(<ImportModal {...testProps} />);
1432
+
1433
+ const fileInput = await waitForFileInput(rerender, testProps);
1434
+ await user.upload(fileInput, file);
1435
+
1436
+ await waitFor(() => {
1437
+ const table = screen.queryByRole('table') || document.querySelector('table');
1438
+ expect(table).toBeInTheDocument();
1439
+ }, { timeout: 10000 });
1440
+
1441
+ const importButton = findButtonByText(/^import$/i);
1442
+ if (importButton) {
1443
+ await user.click(importButton);
1444
+ }
1445
+
1446
+ await waitFor(() => {
1447
+ const failedRowsText = screen.queryByText(/showing first \d+ of \d+ failed/i);
1448
+ if (failedRowsText) {
1449
+ expect(failedRowsText).toBeInTheDocument();
1450
+ }
1451
+ }, { timeout: 5000 });
1452
+ });
1453
+ });
1454
+
1455
+ describe('Chunk Processing', () => {
1456
+ it('processes data in chunks for large datasets', async () => {
1457
+ const user = userEvent.setup();
1458
+ const headers = 'name,email\n';
1459
+ const rows = Array.from({ length: 150 }, (_, i) => `User${i},user${i}@example.com`).join('\n');
1460
+ const csvContent = headers + rows;
1461
+ const file = createCSVFile(csvContent);
1462
+ const onImport = vi.fn().mockResolvedValue(undefined);
1463
+
1464
+ const testProps = { ...baseProps, onImport };
1465
+ const { rerender } = render(<ImportModal {...testProps} />);
1466
+
1467
+ const fileInput = await waitForFileInput(rerender, testProps);
1468
+ await user.upload(fileInput, file);
1469
+
1470
+ await waitFor(() => {
1471
+ const table = screen.queryByRole('table') || document.querySelector('table');
1472
+ expect(table).toBeInTheDocument();
1473
+ }, { timeout: 10000 });
1474
+
1475
+ const importButton = findButtonByText(/^import$/i);
1476
+ if (importButton) {
1477
+ await user.click(importButton);
1478
+ }
1479
+
1480
+ await waitFor(() => {
1481
+ // Should be called multiple times for chunks
1482
+ expect(onImport).toHaveBeenCalled();
1483
+ }, { timeout: 5000 });
1484
+ });
1485
+
1486
+ it('handles chunk processing errors gracefully', async () => {
1487
+ const user = userEvent.setup();
1488
+ const headers = 'name,email\n';
1489
+ const rows = Array.from({ length: 150 }, (_, i) => `User${i},user${i}@example.com`).join('\n');
1490
+ const csvContent = headers + rows;
1491
+ const file = createCSVFile(csvContent);
1492
+ let callCount = 0;
1493
+ const onImport = vi.fn(async () => {
1494
+ callCount++;
1495
+ if (callCount === 2) {
1496
+ throw new Error('Chunk processing failed');
1497
+ }
1498
+ });
1499
+
1500
+ const testProps = { ...baseProps, onImport };
1501
+ const { rerender } = render(<ImportModal {...testProps} />);
1502
+
1503
+ const fileInput = await waitForFileInput(rerender, testProps);
1504
+ await user.upload(fileInput, file);
1505
+
1506
+ await waitFor(() => {
1507
+ const table = screen.queryByRole('table') || document.querySelector('table');
1508
+ expect(table).toBeInTheDocument();
1509
+ }, { timeout: 10000 });
1510
+
1511
+ const importButton = findButtonByText(/^import$/i);
1512
+ if (importButton) {
1513
+ await user.click(importButton);
1514
+ }
1515
+
1516
+ await waitFor(() => {
1517
+ // Should show summary with failures
1518
+ const summary = screen.queryByText(/import completed/i);
1519
+ expect(summary).toBeInTheDocument();
1520
+ }, { timeout: 5000 });
1521
+ });
1522
+ });
1523
+
1524
+ describe('Edge Cases', () => {
1525
+ it('handles very large CSV files', async () => {
1526
+ const user = userEvent.setup();
1527
+ const headers = 'name,email\n';
1528
+ const rows = Array.from({ length: 1000 }, (_, i) => `User${i},user${i}@example.com`).join('\n');
1529
+ const csvContent = headers + rows;
1530
+ const file = createCSVFile(csvContent);
1531
+
1532
+ const { rerender } = render(<ImportModal {...baseProps} />);
1533
+
1534
+ // Wait for file input to be available (pass rerender to handle persistent state)
1535
+ const fileInput = await waitForFileInput(rerender, baseProps);
1536
+ await user.upload(fileInput, file);
1537
+
1538
+ await waitFor(() => {
1539
+ expect(screen.getByText(/total rows to import: 1000/i)).toBeInTheDocument();
1540
+ }, { timeout: 5000 });
1541
+ });
1542
+
1543
+ it('handles CSV with only one data row', async () => {
1544
+ const user = userEvent.setup();
1545
+ const csvContent = 'name,email\nJohn,john@example.com';
1546
+ const file = createCSVFile(csvContent);
1547
+
1548
+ const { rerender } = render(<ImportModal {...baseProps} />);
1549
+
1550
+ const fileInput = await waitForFileInput(rerender, baseProps);
1551
+ await user.upload(fileInput, file);
1552
+
1553
+ await waitFor(() => {
1554
+ expect(screen.getByText(/total rows to import: 1/i)).toBeInTheDocument();
1555
+ }, { timeout: 5000 });
1556
+ });
1557
+
1558
+ it('handles CSV with empty values', async () => {
1559
+ const user = userEvent.setup();
1560
+ const csvContent = 'name,email\nJohn,\n,jane@example.com';
1561
+ const file = createCSVFile(csvContent);
1562
+
1563
+ const { rerender } = render(<ImportModal {...baseProps} />);
1564
+
1565
+ const fileInput = await waitForFileInput(rerender, baseProps);
1566
+ await user.upload(fileInput, file);
1567
+
1568
+ await waitFor(() => {
1569
+ const table = screen.queryByRole('table') || document.querySelector('table');
1570
+ expect(table).toBeInTheDocument();
1571
+ }, { timeout: 10000 });
1572
+ });
1573
+
1574
+ it('handles CSV with special characters in values', async () => {
1575
+ const user = userEvent.setup();
1576
+ const csvContent = 'name,email\n"John, Doe","john@example.com"\n"Jane; Smith","jane@example.com"';
1577
+ const file = createCSVFile(csvContent);
1578
+
1579
+ const { rerender } = render(<ImportModal {...baseProps} />);
1580
+
1581
+ const fileInput = await waitForFileInput(rerender, baseProps);
1582
+ await user.upload(fileInput, file);
1583
+
1584
+ await waitFor(() => {
1585
+ const table = screen.queryByRole('table') || document.querySelector('table');
1586
+ expect(table).toBeInTheDocument();
1587
+ }, { timeout: 10000 });
1588
+ });
1589
+
1590
+ it('handles async onImport that returns Promise', async () => {
1591
+ const user = userEvent.setup();
1592
+ const csvContent = 'name,email\nJohn,john@example.com';
1593
+ const file = createCSVFile(csvContent);
1594
+ const onImport = vi.fn(() => Promise.resolve());
1595
+
1596
+ const testProps = { ...baseProps, onImport };
1597
+ const { rerender } = render(<ImportModal {...testProps} />);
1598
+
1599
+ const fileInput = await waitForFileInput(rerender, testProps);
1600
+ await user.upload(fileInput, file);
1601
+
1602
+ await waitFor(() => {
1603
+ const table = screen.queryByRole('table') || document.querySelector('table');
1604
+ expect(table).toBeInTheDocument();
1605
+ }, { timeout: 10000 });
1606
+
1607
+ const importButton = findButtonByText(/^import$/i);
1608
+ if (importButton) {
1609
+ await user.click(importButton);
1610
+ }
1611
+
1612
+ await waitFor(() => {
1613
+ expect(onImport).toHaveBeenCalled();
1614
+ }, { timeout: 5000 });
1615
+ });
1616
+
1617
+ it('prevents closing modal during processing', async () => {
1618
+ const user = userEvent.setup();
1619
+ const csvContent = 'name,email\nJohn,john@example.com';
1620
+ const file = createCSVFile(csvContent);
1621
+ const onImport = vi.fn(async () => {
1622
+ await new Promise(resolve => setTimeout(resolve, 200));
1623
+ });
1624
+ const onClose = vi.fn();
1625
+
1626
+ const testProps = { ...baseProps, onImport, onClose };
1627
+ const { rerender } = render(<ImportModal {...testProps} />);
1628
+
1629
+ const fileInput = await waitForFileInput(rerender, testProps);
1630
+ await user.upload(fileInput, file);
1631
+
1632
+ await waitFor(() => {
1633
+ const table = screen.queryByRole('table') || document.querySelector('table');
1634
+ expect(table).toBeInTheDocument();
1635
+ }, { timeout: 10000 });
1636
+
1637
+ const importButton = findButtonByText(/^import$/i);
1638
+ if (importButton) {
1639
+ await user.click(importButton);
1640
+ }
1641
+
1642
+ // Try to close modal during processing
1643
+ await waitFor(() => {
1644
+ const dialog = document.querySelector('dialog[role="dialog"]');
1645
+ if (dialog) {
1646
+ // Simulate escape key or outside click
1647
+ const event = new KeyboardEvent('keydown', { key: 'Escape' });
1648
+ dialog.dispatchEvent(event);
1649
+ }
1650
+ }, { timeout: 1000 });
1651
+
1652
+ // Modal should still be open
1653
+ await waitFor(() => {
1654
+ expect(screen.queryByText('Import Data')).toBeInTheDocument();
1655
+ }, { timeout: 1000 });
1656
+ });
1657
+
1658
+ it('prevents closing modal when summary exists', async () => {
1659
+ const user = userEvent.setup();
1660
+ const csvContent = 'name,email\nJohn,john@example.com';
1661
+ const file = createCSVFile(csvContent);
1662
+ const onImport = vi.fn().mockResolvedValue(undefined);
1663
+ const onClose = vi.fn();
1664
+
1665
+ const testProps = { ...baseProps, onImport, onClose };
1666
+ const { rerender } = render(<ImportModal {...testProps} />);
1667
+
1668
+ const fileInput = await waitForFileInput(rerender, testProps);
1669
+ await user.upload(fileInput, file);
1670
+
1671
+ await waitFor(() => {
1672
+ const table = screen.queryByRole('table') || document.querySelector('table');
1673
+ expect(table).toBeInTheDocument();
1674
+ }, { timeout: 10000 });
1675
+
1676
+ const importButton = findButtonByText(/^import$/i);
1677
+ if (importButton) {
1678
+ await user.click(importButton);
1679
+ }
1680
+
1681
+ // Wait for summary to appear
1682
+ await waitFor(() => {
1683
+ expect(screen.getByText(/import completed/i)).toBeInTheDocument();
1684
+ }, { timeout: 5000 });
1685
+
1686
+ // Try to close modal - should be prevented
1687
+ const dialog = document.querySelector('dialog[role="dialog"]');
1688
+ if (dialog) {
1689
+ const event = new KeyboardEvent('keydown', { key: 'Escape' });
1690
+ dialog.dispatchEvent(event);
1691
+ }
1692
+
1693
+ // Modal should still be open
1694
+ await waitFor(() => {
1695
+ expect(screen.queryByText('Import Data')).toBeInTheDocument();
1696
+ }, { timeout: 1000 });
1697
+ });
1698
+ });
1699
+
1700
+ describe('Full import flow and section content', () => {
1701
+ it('full flow renders and asserts File, Preview, Summary section content', async () => {
1702
+ const user = userEvent.setup();
1703
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1704
+ const file = createCSVFile(csvContent);
1705
+ const onImport = vi.fn().mockResolvedValue({ successCount: 2, totalCount: 2, failedCount: 0 });
1706
+
1707
+ const testProps = { ...baseProps, onImport };
1708
+ const { rerender } = render(<ImportModal {...testProps} />);
1709
+
1710
+ const fileInput = await waitForFileInput(rerender, testProps);
1711
+ await user.upload(fileInput, file);
1712
+
1713
+ await waitFor(() => {
1714
+ expect(screen.getByText(/selected:/i)).toBeInTheDocument();
1715
+ }, { timeout: 3000 });
1716
+ const fileSection = document.body;
1717
+ expect(fileSection.textContent).toMatch(new RegExp(file.name, 'i'));
1718
+
1719
+ await waitFor(() => {
1720
+ const table = screen.queryByRole('table') || document.querySelector('table');
1721
+ expect(table).toBeInTheDocument();
1722
+ }, { timeout: 10000 });
1723
+ expect(screen.getByText(/total rows to import: 2/i)).toBeInTheDocument();
1724
+ expect(screen.getByText('John')).toBeInTheDocument();
1725
+ expect(screen.getByText('Jane')).toBeInTheDocument();
1726
+
1727
+ const importButton = findButtonByText(/^import$/i);
1728
+ if (importButton) {
1729
+ await user.click(importButton);
1730
+ }
1731
+
1732
+ await waitFor(() => {
1733
+ expect(screen.getByText(/import completed successfully/i)).toBeInTheDocument();
1734
+ expect(screen.getByText(/2 of 2.*imported/i)).toBeInTheDocument();
1735
+ }, { timeout: 5000 });
1736
+ });
1737
+
1738
+ it('full flow with import error renders Summary and FailedRows section content', async () => {
1739
+ const user = userEvent.setup();
1740
+ const csvContent = 'name,email\nJohn,john@example.com\nJane,jane@example.com';
1741
+ const file = createCSVFile(csvContent);
1742
+ const onImport = vi.fn().mockRejectedValue(new Error('Import failed'));
1743
+
1744
+ const testProps = { ...baseProps, onImport };
1745
+ const { rerender } = render(<ImportModal {...testProps} />);
1746
+
1747
+ const fileInput = await waitForFileInput(rerender, testProps);
1748
+ await user.upload(fileInput, file);
1749
+
1750
+ await waitFor(() => {
1751
+ const table = screen.queryByRole('table') || document.querySelector('table');
1752
+ expect(table).toBeInTheDocument();
1753
+ }, { timeout: 10000 });
1754
+
1755
+ const importButton = findButtonByText(/^import$/i);
1756
+ if (importButton) {
1757
+ await user.click(importButton);
1758
+ }
1759
+
1760
+ await waitFor(() => {
1761
+ expect(screen.getByText(/import completed with errors/i)).toBeInTheDocument();
1762
+ }, { timeout: 5000 });
1763
+ const failedRowsHeading = screen.queryByText(/failed rows/i);
1764
+ if (failedRowsHeading) {
1765
+ expect(failedRowsHeading).toBeInTheDocument();
1766
+ }
1767
+ });
1768
+ });
1769
+
1770
+ describe('Accessibility', () => {
1771
+ it('provides accessible file input', async () => {
1772
+ const { rerender } = render(<ImportModal {...baseProps} />);
1773
+
1774
+ // Wait for file input to be available (pass rerender to handle persistent state)
1775
+ const fileInput = await waitForFileInput(rerender, baseProps);
1776
+ expect(fileInput).toBeInTheDocument();
1777
+ expect(fileInput).toHaveAttribute('type', 'file');
1778
+ expect(fileInput).toHaveAttribute('accept', '.csv');
1779
+ });
1780
+
1781
+ it('provides accessible button labels', async () => {
1782
+ const { rerender } = render(<ImportModal {...baseProps} />);
1783
+
1784
+ // Wait for dialog title first
1785
+ await waitFor(() => {
1786
+ try {
1787
+ expect(screen.getByRole('heading', { name: 'Import Data' })).toBeInTheDocument();
1788
+ } catch (_e) {
1789
+ const elements = screen.getAllByText('Import Data');
1790
+ expect(elements.length).toBeGreaterThan(0);
1791
+ }
1792
+ }, { timeout: 5000 });
1793
+
1794
+ // Check if there's a summary or file already selected - if so, clear it first
1795
+ const hasSummary = screen.queryByText(/import completed/i);
1796
+ const hasSelectedFile = screen.queryByText(/selected:/i);
1797
+ if (hasSummary || hasSelectedFile) {
1798
+ rerender(<ImportModal {...baseProps} isOpen={false} />);
1799
+ await waitFor(() => {
1800
+ expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
1801
+ }, { timeout: 2000 });
1802
+ await new Promise(resolve => setTimeout(resolve, 300));
1803
+ rerender(<ImportModal {...baseProps} isOpen={true} />);
1804
+ await waitForDialog();
1805
+ await waitFor(() => {
1806
+ expect(screen.getByText('Import Data')).toBeInTheDocument();
1807
+ }, { timeout: 5000 });
1808
+ }
1809
+
1810
+ // Wait for file selection area to be rendered (default uploadText is "Choose a CSV file to upload")
1811
+ await waitFor(() => {
1812
+ const uploadText = screen.queryByText(/choose a csv file/i);
1813
+ expect(uploadText).toBeInTheDocument();
1814
+ }, { timeout: 5000 });
1815
+
1816
+ // Then wait for buttons (should exist when no summary and no file selected)
1817
+ await waitFor(() => {
1818
+ const selectFileButton = findButtonByText(/select file/i);
1819
+ expect(selectFileButton).toBeInTheDocument();
1820
+ }, { timeout: 5000 });
1821
+
1822
+ await waitFor(() => {
1823
+ const importButton = findButtonByText(/^import$/i);
1824
+ expect(importButton).toBeInTheDocument();
1825
+ }, { timeout: 5000 });
1826
+
1827
+ await waitFor(() => {
1828
+ const cancelButton = findButtonByText(/cancel/i);
1829
+ expect(cancelButton).toBeInTheDocument();
1830
+ }, { timeout: 5000 });
1831
+ });
1832
+ });
1833
+ });
1834
+