@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,149 @@
1
+ import React, { memo } from "react";
2
+ import { cn } from "../../utils";
3
+
4
+ export type StatColorTheme =
5
+ | "dark"
6
+ | "green"
7
+ | "blue"
8
+ | "purple"
9
+ | "orange"
10
+ | "emerald"
11
+ | "rose"
12
+ | "primary"
13
+ | "destructive"
14
+ | "muted";
15
+
16
+ export interface CompactStatItem {
17
+ id: string;
18
+ label: string;
19
+ value: string | number;
20
+ icon: React.ElementType;
21
+ colorTheme?: StatColorTheme;
22
+ isHighlighted?: boolean;
23
+ }
24
+
25
+ export interface CompactStatBarProps {
26
+ items: CompactStatItem[];
27
+ className?: string;
28
+ }
29
+
30
+ const THEME_STYLES: Record<
31
+ StatColorTheme,
32
+ { wrapper: string; box: string; icon: string; label: string }
33
+ > = {
34
+ dark: {
35
+ wrapper: "hover:bg-[#181d26]/5 dark:hover:bg-slate-100/5",
36
+ box: "bg-[#181d26] group-hover:bg-[#0a2e0e] dark:bg-slate-800 dark:group-hover:bg-slate-700",
37
+ icon: "text-white",
38
+ label: "text-[#181d26] dark:text-slate-300",
39
+ },
40
+ green: {
41
+ wrapper: "hover:bg-[#006400]/10",
42
+ box: "bg-[#006400] group-hover:bg-[#39bf45]",
43
+ icon: "text-white",
44
+ label: "text-[#006400] dark:text-[#39bf45]",
45
+ },
46
+ blue: {
47
+ wrapper: "hover:bg-[#254fad]/10",
48
+ box: "bg-[#254fad] group-hover:bg-[#1d3d8f]",
49
+ icon: "text-white",
50
+ label: "text-[#254fad] dark:text-[#60a5fa]",
51
+ },
52
+ purple: {
53
+ wrapper: "hover:bg-[#7C3AED]/10",
54
+ box: "bg-[#7C3AED] group-hover:bg-[#6D28D9]",
55
+ icon: "text-white",
56
+ label: "text-[#7C3AED] dark:text-[#a78bfa]",
57
+ },
58
+ orange: {
59
+ wrapper: "hover:bg-[#EA580C]/10",
60
+ box: "bg-[#EA580C] group-hover:bg-[#C2410C]",
61
+ icon: "text-white",
62
+ label: "text-[#EA580C] dark:text-[#fb923c]",
63
+ },
64
+ emerald: {
65
+ wrapper: "hover:bg-[#059669]/10",
66
+ box: "bg-[#059669] group-hover:bg-[#047857]",
67
+ icon: "text-white",
68
+ label: "text-[#059669] dark:text-[#34d399]",
69
+ },
70
+ primary: {
71
+ wrapper: "hover:bg-primary/10",
72
+ box: "bg-primary group-hover:bg-primary/90",
73
+ icon: "text-primary-foreground",
74
+ label: "text-primary",
75
+ },
76
+ destructive: {
77
+ wrapper: "hover:bg-destructive/10",
78
+ box: "bg-destructive group-hover:bg-destructive/90",
79
+ icon: "text-destructive-foreground",
80
+ label: "text-destructive",
81
+ },
82
+ muted: {
83
+ wrapper: "hover:bg-muted/50",
84
+ box: "bg-muted group-hover:bg-muted-foreground/20",
85
+ icon: "text-muted-foreground",
86
+ label: "text-muted-foreground",
87
+ },
88
+ rose: {
89
+ wrapper: "hover:bg-[#E11D48]/10",
90
+ box: "bg-[#E11D48] group-hover:bg-[#BE123C]",
91
+ icon: "text-white",
92
+ label: "text-[#E11D48] dark:text-[#fb7185]",
93
+ },
94
+ };
95
+
96
+ export const CompactStatBar = memo(function CompactStatBar({
97
+ items,
98
+ className,
99
+ }: CompactStatBarProps) {
100
+ if (!items || items.length === 0) return null;
101
+
102
+ return (
103
+ <div className={cn("flex flex-col md:flex-row md:items-stretch gap-2 pb-1", className)}>
104
+ <div className="flex items-stretch overflow-x-auto no-scrollbar flex-1 bg-white dark:bg-[#1e293b] border border-slate-200 dark:border-slate-800 shadow-sm rounded-[10px] divide-x divide-slate-100 dark:divide-slate-800">
105
+ {items.map((item) => {
106
+ const theme = THEME_STYLES[item.colorTheme || "dark"];
107
+ const Icon = item.icon;
108
+
109
+ return (
110
+ <div
111
+ key={item.id}
112
+ className={cn(
113
+ "flex items-center gap-3 min-w-fit px-5 py-2 flex-1 transition-colors group cursor-default",
114
+ theme.wrapper
115
+ )}
116
+ >
117
+ <div
118
+ className={cn(
119
+ "p-1.5 rounded-md transition-colors shadow-sm shrink-0",
120
+ theme.box
121
+ )}
122
+ >
123
+ <Icon className={cn("w-3.5 h-3.5", theme.icon)} strokeWidth={2.5} />
124
+ </div>
125
+ <div className="flex flex-col">
126
+ <span
127
+ className={cn(
128
+ "text-[9px] font-bold uppercase tracking-wider mb-0.5",
129
+ item.isHighlighted ? "text-emerald-600 dark:text-emerald-400" : theme.label
130
+ )}
131
+ >
132
+ {item.label}
133
+ </span>
134
+ <span className={cn(
135
+ "font-black leading-none",
136
+ item.isHighlighted
137
+ ? "text-[22px] text-emerald-700 dark:text-emerald-400"
138
+ : "text-[17px] text-slate-900 dark:text-white"
139
+ )}>
140
+ {item.value}
141
+ </span>
142
+ </div>
143
+ </div>
144
+ );
145
+ })}
146
+ </div>
147
+ </div>
148
+ );
149
+ });
@@ -0,0 +1,255 @@
1
+ "use client";
2
+
3
+ import { createContext, useCallback, useContext, useState } from "react";
4
+ import type { ReactNode } from "react";
5
+
6
+ // ============================================================================
7
+ // Types
8
+ // ============================================================================
9
+
10
+ export interface SortingState {
11
+ field: string;
12
+ direction: "asc" | "desc";
13
+ }
14
+
15
+ export interface ActiveFilter {
16
+ name: string;
17
+ value: unknown;
18
+ operator: string;
19
+ }
20
+
21
+ export interface PaginationState {
22
+ page: number;
23
+ pageSize: number;
24
+ }
25
+
26
+ // ============================================================================
27
+ // DataTable Context - For standalone table usage
28
+ // ============================================================================
29
+
30
+ export interface DataTableContextValue {
31
+ // Pagination
32
+ pagination: PaginationState;
33
+ setPagination: (pagination: PaginationState) => void;
34
+ setPage: (page: number) => void;
35
+ setPageSize: (pageSize: number) => void;
36
+
37
+ // Sorting
38
+ sorting: SortingState | null;
39
+ setSorting: (sorting: SortingState | null) => void;
40
+
41
+ // Search
42
+ search: string;
43
+ setSearch: (search: string) => void;
44
+
45
+ // Filters
46
+ filters: ActiveFilter[];
47
+ setFilters: (filters: ActiveFilter[]) => void;
48
+ addFilter: (filter: ActiveFilter) => void;
49
+ removeFilter: (filterName: string) => void;
50
+ updateFilter: (filterName: string, value: unknown) => void;
51
+ clearFilters: () => void;
52
+
53
+ // Selection
54
+ selectedRows: Set<string>;
55
+ setSelectedRows: (rows: Set<string>) => void;
56
+ toggleRowSelection: (rowId: string) => void;
57
+ selectAllRows: (rowIds: string[]) => void;
58
+ clearSelection: () => void;
59
+ isAllSelected: boolean;
60
+ isPartiallySelected: boolean;
61
+ }
62
+
63
+ const DataTableContext = createContext<DataTableContextValue | undefined>(
64
+ undefined,
65
+ );
66
+
67
+ // ============================================================================
68
+ // Provider Props
69
+ // ============================================================================
70
+
71
+ export interface DataTableProviderProps {
72
+ children: ReactNode;
73
+ defaultPagination?: PaginationState;
74
+ defaultSorting?: SortingState | null;
75
+ defaultSearch?: string;
76
+ defaultFilters?: ActiveFilter[];
77
+ }
78
+
79
+ // ============================================================================
80
+ // DataTable Provider
81
+ // ============================================================================
82
+
83
+ export function DataTableProvider({
84
+ children,
85
+ defaultPagination = { page: 1, pageSize: 10 },
86
+ defaultSorting = null,
87
+ defaultSearch = "",
88
+ defaultFilters = [],
89
+ }: DataTableProviderProps) {
90
+ // Pagination state
91
+ const [pagination, setPaginationState] =
92
+ useState<PaginationState>(defaultPagination);
93
+
94
+ const setPagination = useCallback((newPagination: PaginationState) => {
95
+ setPaginationState(newPagination);
96
+ }, []);
97
+
98
+ const setPage = useCallback((page: number) => {
99
+ setPaginationState((prev) => ({ ...prev, page }));
100
+ }, []);
101
+
102
+ const setPageSize = useCallback((pageSize: number) => {
103
+ setPaginationState((prev) => ({ ...prev, pageSize, page: 1 }));
104
+ }, []);
105
+
106
+ // Sorting state
107
+ const [sorting, setSorting] = useState<SortingState | null>(defaultSorting);
108
+
109
+ // Search state
110
+ const [search, setSearch] = useState<string>(defaultSearch);
111
+
112
+ // Filters state
113
+ const [filters, setFiltersState] = useState<ActiveFilter[]>(defaultFilters);
114
+
115
+ const setFilters = useCallback(
116
+ (newFilters: ActiveFilter[]) => {
117
+ setFiltersState(newFilters);
118
+ setPage(1);
119
+ },
120
+ [setPage],
121
+ );
122
+
123
+ const addFilter = useCallback(
124
+ (filter: ActiveFilter) => {
125
+ setFiltersState((prev) => {
126
+ const existing = prev.findIndex((f) => f.name === filter.name);
127
+ if (existing >= 0) {
128
+ const updated = [...prev];
129
+ updated[existing] = filter;
130
+ return updated;
131
+ }
132
+ return [...prev, filter];
133
+ });
134
+ setPage(1);
135
+ },
136
+ [setPage],
137
+ );
138
+
139
+ const removeFilter = useCallback(
140
+ (filterName: string) => {
141
+ setFiltersState((prev) => prev.filter((f) => f.name !== filterName));
142
+ setPage(1);
143
+ },
144
+ [setPage],
145
+ );
146
+
147
+ const updateFilter = useCallback(
148
+ (filterName: string, value: unknown) => {
149
+ setFiltersState((prev) =>
150
+ prev.map((f) => (f.name === filterName ? { ...f, value } : f)),
151
+ );
152
+ setPage(1);
153
+ },
154
+ [setPage],
155
+ );
156
+
157
+ const clearFilters = useCallback(() => {
158
+ setFiltersState([]);
159
+ setPage(1);
160
+ }, [setPage]);
161
+
162
+ // Selection state
163
+ const [selectedRows, setSelectedRowsState] = useState<Set<string>>(new Set());
164
+ const [allRowIds, setAllRowIds] = useState<string[]>([]);
165
+
166
+ const setSelectedRows = useCallback((rows: Set<string>) => {
167
+ setSelectedRowsState(rows);
168
+ }, []);
169
+
170
+ const toggleRowSelection = useCallback((rowId: string) => {
171
+ setSelectedRowsState((prev) => {
172
+ const next = new Set(prev);
173
+ if (next.has(rowId)) {
174
+ next.delete(rowId);
175
+ } else {
176
+ next.add(rowId);
177
+ }
178
+ return next;
179
+ });
180
+ }, []);
181
+
182
+ const selectAllRows = useCallback((rowIds: string[]) => {
183
+ setAllRowIds(rowIds);
184
+ setSelectedRowsState((prev) => {
185
+ if (prev.size === rowIds.length) {
186
+ return new Set();
187
+ }
188
+ return new Set(rowIds);
189
+ });
190
+ }, []);
191
+
192
+ const clearSelection = useCallback(() => {
193
+ setSelectedRowsState(new Set());
194
+ }, []);
195
+
196
+ const isAllSelected =
197
+ allRowIds.length > 0 && selectedRows.size === allRowIds.length;
198
+ const isPartiallySelected =
199
+ selectedRows.size > 0 && selectedRows.size < allRowIds.length;
200
+
201
+ const value: DataTableContextValue = {
202
+ // Pagination
203
+ pagination,
204
+ setPagination,
205
+ setPage,
206
+ setPageSize,
207
+ // Sorting
208
+ sorting,
209
+ setSorting,
210
+ // Search
211
+ search,
212
+ setSearch,
213
+ // Filters
214
+ filters,
215
+ setFilters,
216
+ addFilter,
217
+ removeFilter,
218
+ updateFilter,
219
+ clearFilters,
220
+ // Selection
221
+ selectedRows,
222
+ setSelectedRows,
223
+ toggleRowSelection,
224
+ selectAllRows,
225
+ clearSelection,
226
+ isAllSelected,
227
+ isPartiallySelected,
228
+ };
229
+
230
+ return (
231
+ <DataTableContext.Provider value={value}>
232
+ {children}
233
+ </DataTableContext.Provider>
234
+ );
235
+ }
236
+
237
+ // ============================================================================
238
+ // Hooks
239
+ // ============================================================================
240
+
241
+ export function useDataTable() {
242
+ const context = useContext(DataTableContext);
243
+ if (context === undefined) {
244
+ throw new Error("useDataTable must be used within DataTableProvider");
245
+ }
246
+ return context;
247
+ }
248
+
249
+ /**
250
+ * Optional hook that returns undefined if not in provider
251
+ * Useful for components that can work both with and without context
252
+ */
253
+ export function useDataTableOptional() {
254
+ return useContext(DataTableContext);
255
+ }
@@ -0,0 +1,133 @@
1
+ "use client";
2
+
3
+ import { Database, FileSearch, SearchX } from "lucide-react";
4
+ import type { ReactNode } from "react";
5
+ import { Button } from "../../primitives";
6
+
7
+ export interface DataTableEmptyStateProps {
8
+ /**
9
+ * Icon to display
10
+ */
11
+ icon?: ReactNode;
12
+ /**
13
+ * Title text
14
+ */
15
+ title?: string;
16
+ /**
17
+ * Description text
18
+ */
19
+ description?: string;
20
+ /**
21
+ * Whether search is active
22
+ */
23
+ hasSearch?: boolean;
24
+ /**
25
+ * Whether filters are active
26
+ */
27
+ hasFilters?: boolean;
28
+ /**
29
+ * Clear search callback
30
+ */
31
+ onClearSearch?: () => void;
32
+ /**
33
+ * Clear filters callback
34
+ */
35
+ onClearFilters?: () => void;
36
+ /**
37
+ * Primary action button
38
+ */
39
+ primaryAction?: {
40
+ label: string;
41
+ onClick: () => void;
42
+ icon?: ReactNode;
43
+ };
44
+ /**
45
+ * Custom content to render
46
+ */
47
+ children?: ReactNode;
48
+ }
49
+
50
+ /**
51
+ * Empty state component for DataTable
52
+ * Shows different content based on whether search/filters are active
53
+ */
54
+ export function DataTableEmptyState({
55
+ icon,
56
+ title,
57
+ description,
58
+ hasSearch = false,
59
+ hasFilters = false,
60
+ onClearSearch,
61
+ onClearFilters,
62
+ primaryAction,
63
+ children,
64
+ }: DataTableEmptyStateProps) {
65
+ // Determine the appropriate empty state based on context
66
+ const hasActiveFiltersOrSearch = hasSearch || hasFilters;
67
+
68
+ // Default icon based on state
69
+ const defaultIcon = hasSearch ? (
70
+ <SearchX className="h-12 w-12 text-muted-foreground/50" />
71
+ ) : hasFilters ? (
72
+ <FileSearch className="h-12 w-12 text-muted-foreground/50" />
73
+ ) : (
74
+ <Database className="h-12 w-12 text-muted-foreground/50" />
75
+ );
76
+
77
+ // Default title based on state
78
+ const defaultTitle = hasSearch
79
+ ? "Không tìm thấy kết quả"
80
+ : hasFilters
81
+ ? "Không có dữ liệu phù hợp"
82
+ : "Chưa có dữ liệu";
83
+
84
+ // Default description based on state
85
+ const defaultDescription = hasSearch
86
+ ? "Không có kết quả nào khớp với từ khóa tìm kiếm. Hãy thử với từ khóa khác."
87
+ : hasFilters
88
+ ? "Không có dữ liệu nào phù hợp với bộ lọc hiện tại."
89
+ : "Bắt đầu bằng cách thêm mục đầu tiên.";
90
+
91
+ return (
92
+ <div className="flex flex-col items-center justify-center h-full min-h-[300px] p-8 text-center">
93
+ {/* Icon */}
94
+ <div className="mb-4">{icon || defaultIcon}</div>
95
+
96
+ {/* Title */}
97
+ <h3 className="text-lg font-semibold text-foreground mb-2">
98
+ {title || defaultTitle}
99
+ </h3>
100
+
101
+ {/* Description */}
102
+ <p className="text-sm text-muted-foreground max-w-md mb-6">
103
+ {description || defaultDescription}
104
+ </p>
105
+
106
+ {/* Action buttons */}
107
+ <div className="flex flex-wrap items-center justify-center gap-3">
108
+ {hasSearch && onClearSearch && (
109
+ <Button variant="outline" size="sm" onClick={onClearSearch}>
110
+ <SearchX className="h-4 w-4 mr-2" />
111
+ Xóa tìm kiếm
112
+ </Button>
113
+ )}
114
+
115
+ {hasFilters && onClearFilters && (
116
+ <Button variant="outline" size="sm" onClick={onClearFilters}>
117
+ Xóa bộ lọc
118
+ </Button>
119
+ )}
120
+
121
+ {primaryAction && (
122
+ <Button onClick={primaryAction.onClick}>
123
+ {primaryAction.icon}
124
+ {primaryAction.label}
125
+ </Button>
126
+ )}
127
+ </div>
128
+
129
+ {/* Custom content */}
130
+ {children}
131
+ </div>
132
+ );
133
+ }
@@ -0,0 +1,145 @@
1
+ "use client";
2
+
3
+ import { Skeleton } from "../../primitives";
4
+ import { TableCell, TableRow } from "../../primitives";
5
+
6
+ export interface DataTableSkeletonProps {
7
+ /**
8
+ * Number of columns to render
9
+ */
10
+ columns: number;
11
+ /**
12
+ * Number of rows to render
13
+ * @default 5
14
+ */
15
+ rows?: number;
16
+ /**
17
+ * Height of each skeleton cell
18
+ * @default "h-5"
19
+ */
20
+ cellHeight?: string;
21
+ /**
22
+ * Show checkbox column skeleton
23
+ * @default false
24
+ */
25
+ showCheckbox?: boolean;
26
+ /**
27
+ * Show row number column skeleton
28
+ * @default false
29
+ */
30
+ showRowNumber?: boolean;
31
+ }
32
+
33
+ /**
34
+ * Skeleton loading state for DataTable
35
+ * Renders loading placeholder rows while data is being fetched
36
+ */
37
+ export function DataTableSkeleton({
38
+ columns,
39
+ rows = 5,
40
+ cellHeight = "h-5",
41
+ showCheckbox = false,
42
+ showRowNumber = false,
43
+ }: DataTableSkeletonProps) {
44
+ const _totalColumns =
45
+ columns + (showCheckbox ? 1 : 0) + (showRowNumber ? 1 : 0);
46
+
47
+ return (
48
+ <>
49
+ {Array.from({ length: rows }).map((_, rowIndex) => (
50
+ <TableRow
51
+ key={`skeleton-row-${rowIndex}`}
52
+ className="animate-pulse"
53
+ aria-hidden="true"
54
+ >
55
+ {/* Checkbox skeleton */}
56
+ {showCheckbox && (
57
+ <TableCell key={`skeleton-checkbox-${rowIndex}`} className="w-12">
58
+ <Skeleton className="h-4 w-4 rounded" />
59
+ </TableCell>
60
+ )}
61
+
62
+ {/* Row number skeleton */}
63
+ {showRowNumber && (
64
+ <TableCell
65
+ key={`skeleton-stt-${rowIndex}`}
66
+ className="w-16 text-center"
67
+ >
68
+ <Skeleton className="h-4 w-8 mx-auto" />
69
+ </TableCell>
70
+ )}
71
+
72
+ {/* Data columns skeleton */}
73
+ {Array.from({ length: columns }).map((_, cellIndex) => (
74
+ <TableCell key={`skeleton-cell-${rowIndex}-${cellIndex}`}>
75
+ <Skeleton className={`${cellHeight} w-full`} />
76
+ </TableCell>
77
+ ))}
78
+ </TableRow>
79
+ ))}
80
+ </>
81
+ );
82
+ }
83
+
84
+ /**
85
+ * Full table skeleton including header
86
+ */
87
+ export function DataTableFullSkeleton({
88
+ columns,
89
+ rows = 5,
90
+ showCheckbox = false,
91
+ showRowNumber = false,
92
+ }: DataTableSkeletonProps) {
93
+ const _totalColumns =
94
+ columns + (showCheckbox ? 1 : 0) + (showRowNumber ? 1 : 0);
95
+
96
+ return (
97
+ <div className="w-full border rounded-md overflow-hidden animate-pulse">
98
+ {/* Header skeleton */}
99
+ <div className="flex bg-muted/50 border-b">
100
+ {showCheckbox && (
101
+ <div className="w-12 p-3">
102
+ <Skeleton className="h-4 w-4 rounded" />
103
+ </div>
104
+ )}
105
+ {showRowNumber && (
106
+ <div className="w-16 p-3 text-center">
107
+ <Skeleton className="h-4 w-10 mx-auto" />
108
+ </div>
109
+ )}
110
+ {Array.from({ length: columns }).map((_, i) => (
111
+ <div key={`header-${i}`} className="flex-1 p-3">
112
+ <Skeleton className="h-4 w-24" />
113
+ </div>
114
+ ))}
115
+ </div>
116
+
117
+ {/* Body skeleton */}
118
+ {Array.from({ length: rows }).map((_, rowIndex) => (
119
+ <div
120
+ key={`body-row-${rowIndex}`}
121
+ className="flex border-b last:border-b-0"
122
+ >
123
+ {showCheckbox && (
124
+ <div className="w-12 p-3">
125
+ <Skeleton className="h-4 w-4 rounded" />
126
+ </div>
127
+ )}
128
+ {showRowNumber && (
129
+ <div className="w-16 p-3 text-center">
130
+ <Skeleton className="h-4 w-8 mx-auto" />
131
+ </div>
132
+ )}
133
+ {Array.from({ length: columns }).map((_, cellIndex) => (
134
+ <div
135
+ key={`body-cell-${rowIndex}-${cellIndex}`}
136
+ className="flex-1 p-3"
137
+ >
138
+ <Skeleton className="h-5 w-full" />
139
+ </div>
140
+ ))}
141
+ </div>
142
+ ))}
143
+ </div>
144
+ );
145
+ }