@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,597 @@
1
+ "use client";
2
+
3
+ import { memo, useCallback, useEffect, useMemo } from "react";
4
+ import type { ReactNode, ReactElement } from "react";
5
+ import {
6
+ flexRender,
7
+ getCoreRowModel,
8
+ getFilteredRowModel,
9
+ getPaginationRowModel,
10
+ getSortedRowModel,
11
+ useReactTable,
12
+ } from "@tanstack/react-table";
13
+ import type {
14
+ ColumnDef,
15
+ SortingState as TanstackSortingState,
16
+ Table,
17
+ Row,
18
+ } from "@tanstack/react-table";
19
+
20
+ import { Checkbox } from "../../primitives/client";
21
+ import {
22
+ TableBody,
23
+ TableCell,
24
+ TableHead,
25
+ TableHeader,
26
+ TableRow,
27
+ } from "../../primitives";
28
+ import { DataTablePagination } from "../data-table-pagination";
29
+ import { DataTableSkeleton } from "./data-table-skeleton";
30
+ import { DataTableEmptyState } from "./data-table-empty-state";
31
+ import type { SortingState, PaginationState } from "./data-table-context";
32
+
33
+ // ============================================================================
34
+ // Types
35
+ // ============================================================================
36
+
37
+ export interface DataTablePaginationConfig {
38
+ page: number;
39
+ pageSize: number;
40
+ total: number;
41
+ }
42
+
43
+ export interface DataTableProps<TData> {
44
+ /**
45
+ * Data to display in the table
46
+ */
47
+ data: TData[];
48
+
49
+ /**
50
+ * Column definitions for the table
51
+ */
52
+ columns: ColumnDef<TData>[];
53
+
54
+ /**
55
+ * Pagination configuration (controlled)
56
+ */
57
+ pagination?: DataTablePaginationConfig;
58
+
59
+ /**
60
+ * Callback when pagination changes
61
+ */
62
+ onPaginationChange?: (pagination: PaginationState) => void;
63
+
64
+ /**
65
+ * Current sorting state (controlled)
66
+ */
67
+ sorting?: SortingState | null;
68
+
69
+ /**
70
+ * Callback when sorting changes
71
+ */
72
+ onSortingChange?: (sorting: SortingState | null) => void;
73
+
74
+ /**
75
+ * Enable row selection with checkboxes
76
+ * @default false
77
+ */
78
+ enableRowSelection?: boolean;
79
+
80
+ /**
81
+ * Currently selected row IDs
82
+ */
83
+ selectedRows?: Set<string>;
84
+
85
+ /**
86
+ * Callback when selection changes
87
+ */
88
+ onSelectionChange?: (selectedRows: Set<string>) => void;
89
+
90
+ /**
91
+ * Function to get unique ID from row data
92
+ * @default (row) => row.id
93
+ */
94
+ getRowId?: (row: TData) => string;
95
+
96
+ /**
97
+ * Show row numbers (STT column)
98
+ * @default false
99
+ */
100
+ enableRowNumber?: boolean;
101
+
102
+ /**
103
+ * Whether data is loading
104
+ * @default false
105
+ */
106
+ loading?: boolean;
107
+
108
+ /**
109
+ * Custom empty state component
110
+ */
111
+ emptyState?: ReactNode;
112
+
113
+ /**
114
+ * Empty state configuration
115
+ */
116
+ emptyStateConfig?: {
117
+ hasSearch?: boolean;
118
+ hasFilters?: boolean;
119
+ onClearSearch?: () => void;
120
+ onClearFilters?: () => void;
121
+ onCreate?: () => void;
122
+ createLabel?: string;
123
+ };
124
+
125
+ /**
126
+ * Callback when table instance is ready
127
+ */
128
+ onTableReady?: (table: Table<TData>) => void;
129
+
130
+ /**
131
+ * Custom row click handler
132
+ */
133
+ onRowClick?: (row: TData) => void;
134
+
135
+ /**
136
+ * Custom class name for the table container
137
+ */
138
+ className?: string;
139
+
140
+ /**
141
+ * Height configuration
142
+ * @default "auto"
143
+ */
144
+ height?: "auto" | "full" | string;
145
+ }
146
+
147
+ // ============================================================================
148
+ // DataTable Component
149
+ // ============================================================================
150
+
151
+ export function DataTable<TData extends Record<string, unknown>>({
152
+ data,
153
+ columns: userColumns,
154
+ pagination,
155
+ onPaginationChange,
156
+ sorting,
157
+ onSortingChange,
158
+ enableRowSelection = false,
159
+ selectedRows = new Set(),
160
+ onSelectionChange,
161
+ getRowId = (row) => String((row as Record<string, unknown>).id),
162
+ enableRowNumber = false,
163
+ loading = false,
164
+ emptyState,
165
+ emptyStateConfig,
166
+ onTableReady,
167
+ onRowClick,
168
+ className,
169
+ height = "auto",
170
+ }: DataTableProps<TData>) {
171
+ // Build final columns with selection and row number if enabled
172
+ const columns = useMemo<ColumnDef<TData>[]>(() => {
173
+ const cols: ColumnDef<TData>[] = [];
174
+
175
+ // Selection column
176
+ if (enableRowSelection) {
177
+ cols.push({
178
+ id: "select",
179
+ header: ({ table }) => (
180
+ <Checkbox
181
+ checked={
182
+ table.getIsAllPageRowsSelected() ||
183
+ (table.getIsSomePageRowsSelected() && "indeterminate")
184
+ }
185
+ onCheckedChange={(value) => {
186
+ const rowIds = table
187
+ .getRowModel()
188
+ .rows.map((row) => getRowId(row.original));
189
+
190
+ if (onSelectionChange) {
191
+ if (value) {
192
+ onSelectionChange(new Set(rowIds));
193
+ } else {
194
+ onSelectionChange(new Set());
195
+ }
196
+ }
197
+ table.toggleAllPageRowsSelected(!!value);
198
+ }}
199
+ aria-label="Select all"
200
+ />
201
+ ),
202
+ cell: ({ row }) => {
203
+ const rowId = getRowId(row.original);
204
+ return (
205
+ <Checkbox
206
+ checked={selectedRows.has(rowId)}
207
+ onCheckedChange={() => {
208
+ if (onSelectionChange) {
209
+ const newSelection = new Set(selectedRows);
210
+ if (newSelection.has(rowId)) {
211
+ newSelection.delete(rowId);
212
+ } else {
213
+ newSelection.add(rowId);
214
+ }
215
+ onSelectionChange(newSelection);
216
+ }
217
+ row.toggleSelected(!row.getIsSelected());
218
+ }}
219
+ aria-label="Select row"
220
+ />
221
+ );
222
+ },
223
+ enableSorting: false,
224
+ enableHiding: false,
225
+ size: 50,
226
+ });
227
+ }
228
+
229
+ // Row number (STT) column
230
+ if (enableRowNumber) {
231
+ cols.push({
232
+ id: "stt",
233
+ header: () => (
234
+ <div className="font-semibold text-center text-sm">STT</div>
235
+ ),
236
+ cell: ({ row }) => {
237
+ const rowIndex = row.index;
238
+ const currentPage = pagination?.page ?? 1;
239
+ const pageSize = pagination?.pageSize ?? 10;
240
+ const stt = (currentPage - 1) * pageSize + rowIndex + 1;
241
+ return <div className="text-center font-medium">{stt}</div>;
242
+ },
243
+ enableSorting: false,
244
+ enableHiding: false,
245
+ size: 70,
246
+ minSize: 60,
247
+ maxSize: 80,
248
+ });
249
+ }
250
+
251
+ // Add user columns
252
+ cols.push(...userColumns);
253
+
254
+ return cols;
255
+ }, [
256
+ userColumns,
257
+ enableRowSelection,
258
+ enableRowNumber,
259
+ selectedRows,
260
+ onSelectionChange,
261
+ getRowId,
262
+ pagination?.page,
263
+ pagination?.pageSize,
264
+ ]);
265
+
266
+ // Convert our sorting format to tanstack's format
267
+ const tanstackSorting = useMemo<TanstackSortingState>(() => {
268
+ if (!sorting) return [];
269
+ return [
270
+ {
271
+ id: sorting.field,
272
+ desc: sorting.direction === "desc",
273
+ },
274
+ ];
275
+ }, [sorting]);
276
+
277
+ // Handle sorting change
278
+ const handleSortingChange = useCallback(
279
+ (updater: unknown) => {
280
+ const newSorting =
281
+ typeof updater === "function" ? updater(tanstackSorting) : updater;
282
+
283
+ if (onSortingChange) {
284
+ if (Array.isArray(newSorting) && newSorting.length > 0) {
285
+ const sort = newSorting[0];
286
+ onSortingChange({
287
+ field: sort.id,
288
+ direction: sort.desc ? "desc" : "asc",
289
+ });
290
+ } else {
291
+ onSortingChange(null);
292
+ }
293
+ }
294
+ },
295
+ [tanstackSorting, onSortingChange],
296
+ );
297
+
298
+ // Handle pagination change
299
+ const handlePaginationChange = useCallback(
300
+ (updater: unknown) => {
301
+ if (!onPaginationChange || !pagination) return;
302
+
303
+ const currentPagination = {
304
+ pageIndex: pagination.page - 1,
305
+ pageSize: pagination.pageSize,
306
+ };
307
+ const newPagination =
308
+ typeof updater === "function" ? updater(currentPagination) : updater;
309
+
310
+ onPaginationChange({
311
+ page: newPagination.pageIndex + 1,
312
+ pageSize: newPagination.pageSize,
313
+ });
314
+ },
315
+ [pagination, onPaginationChange],
316
+ );
317
+
318
+ // Calculate page count
319
+ const pageCount = useMemo(() => {
320
+ if (!pagination) return 1;
321
+ const count = Math.ceil(pagination.total / pagination.pageSize);
322
+ return count > 0 ? count : 1;
323
+ }, [pagination]);
324
+
325
+ // Create table instance
326
+ const table = useReactTable({
327
+ data: Array.isArray(data) ? data : [],
328
+ columns,
329
+ getCoreRowModel: getCoreRowModel(),
330
+ getPaginationRowModel: getPaginationRowModel(),
331
+ onSortingChange: handleSortingChange,
332
+ getSortedRowModel: getSortedRowModel(),
333
+ getFilteredRowModel: getFilteredRowModel(),
334
+ state: {
335
+ sorting: tanstackSorting,
336
+ pagination: pagination
337
+ ? {
338
+ pageIndex: pagination.page - 1,
339
+ pageSize: pagination.pageSize,
340
+ }
341
+ : undefined,
342
+ },
343
+ onPaginationChange: handlePaginationChange,
344
+ manualPagination: !!pagination,
345
+ pageCount,
346
+ });
347
+
348
+ // Notify parent when table is ready
349
+ useEffect(() => {
350
+ if (onTableReady && table) {
351
+ onTableReady(table);
352
+ }
353
+ }, [table, onTableReady]);
354
+
355
+ // Get rows safely
356
+ let rows: Row<TData>[] = [];
357
+ try {
358
+ const rowModel = table.getRowModel();
359
+ if (rowModel && Array.isArray(rowModel.rows)) {
360
+ rows = rowModel.rows;
361
+ }
362
+ } catch (error) {
363
+ console.error("Error getting row model:", error);
364
+ }
365
+
366
+ // Height class
367
+ const heightClass =
368
+ height === "auto"
369
+ ? ""
370
+ : height === "full"
371
+ ? "min-h-[calc(100vh-16rem)]"
372
+ : height;
373
+
374
+ return (
375
+ <div
376
+ className={`flex flex-col border border-border dark:border-slate-800 rounded-md overflow-hidden bg-card ${heightClass} ${className || ""}`}
377
+ >
378
+ {/* Table Container - Scrollable */}
379
+ <div className="flex-1 overflow-auto relative">
380
+ <table className="w-max min-w-full caption-bottom text-xs relative">
381
+ <TableHeader className="sticky top-0 z-20 bg-card">
382
+ {table.getHeaderGroups().map((headerGroup) => (
383
+ <TableRow
384
+ key={headerGroup.id}
385
+ className="border-b-2 border-border dark:border-slate-700 hover:bg-transparent"
386
+ >
387
+ {headerGroup.headers.map((header) => (
388
+ <TableHead
389
+ key={header.id}
390
+ className="text-xs font-semibold text-foreground py-2.5"
391
+ style={{
392
+ width: header.getSize(),
393
+ minWidth: header.column.columnDef.minSize,
394
+ maxWidth: header.column.columnDef.maxSize,
395
+ }}
396
+ >
397
+ {header.isPlaceholder
398
+ ? null
399
+ : flexRender(
400
+ header.column.columnDef.header,
401
+ header.getContext(),
402
+ )}
403
+ </TableHead>
404
+ ))}
405
+ </TableRow>
406
+ ))}
407
+ </TableHeader>
408
+ <TableBody>
409
+ {loading ? (
410
+ <>
411
+ <tr aria-live="polite" aria-busy="true" className="sr-only">
412
+ <td colSpan={columns.length} className="sr-only">
413
+ Loading table data...
414
+ </td>
415
+ </tr>
416
+ <DataTableSkeleton
417
+ columns={userColumns.length}
418
+ rows={pagination?.pageSize ?? 10}
419
+ showCheckbox={enableRowSelection}
420
+ showRowNumber={enableRowNumber}
421
+ />
422
+ </>
423
+ ) : rows.length > 0 ? (
424
+ rows.map((row) => (
425
+ <MemoizedTableRow
426
+ key={row.id}
427
+ row={row}
428
+ isSelected={row.getIsSelected()}
429
+ visibleCellsCount={row.getVisibleCells().length}
430
+ onRowClick={onRowClick}
431
+ />
432
+ ))
433
+ ) : null}
434
+ </TableBody>
435
+ </table>
436
+ {/* Empty state - rendered outside table for proper viewport centering */}
437
+ {!loading && rows.length === 0 && (
438
+ <div className="flex items-center justify-center min-h-[300px]">
439
+ {emptyState || (
440
+ <DataTableEmptyState
441
+ hasSearch={emptyStateConfig?.hasSearch}
442
+ hasFilters={emptyStateConfig?.hasFilters}
443
+ onClearSearch={emptyStateConfig?.onClearSearch}
444
+ onClearFilters={emptyStateConfig?.onClearFilters}
445
+ primaryAction={
446
+ emptyStateConfig?.onCreate
447
+ ? {
448
+ label: emptyStateConfig.createLabel || "Thêm mới",
449
+ onClick: emptyStateConfig.onCreate,
450
+ }
451
+ : undefined
452
+ }
453
+ />
454
+ )}
455
+ </div>
456
+ )}
457
+ </div>
458
+
459
+ {/* Pagination Section */}
460
+ {pagination && (
461
+ <div className="border-t border-border dark:border-slate-800 bg-card px-4 sm:px-6 py-2 shrink-0">
462
+ <MemoizedPagination
463
+ table={table}
464
+ currentPage={pagination.page}
465
+ pageCount={pageCount}
466
+ pageSize={pagination.pageSize}
467
+ totalItems={pagination.total}
468
+ />
469
+ </div>
470
+ )}
471
+ </div>
472
+ );
473
+ }
474
+
475
+ // ============================================================================
476
+ // Memoized TableRow
477
+ // ============================================================================
478
+
479
+ interface MemoizedTableRowProps<TData> {
480
+ row: Row<TData>;
481
+ isSelected: boolean;
482
+ visibleCellsCount: number;
483
+ onRowClick?: (row: TData) => void;
484
+ }
485
+
486
+ // Ensure TData is maintained by making it a generic memo component, or casting it
487
+ const MemoizedTableRow = memo(
488
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
489
+ ({ row, isSelected, onRowClick }: MemoizedTableRowProps<any>) => {
490
+ return (
491
+ <TableRow
492
+ data-state={isSelected && "selected"}
493
+ className={`transition-colors duration-100 even:bg-muted/30 hover:bg-accent/50 dark:hover:bg-slate-800/40 data-[state=selected]:bg-primary/5 border-b border-border/40 dark:border-slate-800 relative hover:border-l-[3px] hover:border-l-primary ${
494
+ onRowClick ? "cursor-pointer" : ""
495
+ }`}
496
+ onClick={onRowClick ? () => onRowClick(row.original) : undefined}
497
+ >
498
+ {row.getVisibleCells().map((cell) => (
499
+ <TableCell
500
+ key={cell.id}
501
+ className=""
502
+ style={{
503
+ width: cell.column.getSize(),
504
+ minWidth: cell.column.columnDef.minSize,
505
+ maxWidth: cell.column.columnDef.maxSize,
506
+ }}
507
+ >
508
+ {flexRender(cell.column.columnDef.cell, cell.getContext())}
509
+ </TableCell>
510
+ ))}
511
+ </TableRow>
512
+ );
513
+ },
514
+ (prevProps, nextProps) => {
515
+ // Basic checks
516
+ if (
517
+ prevProps.row.id !== nextProps.row.id ||
518
+ prevProps.isSelected !== nextProps.isSelected ||
519
+ prevProps.onRowClick !== nextProps.onRowClick ||
520
+ prevProps.visibleCellsCount !== nextProps.visibleCellsCount ||
521
+ prevProps.row.original !== nextProps.row.original
522
+ ) {
523
+ return false;
524
+ }
525
+
526
+ // Since sizing could change without cells count changing
527
+ // We check if cell sizes changed
528
+ const prevCells = prevProps.row.getVisibleCells();
529
+ const nextCells = nextProps.row.getVisibleCells();
530
+
531
+ for (let i = 0; i < prevCells.length; i++) {
532
+ if (
533
+ prevCells[i]?.column.getSize() !== nextCells[i]?.column.getSize() ||
534
+ prevCells[i]?.id !== nextCells[i]?.id ||
535
+ prevCells[i]?.column.id !== nextCells[i]?.column.id
536
+ ) {
537
+ return false;
538
+ }
539
+ }
540
+
541
+ return true;
542
+ },
543
+ ) as <TData>(props: MemoizedTableRowProps<TData>) => ReactElement;
544
+ // The type cast ensures the component retains the generic type TData
545
+ // Without this, the component type would default to `any` or `unknown`.
546
+ // Adding displayName to a typecast component can cause a ts error, so we cast it as any to add the display name.
547
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
548
+ (MemoizedTableRow as any).displayName = "MemoizedTableRow";
549
+
550
+ // ============================================================================
551
+ // Memoized Pagination
552
+ // ============================================================================
553
+
554
+ const MemoizedPagination = memo(
555
+ ({
556
+ table,
557
+ currentPage,
558
+ pageSize,
559
+ totalItems,
560
+ }: {
561
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
562
+ table: Table<any>;
563
+ currentPage: number;
564
+ pageCount: number;
565
+ pageSize: number;
566
+ totalItems?: number;
567
+ }) => {
568
+ return (
569
+ <DataTablePagination
570
+ table={table}
571
+ totalItems={totalItems}
572
+ currentPage={currentPage}
573
+ pageSize={pageSize}
574
+ />
575
+ );
576
+ },
577
+ (prevProps, nextProps) => {
578
+ const prevState = prevProps.table.getState().pagination;
579
+ const nextState = nextProps.table.getState().pagination;
580
+
581
+ return (
582
+ prevProps.currentPage === nextProps.currentPage &&
583
+ prevProps.pageCount === nextProps.pageCount &&
584
+ prevProps.pageSize === nextProps.pageSize &&
585
+ prevProps.totalItems === nextProps.totalItems &&
586
+ prevState.pageIndex === nextState.pageIndex &&
587
+ prevState.pageSize === nextState.pageSize
588
+ );
589
+ },
590
+ );
591
+ MemoizedPagination.displayName = "MemoizedPagination";
592
+
593
+ // ============================================================================
594
+ // Exports
595
+ // ============================================================================
596
+
597
+ export type { ColumnDef, Table, Row };
@@ -0,0 +1,44 @@
1
+ // @goerp/core/ui/data-display/data-table
2
+ // Reusable DataTable components for standalone table usage
3
+
4
+ // Context and hooks
5
+ export {
6
+ DataTableProvider,
7
+ useDataTable,
8
+ useDataTableOptional,
9
+ type DataTableContextValue,
10
+ type DataTableProviderProps,
11
+ type SortingState,
12
+ type ActiveFilter,
13
+ type PaginationState,
14
+ } from "./data-table-context";
15
+
16
+ // Main DataTable component
17
+ export {
18
+ DataTable,
19
+ type DataTableProps,
20
+ type DataTablePaginationConfig,
21
+ type ColumnDef,
22
+ type Table,
23
+ type Row,
24
+ } from "./data-table";
25
+
26
+ // Skeleton loading states
27
+ export {
28
+ DataTableSkeleton,
29
+ DataTableFullSkeleton,
30
+ type DataTableSkeletonProps,
31
+ } from "./data-table-skeleton";
32
+
33
+ // Empty state
34
+ export {
35
+ DataTableEmptyState,
36
+ type DataTableEmptyStateProps,
37
+ } from "./data-table-empty-state";
38
+
39
+ // Toolbar
40
+ export {
41
+ DataTableToolbar,
42
+ type DataTableToolbarProps,
43
+ type FilterConfig,
44
+ } from "./data-table-toolbar";
@@ -0,0 +1,75 @@
1
+ // @goerp/core/ui/data-display
2
+ // DataTableColumnHeader component for table column headers with sorting
3
+
4
+ "use client";
5
+
6
+ import { ArrowDown, ArrowDownUp, ArrowUp, EyeOff, X } from "lucide-react";
7
+
8
+ import type { Column } from "@tanstack/react-table";
9
+ import type { ComponentProps } from "react";
10
+
11
+ import { cn } from "../../utils";
12
+
13
+ import { Button } from "../primitives";
14
+ import {
15
+ DropdownMenu,
16
+ DropdownMenuContent,
17
+ DropdownMenuItem,
18
+ DropdownMenuSeparator,
19
+ DropdownMenuTrigger,
20
+ } from "../primitives/client";
21
+
22
+ interface DataTableColumnHeaderProps<TData, TValue>
23
+ extends ComponentProps<"div"> {
24
+ column: Column<TData, TValue>;
25
+ title: string;
26
+ className?: string;
27
+ }
28
+
29
+ export function DataTableColumnHeader<TData, TValue>({
30
+ column,
31
+ title,
32
+ className,
33
+ }: DataTableColumnHeaderProps<TData, TValue>) {
34
+ if (!column.getCanSort()) {
35
+ return <div className={cn(className)}>{title}</div>;
36
+ }
37
+
38
+ return (
39
+ <div className={cn("flex items-center space-x-2", className)}>
40
+ <DropdownMenu>
41
+ <DropdownMenuTrigger asChild>
42
+ <Button
43
+ variant="ghost"
44
+ size="sm"
45
+ className="-ml-3 h-8 data-[state=open]:bg-accent"
46
+ >
47
+ <span>{title}</span>
48
+ {column.getIsSorted() === "desc" ? (
49
+ <ArrowDown className="ml-2 h-4 w-4" />
50
+ ) : column.getIsSorted() === "asc" ? (
51
+ <ArrowUp className="ml-2 h-4 w-4" />
52
+ ) : (
53
+ <ArrowDownUp className="ml-2 h-4 w-4" />
54
+ )}
55
+ </Button>
56
+ </DropdownMenuTrigger>
57
+ <DropdownMenuContent align="start">
58
+ <DropdownMenuItem onClick={() => column.toggleSorting(false)}>
59
+ <ArrowUp className="mr-2 h-3.5 w-3.5 text-muted-foreground/70" />
60
+ Asc
61
+ </DropdownMenuItem>
62
+ <DropdownMenuItem onClick={() => column.toggleSorting(true)}>
63
+ <ArrowDown className="mr-2 h-3.5 w-3.5 text-muted-foreground/70" />
64
+ Desc
65
+ </DropdownMenuItem>
66
+ <DropdownMenuSeparator />
67
+ <DropdownMenuItem onClick={() => column.toggleVisibility(false)}>
68
+ <EyeOff className="mr-2 h-3.5 w-3.5 text-muted-foreground/70" />
69
+ Hide
70
+ </DropdownMenuItem>
71
+ </DropdownMenuContent>
72
+ </DropdownMenu>
73
+ </div>
74
+ );
75
+ }