@goplusvn/core 0.1.0 → 0.1.2

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 (591) hide show
  1. package/package.json +31 -175
  2. package/src/assets/erp_wallpaper.png +0 -0
  3. package/src/assets/goeat_logo.png +0 -0
  4. package/src/audit/audit-manager.ts +139 -0
  5. package/src/audit/index.ts +11 -0
  6. package/src/audit/memory-audit-logger.ts +86 -0
  7. package/src/audit/types.ts +50 -0
  8. package/src/auth/auth-service.ts +97 -0
  9. package/src/auth/index.ts +266 -0
  10. package/src/code-generation/index.ts +69 -0
  11. package/src/configs/auth-routes.ts +17 -0
  12. package/src/configs/crud.ts +136 -0
  13. package/src/configs/data/navigations.ts +781 -0
  14. package/src/configs/data/oauth-links.ts +10 -0
  15. package/src/configs/entities/material-categories.config.ts +125 -0
  16. package/src/configs/i18n.ts +12 -0
  17. package/src/configs/index.ts +26 -0
  18. package/src/configs/status.ts +25 -0
  19. package/src/configs/themes.ts +100 -0
  20. package/src/crud/components/crud-bulk-actions.tsx +91 -0
  21. package/src/crud/components/crud-card-view.tsx +241 -0
  22. package/src/crud/components/crud-context.tsx +122 -0
  23. package/src/crud/components/crud-delete-dialog.tsx +145 -0
  24. package/src/crud/components/crud-dialog.tsx +406 -0
  25. package/src/crud/components/crud-empty-state.tsx +104 -0
  26. package/src/crud/components/crud-export-button.tsx +170 -0
  27. package/src/crud/components/crud-field-renderer.tsx +653 -0
  28. package/src/crud/components/crud-filter-chips.tsx +102 -0
  29. package/src/crud/components/crud-filters/checkbox-filter.tsx +97 -0
  30. package/src/crud/components/crud-filters/datetime-filter.tsx +83 -0
  31. package/src/crud/components/crud-filters/filter-builder.tsx +66 -0
  32. package/src/crud/components/crud-filters/index.tsx +76 -0
  33. package/src/crud/components/crud-filters/radio-filter.tsx +86 -0
  34. package/src/crud/components/crud-filters/select-filter.tsx +141 -0
  35. package/src/crud/components/crud-filters/text-filter.tsx +86 -0
  36. package/src/crud/components/crud-form.tsx +642 -0
  37. package/src/crud/components/crud-import-dialog.tsx +440 -0
  38. package/src/crud/components/crud-infinite-scroll.tsx +116 -0
  39. package/src/crud/components/crud-page.tsx +1017 -0
  40. package/src/crud/components/crud-provider.tsx +277 -0
  41. package/src/crud/components/crud-row-actions.tsx +189 -0
  42. package/src/crud/components/crud-search.tsx +82 -0
  43. package/src/crud/components/crud-sheet.tsx +336 -0
  44. package/src/crud/components/crud-table-skeleton.tsx +26 -0
  45. package/src/crud/components/crud-table-toolbar.tsx +91 -0
  46. package/src/crud/components/crud-table.tsx +352 -0
  47. package/src/crud/components/crud-virtual-table.tsx +55 -0
  48. package/src/crud/components/index.tsx +20 -0
  49. package/src/crud/crud-filters/checkbox-filter.tsx +87 -0
  50. package/src/crud/crud-filters/datetime-filter.tsx +82 -0
  51. package/src/crud/crud-filters/filter-builder.tsx +64 -0
  52. package/src/crud/crud-filters/index.tsx +78 -0
  53. package/src/crud/crud-filters/radio-filter.tsx +79 -0
  54. package/src/crud/crud-filters/select-filter.tsx +148 -0
  55. package/src/crud/crud-filters/text-filter.tsx +81 -0
  56. package/src/crud/index.ts +43 -0
  57. package/src/crud/lib/crud-service.test.ts +334 -0
  58. package/src/crud/lib/crud-service.ts +358 -0
  59. package/src/crud/lib/crud-utils.test.ts +354 -0
  60. package/src/crud/lib/crud-utils.ts +299 -0
  61. package/src/crud/lib/crud-validator.ts +247 -0
  62. package/src/crud/lib/data-loader.ts +234 -0
  63. package/src/crud/lib/field-calculator.ts +241 -0
  64. package/src/crud/lib/field-formatter.ts +240 -0
  65. package/src/crud/lib/import-export-service.test.ts +290 -0
  66. package/src/crud/lib/import-export-service.ts +352 -0
  67. package/src/crud/lib/import-server-utils.ts +109 -0
  68. package/src/crud/lib/lazy-loader.ts +241 -0
  69. package/src/crud/lib/parse-filters.ts +85 -0
  70. package/src/crud/lib/permissions.ts +52 -0
  71. package/src/crud/lib/serialize-config.ts +60 -0
  72. package/src/crud/lib/stream-loader.ts +145 -0
  73. package/src/crud/lib/translate-config.ts +335 -0
  74. package/src/crud/lib/types.ts +11 -0
  75. package/src/crud/pages/entity-crud-page.tsx +144 -0
  76. package/src/crud/server.ts +8 -0
  77. package/src/home/constants.tsx +142 -0
  78. package/src/home/feature-showcase.tsx +171 -0
  79. package/src/home/home-page.tsx +191 -0
  80. package/src/home/hooks/index.ts +1 -0
  81. package/src/home/hooks/useWidgetPreferences.ts +167 -0
  82. package/src/home/index.ts +33 -0
  83. package/src/home/quick-access-dialog.tsx +271 -0
  84. package/src/home/quick-access-menu.tsx +267 -0
  85. package/src/home/types.ts +140 -0
  86. package/src/home/welcome-card.tsx +92 -0
  87. package/src/home/widget-container.tsx +258 -0
  88. package/src/home/widgets/base-widget.tsx +200 -0
  89. package/src/home/widgets/customers-widget.tsx +74 -0
  90. package/src/home/widgets/index.ts +6 -0
  91. package/src/home/widgets/orders-widget.tsx +87 -0
  92. package/src/home/widgets/revenue-widget.tsx +71 -0
  93. package/src/home/widgets/stock-widget.tsx +109 -0
  94. package/src/hooks/index.tsx +598 -0
  95. package/src/hooks/use-tenant.test.tsx +30 -0
  96. package/src/hooks/use-tenant.ts +5 -0
  97. package/src/index.ts +17 -0
  98. package/src/infrastructure/__tests__/architecture-verification.spec.ts +103 -0
  99. package/src/infrastructure/api-service.ts +317 -0
  100. package/src/infrastructure/cache/cache-manager.ts +107 -0
  101. package/src/infrastructure/cache/cache.ts +120 -0
  102. package/src/infrastructure/cache/index.ts +8 -0
  103. package/src/infrastructure/cache/types.ts +48 -0
  104. package/src/infrastructure/cron/cron-manager.ts +239 -0
  105. package/src/infrastructure/cron/index.ts +6 -0
  106. package/src/infrastructure/cron/types.ts +41 -0
  107. package/src/infrastructure/event-bus/event-bus.ts +145 -0
  108. package/src/infrastructure/event-bus/index.ts +2 -0
  109. package/src/infrastructure/event-bus/types.ts +22 -0
  110. package/src/infrastructure/index.ts +32 -0
  111. package/src/infrastructure/lock/decorators.ts +67 -0
  112. package/src/infrastructure/lock/index.ts +2 -0
  113. package/src/infrastructure/lock/lock-manager.ts +33 -0
  114. package/src/infrastructure/logger/index.ts +2 -0
  115. package/src/infrastructure/logger/logger.ts +96 -0
  116. package/src/infrastructure/logger/types.ts +25 -0
  117. package/src/layout/index.tsx +185 -0
  118. package/src/navigation/index.ts +91 -0
  119. package/src/notification/index.ts +14 -0
  120. package/src/notification/notification-service.ts +120 -0
  121. package/src/notification/storage/in-memory.ts +56 -0
  122. package/src/notification/storage/index.ts +1 -0
  123. package/src/notification/types.ts +51 -0
  124. package/src/organization/branch-service.ts +299 -0
  125. package/src/organization/branches.config.ts +154 -0
  126. package/src/organization/index.ts +5 -0
  127. package/src/plugin/apps-registry.ts +97 -0
  128. package/src/plugin/index.ts +5 -0
  129. package/src/plugin/types.ts +41 -0
  130. package/src/providers/index.tsx +109 -0
  131. package/src/providers/tenant-provider.tsx +45 -0
  132. package/src/rbac/components/roles/role-card.tsx +158 -0
  133. package/src/rbac/components/roles/role-stats-cards.tsx +29 -0
  134. package/src/rbac/components/roles/role-toolbar.tsx +123 -0
  135. package/src/rbac/hooks/use-role-operations.ts +159 -0
  136. package/src/rbac/hooks/use-roles-data.ts +59 -0
  137. package/src/rbac/index.ts +297 -0
  138. package/src/rbac/lib/permission-helpers.ts +63 -0
  139. package/src/rbac/pages/action-list-page.tsx +25 -0
  140. package/src/rbac/pages/resource-list-page.tsx +25 -0
  141. package/src/rbac/pages/role-list-page.tsx +378 -0
  142. package/src/rbac/permission-service.ts +140 -0
  143. package/src/rbac/permissions.ts +135 -0
  144. package/src/rbac/resource-service.ts +115 -0
  145. package/src/rbac/resource-validator.ts +119 -0
  146. package/src/rbac/role-service.ts +165 -0
  147. package/src/rbac/server.ts +16 -0
  148. package/src/rbac/types.ts +38 -0
  149. package/src/schemas/action.schema.ts +66 -0
  150. package/src/schemas/branch.schema.ts +52 -0
  151. package/src/schemas/coming-soon-schema.ts +9 -0
  152. package/src/schemas/company.schema.ts +44 -0
  153. package/src/schemas/forgot-passward-schema.ts +9 -0
  154. package/src/schemas/index.ts +30 -0
  155. package/src/schemas/material-category.schema.ts +43 -0
  156. package/src/schemas/material-pricing.schema.ts +74 -0
  157. package/src/schemas/material.schema.ts +76 -0
  158. package/src/schemas/materials.ts +52 -0
  159. package/src/schemas/new-passward-schema.ts +15 -0
  160. package/src/schemas/partner-company.schema.ts +149 -0
  161. package/src/schemas/register-schema.ts +36 -0
  162. package/src/schemas/resource.schema.ts +133 -0
  163. package/src/schemas/role.schema.ts +11 -0
  164. package/src/schemas/sign-in-schema.ts +24 -0
  165. package/src/schemas/supplier-pricing.schema.ts +15 -0
  166. package/src/schemas/supplier.schema.ts +120 -0
  167. package/src/schemas/system-category-group.schema.ts +67 -0
  168. package/src/schemas/system-category.schema.ts +77 -0
  169. package/src/schemas/system-config.schema.ts +118 -0
  170. package/src/schemas/uom.schema.ts +75 -0
  171. package/src/schemas/user-supplier.schema.ts +179 -0
  172. package/src/schemas/user.schema.ts +18 -0
  173. package/src/schemas/verify-email-schema.ts +9 -0
  174. package/src/schemas/warehouse.schema.ts +49 -0
  175. package/src/system/components/categories/category-list.tsx +529 -0
  176. package/src/system/components/categories/category-manager.tsx +89 -0
  177. package/src/system/components/categories/group-sidebar.tsx +308 -0
  178. package/src/system/components/settings/setting-dialogs.tsx +197 -0
  179. package/src/system/components/settings/setting-field.tsx +291 -0
  180. package/src/system/components/settings/setting-form-dialog.tsx +308 -0
  181. package/src/system/components/settings/settings-groups.ts +80 -0
  182. package/src/system/components/settings/settings-search.tsx +71 -0
  183. package/src/system/components/settings/settings-section.tsx +74 -0
  184. package/src/system/components/settings/settings-sidebar.tsx +81 -0
  185. package/src/system/constants.ts +3 -0
  186. package/src/system/index.ts +150 -0
  187. package/src/system/job-manager.ts +176 -0
  188. package/src/system/pages/components/categories/category-list.tsx +537 -0
  189. package/src/system/pages/components/categories/category-manager.tsx +90 -0
  190. package/src/system/pages/components/categories/group-sidebar.tsx +311 -0
  191. package/src/system/pages/components/settings/sales-rules-settings.tsx +222 -0
  192. package/src/system/pages/components/settings/setting-dialogs.tsx +197 -0
  193. package/src/system/pages/components/settings/setting-field.tsx +292 -0
  194. package/src/system/pages/components/settings/setting-form-dialog.tsx +308 -0
  195. package/src/system/pages/components/settings/settings-groups.ts +87 -0
  196. package/src/system/pages/components/settings/settings-page.tsx +372 -0
  197. package/src/system/pages/components/settings/settings-search.tsx +71 -0
  198. package/src/system/pages/components/settings/settings-section.tsx +74 -0
  199. package/src/system/pages/components/settings/settings-sidebar.tsx +81 -0
  200. package/src/system/pages/components/settings/system-settings.tsx +244 -0
  201. package/src/system/pages/system-category-page.tsx +15 -0
  202. package/src/system/pages/system-settings-page.tsx +380 -0
  203. package/src/system/schemas/system-category-group.schema.ts +46 -0
  204. package/src/system/schemas/system-category.schema.ts +56 -0
  205. package/src/system/services/settings-service.ts +127 -0
  206. package/src/system/services/system-category-service.ts +63 -0
  207. package/src/system/types.ts +45 -0
  208. package/src/types/index.ts +703 -0
  209. package/src/ui/auth/auth-layout.tsx +135 -0
  210. package/src/ui/auth/forgot-password-form.tsx +98 -0
  211. package/src/ui/auth/index.tsx +7 -0
  212. package/src/ui/auth/new-password-form.tsx +107 -0
  213. package/src/ui/auth/oauth-links.tsx +30 -0
  214. package/src/ui/auth/register-form.tsx +202 -0
  215. package/src/ui/auth/sign-in-form.tsx +238 -0
  216. package/src/ui/auth/verify-email-form.tsx +104 -0
  217. package/src/ui/crud/index.tsx +10 -0
  218. package/src/ui/data-display/accordion.tsx +65 -0
  219. package/src/ui/data-display/aspect-ratio.tsx +11 -0
  220. package/src/ui/data-display/avatar.tsx +163 -0
  221. package/src/ui/data-display/bento-grid.tsx +77 -0
  222. package/src/ui/data-display/carousel.tsx +249 -0
  223. package/src/ui/data-display/chart.tsx +363 -0
  224. package/src/ui/data-display/code-block-highlight.tsx +54 -0
  225. package/src/ui/data-display/collapsible.tsx +42 -0
  226. package/src/ui/data-display/compact-stat-bar.tsx +149 -0
  227. package/src/ui/data-display/data-table/data-table-context.tsx +255 -0
  228. package/src/ui/data-display/data-table/data-table-empty-state.tsx +133 -0
  229. package/src/ui/data-display/data-table/data-table-skeleton.tsx +145 -0
  230. package/src/ui/data-display/data-table/data-table-toolbar.tsx +353 -0
  231. package/src/ui/data-display/data-table/data-table.tsx +597 -0
  232. package/src/ui/data-display/data-table/index.ts +44 -0
  233. package/src/ui/data-display/data-table-column-header.tsx +75 -0
  234. package/src/ui/data-display/data-table-pagination.tsx +130 -0
  235. package/src/ui/data-display/data-table-view-options.tsx +59 -0
  236. package/src/ui/data-display/formatted-number-input.tsx +210 -0
  237. package/src/ui/data-display/highlight.tsx +20 -0
  238. package/src/ui/data-display/hover-card.tsx +48 -0
  239. package/src/ui/data-display/index.tsx +50 -0
  240. package/src/ui/data-display/iphone-15-pro.tsx +114 -0
  241. package/src/ui/data-display/kanban/index.ts +4 -0
  242. package/src/ui/data-display/kanban/kanban-board.tsx +192 -0
  243. package/src/ui/data-display/kanban/kanban-column.tsx +74 -0
  244. package/src/ui/data-display/kanban/kanban-item.tsx +50 -0
  245. package/src/ui/data-display/kanban/kanban-types.ts +21 -0
  246. package/src/ui/data-display/kpi-card.tsx +68 -0
  247. package/src/ui/data-display/media-grid.tsx +110 -0
  248. package/src/ui/data-display/safari.tsx +175 -0
  249. package/src/ui/data-display/show-more-text.tsx +55 -0
  250. package/src/ui/data-display/tabs.tsx +68 -0
  251. package/src/ui/data-display/timeline.tsx +256 -0
  252. package/src/ui/feedback/alert.tsx +60 -0
  253. package/src/ui/feedback/context-menu.tsx +245 -0
  254. package/src/ui/feedback/drawer.tsx +132 -0
  255. package/src/ui/feedback/error-dialog.tsx +273 -0
  256. package/src/ui/feedback/index.tsx +183 -0
  257. package/src/ui/feedback/progress.tsx +32 -0
  258. package/src/ui/feedback/sheet.tsx +148 -0
  259. package/src/ui/feedback/sonner.tsx +36 -0
  260. package/src/ui/forms/command.tsx +157 -0
  261. package/src/ui/forms/date-picker.tsx +73 -0
  262. package/src/ui/forms/date-range-picker.tsx +76 -0
  263. package/src/ui/forms/date-time-picker.tsx +109 -0
  264. package/src/ui/forms/editor/editor-menu-bar.tsx +394 -0
  265. package/src/ui/forms/editor/index.tsx +130 -0
  266. package/src/ui/forms/editor/multi-select-example.tsx +1234 -0
  267. package/src/ui/forms/emoji-picker.tsx +109 -0
  268. package/src/ui/forms/file-dropzone.tsx +169 -0
  269. package/src/ui/forms/file-thumbnail.tsx +29 -0
  270. package/src/ui/forms/index.tsx +201 -0
  271. package/src/ui/forms/input-file.tsx +99 -0
  272. package/src/ui/forms/input-group.tsx +46 -0
  273. package/src/ui/forms/input-otp.tsx +81 -0
  274. package/src/ui/forms/input-phone.tsx +172 -0
  275. package/src/ui/forms/input-spin.tsx +116 -0
  276. package/src/ui/forms/input-tags.tsx +219 -0
  277. package/src/ui/forms/input-time.tsx +42 -0
  278. package/src/ui/forms/multi-select.tsx +629 -0
  279. package/src/ui/forms/multiple-date-picker.tsx +74 -0
  280. package/src/ui/forms/radio-group.tsx +42 -0
  281. package/src/ui/forms/rating.tsx +158 -0
  282. package/src/ui/forms/time-picker.tsx +57 -0
  283. package/src/ui/index.tsx +17 -0
  284. package/src/ui/layout/animated-list.tsx +77 -0
  285. package/src/ui/layout/animated-sidebar.tsx +294 -0
  286. package/src/ui/layout/command-menu.tsx +355 -0
  287. package/src/ui/layout/customizer.tsx +324 -0
  288. package/src/ui/layout/footer.tsx +43 -0
  289. package/src/ui/layout/full-screen-toggle.tsx +52 -0
  290. package/src/ui/layout/header-breadcrumb.tsx +77 -0
  291. package/src/ui/layout/horizontal-layout-header.tsx +83 -0
  292. package/src/ui/layout/horizontal-layout.tsx +50 -0
  293. package/src/ui/layout/index.tsx +25 -0
  294. package/src/ui/layout/language-dropdown.tsx +103 -0
  295. package/src/ui/layout/logo.tsx +63 -0
  296. package/src/ui/layout/main-layout.tsx +57 -0
  297. package/src/ui/layout/mode-dropdown.tsx +58 -0
  298. package/src/ui/layout/notification-dropdown.tsx +127 -0
  299. package/src/ui/layout/page-tabs.tsx +306 -0
  300. package/src/ui/layout/route-cache.tsx +214 -0
  301. package/src/ui/layout/sidebar-group-icon-menu.tsx +195 -0
  302. package/src/ui/layout/sidebar.tsx +279 -0
  303. package/src/ui/layout/tab-content-cache.tsx +201 -0
  304. package/src/ui/layout/tab-navigation-provider.tsx +536 -0
  305. package/src/ui/layout/toggle-mobile-sidebar.tsx +33 -0
  306. package/src/ui/layout/top-bar-header-menubar.tsx +412 -0
  307. package/src/ui/layout/user-dropdown.tsx +188 -0
  308. package/src/ui/layout/vertical-layout-header.tsx +65 -0
  309. package/src/ui/layout/vertical-layout.tsx +47 -0
  310. package/src/ui/management/audit-log-page.tsx +209 -0
  311. package/src/ui/management/cache-management.tsx +349 -0
  312. package/src/ui/management/index.ts +3 -0
  313. package/src/ui/management/job-management.tsx +308 -0
  314. package/src/ui/pages/not-found.tsx +30 -0
  315. package/src/ui/primitives/badge.tsx +66 -0
  316. package/src/ui/primitives/breadcrumb.tsx +103 -0
  317. package/src/ui/primitives/button.tsx +129 -0
  318. package/src/ui/primitives/calendar.tsx +74 -0
  319. package/src/ui/primitives/card.tsx +86 -0
  320. package/src/ui/primitives/checkbox.tsx +31 -0
  321. package/src/ui/primitives/client.ts +30 -0
  322. package/src/ui/primitives/combobox.tsx +290 -0
  323. package/src/ui/primitives/dialog.tsx +121 -0
  324. package/src/ui/primitives/dropdown-menu.tsx +239 -0
  325. package/src/ui/primitives/dynamic-icon.tsx +24 -0
  326. package/src/ui/primitives/index.tsx +134 -0
  327. package/src/ui/primitives/input-number.tsx +131 -0
  328. package/src/ui/primitives/input.tsx +22 -0
  329. package/src/ui/primitives/keyboard.tsx +23 -0
  330. package/src/ui/primitives/label.tsx +24 -0
  331. package/src/ui/primitives/menubar.tsx +262 -0
  332. package/src/ui/primitives/navigation-menu.tsx +157 -0
  333. package/src/ui/primitives/pagination.tsx +118 -0
  334. package/src/ui/primitives/popover.tsx +56 -0
  335. package/src/ui/primitives/prefetch-link.tsx +60 -0
  336. package/src/ui/primitives/resizable.tsx +59 -0
  337. package/src/ui/primitives/scroll-area.tsx +63 -0
  338. package/src/ui/primitives/select.tsx +172 -0
  339. package/src/ui/primitives/separator.tsx +51 -0
  340. package/src/ui/primitives/sidebar.tsx +844 -0
  341. package/src/ui/primitives/slider.tsx +27 -0
  342. package/src/ui/primitives/status-badge.tsx +47 -0
  343. package/src/ui/primitives/sticky-layout.tsx +50 -0
  344. package/src/ui/primitives/switch.tsx +29 -0
  345. package/src/ui/primitives/table.tsx +116 -0
  346. package/src/ui/primitives/tabs.tsx +55 -0
  347. package/src/ui/primitives/toggle-group.tsx +70 -0
  348. package/src/ui/primitives/toggle.tsx +47 -0
  349. package/src/ui/primitives/tooltip.tsx +59 -0
  350. package/src/user/components/dangerous-zone.tsx +34 -0
  351. package/src/user/components/delete-account-form.tsx +40 -0
  352. package/src/user/components/index.ts +4 -0
  353. package/src/user/components/profile-info-form.tsx +390 -0
  354. package/src/user/components/profile-info.tsx +32 -0
  355. package/src/user/components/unified-profile-dialog.tsx +1019 -0
  356. package/src/user/components/user-stats.tsx +27 -0
  357. package/src/user/components/user-toolbar.tsx +137 -0
  358. package/src/user/components/users-card-view.tsx +253 -0
  359. package/src/user/index.ts +11 -0
  360. package/src/user/pages/user-list-page.tsx +234 -0
  361. package/src/user/pages/users-client-page.tsx +385 -0
  362. package/src/user/profile-page.tsx +19 -0
  363. package/src/user/schemas.ts +68 -0
  364. package/src/user/types.ts +34 -0
  365. package/src/user/user-service.ts +538 -0
  366. package/src/utils/index.ts +906 -0
  367. package/src/workflow/activity-timeline.tsx +412 -0
  368. package/src/workflow/approval-workflow.tsx +31 -0
  369. package/src/workflow/index.ts +2 -0
  370. package/dist/audit/index.d.mts +0 -115
  371. package/dist/audit/index.d.ts +0 -115
  372. package/dist/audit/index.js +0 -204
  373. package/dist/audit/index.js.map +0 -1
  374. package/dist/audit/index.mjs +0 -200
  375. package/dist/audit/index.mjs.map +0 -1
  376. package/dist/auth/index.d.mts +0 -86
  377. package/dist/auth/index.d.ts +0 -86
  378. package/dist/auth/index.js +0 -210
  379. package/dist/auth/index.js.map +0 -1
  380. package/dist/auth/index.mjs +0 -198
  381. package/dist/auth/index.mjs.map +0 -1
  382. package/dist/button-1dWvP9Ib.d.mts +0 -30
  383. package/dist/button-1dWvP9Ib.d.ts +0 -30
  384. package/dist/calendar-2QzdEo1z.d.mts +0 -20
  385. package/dist/calendar-2QzdEo1z.d.ts +0 -20
  386. package/dist/code-generation/index.d.mts +0 -30
  387. package/dist/code-generation/index.d.ts +0 -30
  388. package/dist/code-generation/index.js +0 -31
  389. package/dist/code-generation/index.js.map +0 -1
  390. package/dist/code-generation/index.mjs +0 -28
  391. package/dist/code-generation/index.mjs.map +0 -1
  392. package/dist/configs/index.d.mts +0 -175
  393. package/dist/configs/index.d.ts +0 -175
  394. package/dist/configs/index.js +0 -254
  395. package/dist/configs/index.js.map +0 -1
  396. package/dist/configs/index.mjs +0 -233
  397. package/dist/configs/index.mjs.map +0 -1
  398. package/dist/crud/index.d.mts +0 -646
  399. package/dist/crud/index.d.ts +0 -646
  400. package/dist/crud/index.js +0 -11772
  401. package/dist/crud/index.js.map +0 -1
  402. package/dist/crud/index.mjs +0 -11665
  403. package/dist/crud/index.mjs.map +0 -1
  404. package/dist/crud/server.d.mts +0 -20
  405. package/dist/crud/server.d.ts +0 -20
  406. package/dist/crud/server.js +0 -123
  407. package/dist/crud/server.js.map +0 -1
  408. package/dist/crud/server.mjs +0 -120
  409. package/dist/crud/server.mjs.map +0 -1
  410. package/dist/data-table-skeleton-12NA8Mjx.d.mts +0 -39
  411. package/dist/data-table-skeleton-12NA8Mjx.d.ts +0 -39
  412. package/dist/dialog-bKfjZMTd.d.mts +0 -22
  413. package/dist/dialog-bKfjZMTd.d.ts +0 -22
  414. package/dist/dynamic-icon-DrGIiu2N.d.mts +0 -10
  415. package/dist/dynamic-icon-DrGIiu2N.d.ts +0 -10
  416. package/dist/home/index.d.mts +0 -269
  417. package/dist/home/index.d.ts +0 -269
  418. package/dist/home/index.js +0 -1678
  419. package/dist/home/index.js.map +0 -1
  420. package/dist/home/index.mjs +0 -1635
  421. package/dist/home/index.mjs.map +0 -1
  422. package/dist/hooks/index.d.mts +0 -7
  423. package/dist/hooks/index.d.ts +0 -7
  424. package/dist/hooks/index.js +0 -8316
  425. package/dist/hooks/index.js.map +0 -1
  426. package/dist/hooks/index.mjs +0 -8255
  427. package/dist/hooks/index.mjs.map +0 -1
  428. package/dist/index-50hpiPrV.d.ts +0 -116
  429. package/dist/index-B9zQVEVi.d.mts +0 -116
  430. package/dist/index.d.mts +0 -5
  431. package/dist/index.d.ts +0 -5
  432. package/dist/index.js +0 -123
  433. package/dist/index.js.map +0 -1
  434. package/dist/index.mjs +0 -118
  435. package/dist/index.mjs.map +0 -1
  436. package/dist/infrastructure/index.d.mts +0 -423
  437. package/dist/infrastructure/index.d.ts +0 -423
  438. package/dist/infrastructure/index.js +0 -633
  439. package/dist/infrastructure/index.js.map +0 -1
  440. package/dist/infrastructure/index.mjs +0 -619
  441. package/dist/infrastructure/index.mjs.map +0 -1
  442. package/dist/label-DWTEkNPo.d.ts +0 -226
  443. package/dist/label-LPpdcoBx.d.mts +0 -226
  444. package/dist/layout/index.d.mts +0 -48
  445. package/dist/layout/index.d.ts +0 -48
  446. package/dist/layout/index.js +0 -117
  447. package/dist/layout/index.js.map +0 -1
  448. package/dist/layout/index.mjs +0 -90
  449. package/dist/layout/index.mjs.map +0 -1
  450. package/dist/navigation/index.d.mts +0 -16
  451. package/dist/navigation/index.d.ts +0 -16
  452. package/dist/navigation/index.js +0 -53
  453. package/dist/navigation/index.js.map +0 -1
  454. package/dist/navigation/index.mjs +0 -50
  455. package/dist/navigation/index.mjs.map +0 -1
  456. package/dist/notification/index.d.mts +0 -105
  457. package/dist/notification/index.d.ts +0 -105
  458. package/dist/notification/index.js +0 -278
  459. package/dist/notification/index.js.map +0 -1
  460. package/dist/notification/index.mjs +0 -274
  461. package/dist/notification/index.mjs.map +0 -1
  462. package/dist/organization/index.d.mts +0 -99
  463. package/dist/organization/index.d.ts +0 -99
  464. package/dist/organization/index.js +0 -360
  465. package/dist/organization/index.js.map +0 -1
  466. package/dist/organization/index.mjs +0 -352
  467. package/dist/organization/index.mjs.map +0 -1
  468. package/dist/plugin/index.d.mts +0 -83
  469. package/dist/plugin/index.d.ts +0 -83
  470. package/dist/plugin/index.js +0 -86
  471. package/dist/plugin/index.js.map +0 -1
  472. package/dist/plugin/index.mjs +0 -84
  473. package/dist/plugin/index.mjs.map +0 -1
  474. package/dist/providers/index.d.mts +0 -25
  475. package/dist/providers/index.d.ts +0 -25
  476. package/dist/providers/index.js +0 -84
  477. package/dist/providers/index.js.map +0 -1
  478. package/dist/providers/index.mjs +0 -77
  479. package/dist/providers/index.mjs.map +0 -1
  480. package/dist/rbac/index.d.mts +0 -226
  481. package/dist/rbac/index.d.ts +0 -226
  482. package/dist/rbac/index.js +0 -4784
  483. package/dist/rbac/index.js.map +0 -1
  484. package/dist/rbac/index.mjs +0 -4722
  485. package/dist/rbac/index.mjs.map +0 -1
  486. package/dist/rbac/permissions.d.mts +0 -26
  487. package/dist/rbac/permissions.d.ts +0 -26
  488. package/dist/rbac/permissions.js +0 -94
  489. package/dist/rbac/permissions.js.map +0 -1
  490. package/dist/rbac/permissions.mjs +0 -90
  491. package/dist/rbac/permissions.mjs.map +0 -1
  492. package/dist/rbac/server.d.mts +0 -1
  493. package/dist/rbac/server.d.ts +0 -1
  494. package/dist/rbac/server.js +0 -128
  495. package/dist/rbac/server.js.map +0 -1
  496. package/dist/rbac/server.mjs +0 -124
  497. package/dist/rbac/server.mjs.map +0 -1
  498. package/dist/schemas/index.d.mts +0 -1257
  499. package/dist/schemas/index.d.ts +0 -1257
  500. package/dist/schemas/index.js +0 -572
  501. package/dist/schemas/index.js.map +0 -1
  502. package/dist/schemas/index.mjs +0 -523
  503. package/dist/schemas/index.mjs.map +0 -1
  504. package/dist/server-QuYCTa89.d.mts +0 -83
  505. package/dist/server-QuYCTa89.d.ts +0 -83
  506. package/dist/sonner-C74GlRDQ.d.mts +0 -71
  507. package/dist/sonner-C74GlRDQ.d.ts +0 -71
  508. package/dist/status-BOXZgIqX.d.mts +0 -12
  509. package/dist/status-BOXZgIqX.d.ts +0 -12
  510. package/dist/system/index.d.mts +0 -77
  511. package/dist/system/index.d.ts +0 -77
  512. package/dist/system/index.js +0 -102
  513. package/dist/system/index.js.map +0 -1
  514. package/dist/system/index.mjs +0 -100
  515. package/dist/system/index.mjs.map +0 -1
  516. package/dist/tabs-C6FfBwPY.d.mts +0 -18
  517. package/dist/tabs-C6FfBwPY.d.ts +0 -18
  518. package/dist/tenant-provider-B8eC_Wpb.d.mts +0 -27
  519. package/dist/tenant-provider-B8eC_Wpb.d.ts +0 -27
  520. package/dist/types/index.d.mts +0 -469
  521. package/dist/types/index.d.ts +0 -469
  522. package/dist/types/index.js +0 -25
  523. package/dist/types/index.js.map +0 -1
  524. package/dist/types/index.mjs +0 -21
  525. package/dist/types/index.mjs.map +0 -1
  526. package/dist/ui/auth.d.mts +0 -39
  527. package/dist/ui/auth.d.ts +0 -39
  528. package/dist/ui/auth.js +0 -4941
  529. package/dist/ui/auth.js.map +0 -1
  530. package/dist/ui/auth.mjs +0 -4896
  531. package/dist/ui/auth.mjs.map +0 -1
  532. package/dist/ui/crud.d.mts +0 -2
  533. package/dist/ui/crud.d.ts +0 -2
  534. package/dist/ui/crud.js +0 -4
  535. package/dist/ui/crud.js.map +0 -1
  536. package/dist/ui/crud.mjs +0 -3
  537. package/dist/ui/crud.mjs.map +0 -1
  538. package/dist/ui/data-display.d.mts +0 -596
  539. package/dist/ui/data-display.d.ts +0 -596
  540. package/dist/ui/data-display.js +0 -5307
  541. package/dist/ui/data-display.js.map +0 -1
  542. package/dist/ui/data-display.mjs +0 -5212
  543. package/dist/ui/data-display.mjs.map +0 -1
  544. package/dist/ui/feedback.d.mts +0 -55
  545. package/dist/ui/feedback.d.ts +0 -55
  546. package/dist/ui/feedback.js +0 -2608
  547. package/dist/ui/feedback.js.map +0 -1
  548. package/dist/ui/feedback.mjs +0 -2526
  549. package/dist/ui/feedback.mjs.map +0 -1
  550. package/dist/ui/forms.d.mts +0 -309
  551. package/dist/ui/forms.d.ts +0 -309
  552. package/dist/ui/forms.js +0 -4656
  553. package/dist/ui/forms.js.map +0 -1
  554. package/dist/ui/forms.mjs +0 -4571
  555. package/dist/ui/forms.mjs.map +0 -1
  556. package/dist/ui/index.d.mts +0 -331
  557. package/dist/ui/index.d.ts +0 -331
  558. package/dist/ui/index.js +0 -16953
  559. package/dist/ui/index.js.map +0 -1
  560. package/dist/ui/index.mjs +0 -16598
  561. package/dist/ui/index.mjs.map +0 -1
  562. package/dist/ui/primitives/client.d.mts +0 -61
  563. package/dist/ui/primitives/client.d.ts +0 -61
  564. package/dist/ui/primitives/client.js +0 -3408
  565. package/dist/ui/primitives/client.js.map +0 -1
  566. package/dist/ui/primitives/client.mjs +0 -3256
  567. package/dist/ui/primitives/client.mjs.map +0 -1
  568. package/dist/ui/primitives.d.mts +0 -113
  569. package/dist/ui/primitives.d.ts +0 -113
  570. package/dist/ui/primitives.js +0 -3356
  571. package/dist/ui/primitives.js.map +0 -1
  572. package/dist/ui/primitives.mjs +0 -3227
  573. package/dist/ui/primitives.mjs.map +0 -1
  574. package/dist/user/index.d.mts +0 -228
  575. package/dist/user/index.d.ts +0 -228
  576. package/dist/user/index.js +0 -4306
  577. package/dist/user/index.js.map +0 -1
  578. package/dist/user/index.mjs +0 -4260
  579. package/dist/user/index.mjs.map +0 -1
  580. package/dist/utils/index.d.mts +0 -205
  581. package/dist/utils/index.d.ts +0 -205
  582. package/dist/utils/index.js +0 -574
  583. package/dist/utils/index.js.map +0 -1
  584. package/dist/utils/index.mjs +0 -514
  585. package/dist/utils/index.mjs.map +0 -1
  586. package/dist/workflow/index.d.mts +0 -40
  587. package/dist/workflow/index.d.ts +0 -40
  588. package/dist/workflow/index.js +0 -3710
  589. package/dist/workflow/index.js.map +0 -1
  590. package/dist/workflow/index.mjs +0 -3677
  591. package/dist/workflow/index.mjs.map +0 -1
