@goplusvn/core 0.1.0 → 0.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (369) hide show
  1. package/package.json +2 -1
  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
@@ -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
+ }