@@ -0,0 +1,629 @@
1
+ "use client";
2
+
3
+ import * as React from "react";
4
+ import { useCallback, useEffect, useMemo, useRef, useState } from "react";
5
+ import { Check, ChevronsUpDown, Search, X } from "lucide-react";
6
+
7
+ import { cn } from "../../utils";
8
+
9
+ import { Badge, Button, Input } from "../primitives";
10
+ import { ScrollArea } from "../primitives/client";
11
+
12
+ export interface MultiSelectOption {
13
+ value: string | number | boolean;
14
+ label: string;
15
+ /** Whether this option is disabled and cannot be selected */
16
+ disabled?: boolean;
17
+ /** Optional icon component to display alongside the option */
18
+ icon?: React.ComponentType<{ className?: string }>;
19
+ }
20
+
21
+ interface MultiSelectProps {
22
+ options: MultiSelectOption[];
23
+ value?: (string | number | boolean)[];
24
+ onValueChange?: (value: (string | number | boolean)[]) => void;
25
+ /** Default value for uncontrolled mode */
26
+ defaultValue?: (string | number | boolean)[];
27
+ placeholder?: string;
28
+ searchPlaceholder?: string;
29
+ emptyText?: string;
30
+ disabled?: boolean;
31
+ /**
32
+ * Maximum number of items to display. Extra selected items will be summarized
33
+ * Optional, defaults to 3
34
+ */
35
+ maxCount?: number;
36
+ /**
37
+ * Responsive configuration for different screen sizes
38
+ * Can be boolean true for default responsive behavior or an object for custom configuration
39
+ */
40
+ responsive?: boolean | ResponsiveConfig;
41
+ className?: string;
42
+ id?: string;
43
+ }
44
+
45
+ /**
46
+ * Responsive configuration for different screen sizes
47
+ */
48
+ interface ResponsiveConfig {
49
+ /** Configuration for mobile devices (< 640px) */
50
+ mobile?: {
51
+ maxCount?: number;
52
+ compactMode?: boolean;
53
+ };
54
+ /** Configuration for tablet devices (640px - 1024px) */
55
+ tablet?: {
56
+ maxCount?: number;
57
+ compactMode?: boolean;
58
+ };
59
+ /** Configuration for desktop devices (> 1024px) */
60
+ desktop?: {
61
+ maxCount?: number;
62
+ compactMode?: boolean;
63
+ };
64
+ }
65
+
66
+ /**
67
+ * Imperative methods that will be exposed through ref
68
+ * (Currently not implemented - will be added in later steps)
69
+ */
70
+ export interface MultiSelectRef {
71
+ /** Reset to default value */
72
+ reset: () => void;
73
+ /** Get current selected values */
74
+ getSelectedValues: () => (string | number | boolean)[];
75
+ /** Set selected values programmatically */
76
+ setSelectedValues: (values: (string | number | boolean)[]) => void;
77
+ /** Clear all selected values */
78
+ clear: () => void;
79
+ /** Focus the component */
80
+ focus: () => void;
81
+ }
82
+
83
+ export const MultiSelect = React.forwardRef<MultiSelectRef, MultiSelectProps>(
84
+ (
85
+ {
86
+ options,
87
+ value: controlledValue,
88
+ onValueChange,
89
+ defaultValue = [],
90
+ placeholder = "Select options...",
91
+ searchPlaceholder = "Search...",
92
+ emptyText = "No option found.",
93
+ maxCount = 3,
94
+ disabled = false,
95
+ responsive = false,
96
+ className,
97
+ id,
98
+ },
99
+ ref,
100
+ ) => {
101
+ // State for uncontrolled mode
102
+ const [internalValue, setInternalValue] =
103
+ useState<(string | number | boolean)[]>(defaultValue);
104
+ const [open, setOpen] = useState(false);
105
+ const [searchValue, setSearchValue] = useState("");
106
+
107
+ // Determine if we're in controlled or uncontrolled mode
108
+ const isControlled = controlledValue !== undefined;
109
+ const value = isControlled ? controlledValue : internalValue;
110
+
111
+ // Refs
112
+ const containerRef = useRef<HTMLDivElement>(null);
113
+ const searchInputRef = useRef<HTMLInputElement>(null);
114
+ const dropdownRef = useRef<HTMLDivElement>(null);
115
+ const buttonRef = useRef<HTMLButtonElement>(null);
116
+
117
+ // Accessibility - Live regions for screen reader announcements
118
+ const [politeMessage, setPoliteMessage] = useState("");
119
+ const [assertiveMessage, setAssertiveMessage] = useState("");
120
+ const prevSelectedCount = useRef(value.length);
121
+
122
+ const announce = useCallback(
123
+ (message: string, priority: "polite" | "assertive" = "polite") => {
124
+ if (priority === "assertive") {
125
+ setAssertiveMessage(message);
126
+ setTimeout(() => setAssertiveMessage(""), 100);
127
+ } else {
128
+ setPoliteMessage(message);
129
+ setTimeout(() => setPoliteMessage(""), 100);
130
+ }
131
+ },
132
+ [],
133
+ );
134
+
135
+ // Responsive Design
136
+ const [screenSize, setScreenSize] = useState<
137
+ "mobile" | "tablet" | "desktop"
138
+ >("desktop");
139
+
140
+ useEffect(() => {
141
+ if (typeof window === "undefined") return;
142
+
143
+ const handleResize = () => {
144
+ const width = window.innerWidth;
145
+ if (width < 640) {
146
+ setScreenSize("mobile");
147
+ } else if (width < 1024) {
148
+ setScreenSize("tablet");
149
+ } else {
150
+ setScreenSize("desktop");
151
+ }
152
+ };
153
+
154
+ handleResize();
155
+ window.addEventListener("resize", handleResize);
156
+ return () => window.removeEventListener("resize", handleResize);
157
+ }, []);
158
+
159
+ const getResponsiveSettings = useCallback(() => {
160
+ if (!responsive) {
161
+ return {
162
+ maxCount: maxCount,
163
+ compactMode: false,
164
+ };
165
+ }
166
+
167
+ if (responsive === true) {
168
+ const defaultResponsive = {
169
+ mobile: { maxCount: 2, compactMode: true },
170
+ tablet: { maxCount: 4, compactMode: false },
171
+ desktop: { maxCount: 6, compactMode: false },
172
+ };
173
+ const currentSettings = defaultResponsive[screenSize];
174
+ return {
175
+ maxCount: currentSettings?.maxCount ?? maxCount,
176
+ compactMode: currentSettings?.compactMode ?? false,
177
+ };
178
+ }
179
+
180
+ const currentSettings = responsive[screenSize];
181
+ return {
182
+ maxCount: currentSettings?.maxCount ?? maxCount,
183
+ compactMode: currentSettings?.compactMode ?? false,
184
+ };
185
+ }, [responsive, screenSize, maxCount]);
186
+
187
+ const responsiveSettings = getResponsiveSettings();
188
+
189
+ // Filter options based on search value
190
+ const filteredOptions = useMemo(() => {
191
+ if (!searchValue.trim()) {
192
+ return options;
193
+ }
194
+ const searchLower = searchValue.toLowerCase();
195
+ return options.filter(
196
+ (option) =>
197
+ option.label.toLowerCase().includes(searchLower) ||
198
+ String(option.value).toLowerCase().includes(searchLower),
199
+ );
200
+ }, [options, searchValue]);
201
+
202
+ // Reset search when dropdown closes
203
+ useEffect(() => {
204
+ if (!open) {
205
+ setSearchValue("");
206
+ }
207
+ }, [open]);
208
+
209
+ // Focus search input when dropdown opens
210
+ useEffect(() => {
211
+ if (open && searchInputRef.current) {
212
+ // Small delay to ensure dropdown is rendered
213
+ setTimeout(() => {
214
+ searchInputRef.current?.focus();
215
+ }, 50);
216
+ }
217
+ }, [open]);
218
+
219
+ // Close dropdown when clicking outside
220
+ useEffect(() => {
221
+ if (!open) return;
222
+
223
+ const handleClickOutside = (event: MouseEvent) => {
224
+ const target = event.target as HTMLElement;
225
+ if (
226
+ containerRef.current &&
227
+ !containerRef.current.contains(target) &&
228
+ dropdownRef.current &&
229
+ !dropdownRef.current.contains(target)
230
+ ) {
231
+ // Don't close if clicking inside dialog
232
+ if (target.closest('[data-slot="dialog-content"]')) {
233
+ return;
234
+ }
235
+ setOpen(false);
236
+ }
237
+ };
238
+
239
+ const handleEscape = (event: KeyboardEvent) => {
240
+ if (event.key === "Escape" && open) {
241
+ event.stopPropagation();
242
+ setOpen(false);
243
+ }
244
+ };
245
+
246
+ // Use capture phase to catch events before Dialog
247
+ document.addEventListener("mousedown", handleClickOutside, true);
248
+ document.addEventListener("keydown", handleEscape, true);
249
+
250
+ return () => {
251
+ document.removeEventListener("mousedown", handleClickOutside, true);
252
+ document.removeEventListener("keydown", handleEscape, true);
253
+ };
254
+ }, [open]);
255
+
256
+ // Accessibility announcements
257
+ useEffect(() => {
258
+ const selectedCount = value.length;
259
+ const totalOptions = options.filter((opt) => !opt.disabled).length;
260
+
261
+ if (selectedCount !== prevSelectedCount.current) {
262
+ const diff = selectedCount - prevSelectedCount.current;
263
+ if (diff > 0) {
264
+ announce(`${selectedCount} of ${totalOptions} options selected.`);
265
+ } else if (diff < 0) {
266
+ announce(
267
+ `Option removed. ${selectedCount} of ${totalOptions} options selected.`,
268
+ );
269
+ }
270
+ prevSelectedCount.current = selectedCount;
271
+ }
272
+ }, [value, announce, options]);
273
+
274
+ // Helper to update value (works for both controlled and uncontrolled)
275
+ const updateValue = useCallback(
276
+ (newValue: (string | number | boolean)[]) => {
277
+ if (!isControlled) {
278
+ setInternalValue(newValue);
279
+ }
280
+ onValueChange?.(newValue);
281
+ },
282
+ [isControlled, onValueChange],
283
+ );
284
+
285
+ const handleSelect = useCallback(
286
+ (optionValue: string | number | boolean) => {
287
+ // Check if option is disabled
288
+ const option = options.find((opt) => opt.value === optionValue);
289
+ if (option?.disabled) {
290
+ return; // Don't allow selecting disabled options
291
+ }
292
+
293
+ const newValue = value.includes(optionValue)
294
+ ? value.filter((v) => v !== optionValue)
295
+ : [...value, optionValue];
296
+ updateValue(newValue);
297
+ },
298
+ [value, options, updateValue],
299
+ );
300
+
301
+ const handleSelectAll = useCallback(() => {
302
+ updateValue(
303
+ filteredOptions.filter((opt) => !opt.disabled).map((opt) => opt.value),
304
+ );
305
+ }, [updateValue, filteredOptions]);
306
+
307
+ const handleClearAll = useCallback(() => {
308
+ updateValue([]);
309
+ }, [updateValue]);
310
+
311
+ const handleRemove = useCallback(
312
+ (optionValue: string | number | boolean, e: React.MouseEvent) => {
313
+ e.stopPropagation();
314
+ updateValue(value.filter((v) => v !== optionValue));
315
+ },
316
+ [updateValue, value],
317
+ );
318
+
319
+ const clearExtraOptions = useCallback(() => {
320
+ const newSelectedValues = value.slice(0, responsiveSettings.maxCount);
321
+ updateValue(newSelectedValues);
322
+ }, [value, responsiveSettings.maxCount, updateValue]);
323
+
324
+ // Imperative API via ref
325
+ React.useImperativeHandle(
326
+ ref,
327
+ () => ({
328
+ reset: () => {
329
+ updateValue(defaultValue);
330
+ setOpen(false);
331
+ setSearchValue("");
332
+ },
333
+ getSelectedValues: () => value,
334
+ setSelectedValues: (values: (string | number | boolean)[]) => {
335
+ updateValue(values);
336
+ },
337
+ clear: () => {
338
+ updateValue([]);
339
+ },
340
+ focus: () => {
341
+ buttonRef.current?.focus();
342
+ },
343
+ }),
344
+ [value, defaultValue, updateValue],
345
+ );
346
+
347
+ const selectedOptions = options.filter((option) =>
348
+ value.includes(option.value),
349
+ );
350
+
351
+ // Get badge color based on index for variety
352
+ const getBadgeColor = (index: number) => {
353
+ const colors = [
354
+ "bg-blue-100 text-blue-700 hover:bg-blue-200",
355
+ "bg-green-100 text-green-700 hover:bg-green-200",
356
+ "bg-purple-100 text-purple-700 hover:bg-purple-200",
357
+ "bg-orange-100 text-orange-700 hover:bg-orange-200",
358
+ "bg-pink-100 text-pink-700 hover:bg-pink-200",
359
+ "bg-cyan-100 text-cyan-700 hover:bg-cyan-200",
360
+ ];
361
+ return colors[index % colors.length];
362
+ };
363
+
364
+ return (
365
+ <div ref={containerRef} className="relative w-full">
366
+ {/* Screen Reader Live Regions */}
367
+ <div className="sr-only">
368
+ <div aria-live="polite" aria-atomic="true" role="status">
369
+ {politeMessage}
370
+ </div>
371
+ <div aria-live="assertive" aria-atomic="true" role="alert">
372
+ {assertiveMessage}
373
+ </div>
374
+ </div>
375
+
376
+ <Button
377
+ ref={buttonRef}
378
+ type="button"
379
+ variant="outline"
380
+ role="combobox"
381
+ aria-expanded={open}
382
+ disabled={disabled}
383
+ className={cn(
384
+ "w-full justify-between h-auto min-h-10 px-3 py-2",
385
+ responsiveSettings.compactMode && "min-h-8 text-sm",
386
+ className,
387
+ )}
388
+ id={id}
389
+ onClick={(e) => {
390
+ e.stopPropagation();
391
+ setOpen(!open);
392
+ }}
393
+ >
394
+ <div className="flex flex-wrap gap-1">
395
+ {selectedOptions.length > 0 ? (
396
+ <>
397
+ {selectedOptions
398
+ .slice(0, responsiveSettings.maxCount)
399
+ .map((option, index) => (
400
+ <Badge
401
+ key={String(option.value)}
402
+ className={cn(
403
+ "mr-1 mb-1",
404
+ getBadgeColor(index),
405
+ responsiveSettings.compactMode &&
406
+ "text-xs px-1.5 py-0.5",
407
+ )}
408
+ onClick={(e) => handleRemove(option.value, e)}
409
+ >
410
+ {option.icon && !responsiveSettings.compactMode && (
411
+ <option.icon className="mr-2 h-4 w-4" />
412
+ )}
413
+ {option.label}
414
+ <div
415
+ role="button"
416
+ tabIndex={0}
417
+ className="ml-1 ring-offset-background rounded-full outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 cursor-pointer"
418
+ onKeyDown={(e) => {
419
+ if (e.key === "Enter") {
420
+ handleRemove(option.value, e as any);
421
+ }
422
+ }}
423
+ onMouseDown={(e) => {
424
+ e.preventDefault();
425
+ e.stopPropagation();
426
+ }}
427
+ onClick={(e) => handleRemove(option.value, e)}
428
+ >
429
+ <X className="h-3 w-3 text-muted-foreground hover:text-foreground" />
430
+ </div>
431
+ </Badge>
432
+ ))}
433
+ {value.length > responsiveSettings.maxCount && (
434
+ <Badge
435
+ className={cn(
436
+ "mr-1 mb-1 bg-muted text-muted-foreground hover:bg-muted",
437
+ responsiveSettings.compactMode && "text-xs px-1.5 py-0.5",
438
+ )}
439
+ >
440
+ +{value.length - responsiveSettings.maxCount} more
441
+ <div
442
+ role="button"
443
+ tabIndex={0}
444
+ className="ml-1 ring-offset-background rounded-full outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 cursor-pointer"
445
+ onClick={(e) => {
446
+ e.stopPropagation();
447
+ clearExtraOptions();
448
+ }}
449
+ >
450
+ <X className="h-3 w-3" />
451
+ </div>
452
+ </Badge>
453
+ )}
454
+ </>
455
+ ) : (
456
+ <span className="text-muted-foreground">{placeholder}</span>
457
+ )}
458
+ </div>
459
+ <ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />
460
+ </Button>
461
+
462
+ {open && (
463
+ <div
464
+ ref={dropdownRef}
465
+ className="absolute z-[100] mt-1 w-full rounded-md border bg-popover text-popover-foreground shadow-md"
466
+ style={{
467
+ top: "100%",
468
+ left: 0,
469
+ }}
470
+ >
471
+ <div className="flex flex-col">
472
+ {/* Search Input */}
473
+ <div className="flex items-center border-b px-3 py-2">
474
+ <Search className="mr-2 h-4 w-4 shrink-0 opacity-50" />
475
+ <Input
476
+ ref={searchInputRef}
477
+ placeholder={searchPlaceholder}
478
+ value={searchValue}
479
+ onChange={(e) => {
480
+ e.stopPropagation();
481
+ setSearchValue(e.target.value);
482
+ }}
483
+ onKeyDown={(e) => {
484
+ // Prevent Dialog from intercepting keyboard events
485
+ e.stopPropagation();
486
+ // Close dropdown on Escape
487
+ if (e.key === "Escape") {
488
+ e.preventDefault();
489
+ setOpen(false);
490
+ }
491
+ }}
492
+ onClick={(e) => {
493
+ e.stopPropagation();
494
+ }}
495
+ onFocus={(e) => {
496
+ e.stopPropagation();
497
+ }}
498
+ className="border-0 focus-visible:ring-0 focus-visible:ring-offset-0 h-9 bg-transparent px-0"
499
+ />
500
+ {searchValue && (
501
+ <Button
502
+ type="button"
503
+ variant="ghost"
504
+ size="sm"
505
+ className="h-6 w-6 p-0"
506
+ onClick={(e) => {
507
+ e.stopPropagation();
508
+ setSearchValue("");
509
+ searchInputRef.current?.focus();
510
+ }}
511
+ >
512
+ <X className="h-3 w-3" />
513
+ </Button>
514
+ )}
515
+ </div>
516
+
517
+ {/* Action Buttons */}
518
+ <div className="flex items-center justify-between border-b p-1">
519
+ <Button
520
+ type="button"
521
+ variant="ghost"
522
+ size="sm"
523
+ className="h-8 px-2 text-xs"
524
+ onClick={(e) => {
525
+ e.stopPropagation();
526
+ handleSelectAll();
527
+ }}
528
+ >
529
+ Select All
530
+ </Button>
531
+ <Button
532
+ type="button"
533
+ variant="ghost"
534
+ size="sm"
535
+ className="h-8 px-2 text-xs"
536
+ onClick={(e) => {
537
+ e.stopPropagation();
538
+ handleClearAll();
539
+ }}
540
+ >
541
+ Clear All
542
+ </Button>
543
+ </div>
544
+
545
+ {/* Options List */}
546
+ <ScrollArea className="max-h-[300px]">
547
+ {filteredOptions.length === 0 ? (
548
+ <div className="py-6 text-center text-sm text-muted-foreground">
549
+ {emptyText}
550
+ </div>
551
+ ) : (
552
+ <div className="p-1">
553
+ {filteredOptions.map((option) => {
554
+ const isSelected = value.includes(option.value);
555
+ const isDisabled = option.disabled || false;
556
+ return (
557
+ <div
558
+ key={String(option.value)}
559
+ role="option"
560
+ aria-selected={isSelected}
561
+ aria-disabled={isDisabled}
562
+ className={cn(
563
+ "relative flex select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none",
564
+ isDisabled
565
+ ? "cursor-not-allowed opacity-50"
566
+ : "cursor-pointer hover:bg-accent hover:text-accent-foreground",
567
+ isSelected &&
568
+ !isDisabled &&
569
+ "bg-accent text-accent-foreground",
570
+ )}
571
+ onMouseDown={(e) => {
572
+ e.preventDefault();
573
+ e.stopPropagation();
574
+ }}
575
+ onClick={(e) => {
576
+ e.preventDefault();
577
+ e.stopPropagation();
578
+ if (!isDisabled) {
579
+ handleSelect(option.value);
580
+ }
581
+ }}
582
+ >
583
+ <div
584
+ className={cn(
585
+ "mr-2 flex h-4 w-4 items-center justify-center rounded-sm border border-primary",
586
+ isSelected
587
+ ? "bg-primary text-primary-foreground"
588
+ : "opacity-50 [&_svg]:invisible",
589
+ )}
590
+ >
591
+ <Check className={cn("h-4 w-4")} />
592
+ </div>
593
+ {option.icon && (
594
+ <option.icon className="mr-2 h-4 w-4 text-muted-foreground" />
595
+ )}
596
+ <span className="flex-1 truncate">
597
+ {option.label}
598
+ </span>
599
+ </div>
600
+ );
601
+ })}
602
+ </div>
603
+ )}
604
+ </ScrollArea>
605
+
606
+ {/* Close Button */}
607
+ <div className="border-t p-2">
608
+ <Button
609
+ type="button"
610
+ variant="outline"
611
+ size="sm"
612
+ className="w-full h-8 text-xs"
613
+ onClick={(e) => {
614
+ e.stopPropagation();
615
+ setOpen(false);
616
+ }}
617
+ >
618
+ Close
619
+ </Button>
620
+ </div>
621
+ </div>
622
+ </div>
623
+ )}
624
+ </div>
625
+ );
626
+ },
627
+ );
628
+
629
+ MultiSelect.displayName = "MultiSelect";
@@ -0,0 +1,74 @@
1
+ "use client";
2
+
3
+ import { format } from "date-fns";
4
+ import { CalendarIcon } from "lucide-react";
5
+
6
+ import type { ComponentProps } from "react";
7
+
8
+ import { cn } from "../../utils";
9
+
10
+ import { Button } from "../primitives";
11
+ import { Calendar } from "../primitives/client";
12
+ import { Popover, PopoverContent, PopoverTrigger } from "../primitives/client";
13
+
14
+ type MultipleDatesPickerProps = Omit<
15
+ ComponentProps<typeof Calendar>,
16
+ "mode" | "selected" | "onSelect"
17
+ > & {
18
+ value?: Date[];
19
+ onValueChange: (dates?: Date[]) => void;
20
+ formatStr?: string;
21
+ popoverContentClassName?: string;
22
+ popoverContentOptions?: ComponentProps<typeof PopoverContent>;
23
+ buttonClassName?: string;
24
+ buttonOptions?: ComponentProps<typeof Button>;
25
+ placeholder?: string;
26
+ };
27
+
28
+ export function MultipleDatesPicker({
29
+ value,
30
+ onValueChange,
31
+ formatStr = "yyyy-MM-dd p",
32
+ popoverContentClassName,
33
+ popoverContentOptions,
34
+ buttonClassName,
35
+ buttonOptions,
36
+ placeholder = "Pick dates",
37
+ ...props
38
+ }: MultipleDatesPickerProps) {
39
+ return (
40
+ <Popover modal>
41
+ <PopoverTrigger asChild>
42
+ <Button
43
+ variant="outline"
44
+ className={cn(
45
+ "w-full px-3 text-start font-normal overflow-hidden",
46
+ buttonClassName,
47
+ )}
48
+ {...buttonOptions}
49
+ >
50
+ {value && value.length > 0 ? (
51
+ <div className="truncate me-1">
52
+ {value.map((date) => format(date, formatStr)).join(", ")}
53
+ </div>
54
+ ) : (
55
+ <span className="text-muted-foreground">{placeholder}</span>
56
+ )}
57
+ <CalendarIcon className="shrink-0 ms-auto h-4 w-4 text-muted-foreground" />
58
+ </Button>
59
+ </PopoverTrigger>
60
+ <PopoverContent
61
+ className={cn("w-auto p-0", popoverContentClassName)}
62
+ align="start"
63
+ {...popoverContentOptions}
64
+ >
65
+ <Calendar
66
+ mode="multiple"
67
+ selected={value}
68
+ onSelect={onValueChange}
69
+ {...props}
70
+ />
71
+ </PopoverContent>
72
+ </Popover>
73
+ );
74
+ }