@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,412 @@
1
+ "use client";
2
+
3
+ import React from "react";
4
+ import { format } from "date-fns";
5
+ import { vi } from "date-fns/locale"; // Note: This might need i18n handling if we support EN
6
+ import {
7
+ CheckCircle2,
8
+ Circle,
9
+ Clock,
10
+ XCircle,
11
+ ArrowRight,
12
+ ArrowDown,
13
+ } from "lucide-react";
14
+ import {
15
+ Timeline,
16
+ TimelineContent,
17
+ TimelineDot,
18
+ TimelineHeading,
19
+ TimelineItem,
20
+ TimelineLine,
21
+ Badge,
22
+ AvatarStack,
23
+ DynamicIcon,
24
+ } from "../ui";
25
+ import { cn } from "../utils";
26
+ import type { DynamicIconNameType } from "../types";
27
+
28
+ export interface ActivityItem {
29
+ id: string;
30
+ title: string;
31
+ description?: string;
32
+ date?: Date | string;
33
+ status?: "completed" | "current" | "upcoming" | "error";
34
+ badgeText?: string; // Text to show in the badge (e.g. "Opened", "Resolved")
35
+ iconName?: DynamicIconNameType;
36
+ fill?: string;
37
+ assignedMembers?: Array<{
38
+ name: string;
39
+ avatar?: string;
40
+ href?: string;
41
+ }>;
42
+ }
43
+
44
+ interface SharedActivityTimelineProps {
45
+ items: ActivityItem[];
46
+ className?: string;
47
+ orientation?: "vertical" | "horizontal";
48
+ }
49
+
50
+ function RenderAvatarStack({
51
+ assignedMembers,
52
+ }: {
53
+ assignedMembers: ActivityItem["assignedMembers"];
54
+ }) {
55
+ if (assignedMembers && assignedMembers.length > 0) {
56
+ return (
57
+ <AvatarStack
58
+ avatars={assignedMembers.map((member) => ({
59
+ src: member.avatar,
60
+ alt: member.name,
61
+ href: member.href || "#",
62
+ }))}
63
+ size="sm"
64
+ className="mt-2"
65
+ />
66
+ );
67
+ }
68
+
69
+ return null;
70
+ }
71
+
72
+ // Horizontal Step Component - Clear distinction: completed (active), current (highlighted), upcoming (inactive)
73
+ function HorizontalStep({
74
+ item,
75
+ isLast,
76
+ stepNumber,
77
+ }: {
78
+ item: ActivityItem;
79
+ isLast: boolean;
80
+ stepNumber: number;
81
+ }) {
82
+ const getStepIcon = () => {
83
+ if (item.status === "completed") {
84
+ return (
85
+ <CheckCircle2 className="h-4 w-4 text-green-600 dark:text-green-400" />
86
+ );
87
+ }
88
+ if (item.status === "error") {
89
+ return <XCircle className="h-4 w-4 text-red-600 dark:text-red-400" />;
90
+ }
91
+ if (item.status === "current") {
92
+ return <Clock className="h-4 w-4 text-blue-600 dark:text-blue-400" />;
93
+ }
94
+ return <Circle className="h-4 w-4 text-muted-foreground opacity-50" />;
95
+ };
96
+
97
+ const isCurrent = item.status === "current";
98
+ const isCompleted = item.status === "completed";
99
+ const isUpcoming = item.status === "upcoming" || !item.status;
100
+
101
+ return (
102
+ <div className="flex items-center gap-2 flex-1 relative min-w-0 shrink-0">
103
+ {/* Step Wrapper - Only wrap content, not arrow */}
104
+ <div
105
+ className={cn(
106
+ "flex items-center gap-1.5 flex-1 relative z-10 transition-all shrink-0",
107
+ isCurrent &&
108
+ "bg-blue-50 dark:bg-blue-950/30 border-2 border-blue-500 dark:border-blue-400 rounded-md px-1.5 py-1 shadow-sm",
109
+ )}
110
+ >
111
+ {/* Step Number */}
112
+ <div
113
+ className={cn(
114
+ "shrink-0 w-4 h-4 rounded-full flex items-center justify-center text-[10px] font-bold transition-all",
115
+ isCompleted && "bg-green-500 dark:bg-green-400 text-white",
116
+ isCurrent && "bg-blue-500 dark:bg-blue-400 text-white",
117
+ isUpcoming && "bg-muted text-muted-foreground opacity-50",
118
+ )}
119
+ >
120
+ {stepNumber}
121
+ </div>
122
+
123
+ {/* Step Icon */}
124
+ <div
125
+ className={cn(
126
+ "shrink-0 transition-all",
127
+ isCompleted && "text-green-600 dark:text-green-400",
128
+ isCurrent && "text-blue-600 dark:text-blue-400",
129
+ isUpcoming && "text-muted-foreground opacity-50",
130
+ )}
131
+ >
132
+ {item.iconName ? (
133
+ <div
134
+ className={cn(
135
+ "rounded-full p-0.5 transition-all",
136
+ isCurrent && "scale-105",
137
+ )}
138
+ style={
139
+ item.fill
140
+ ? { backgroundColor: item.fill, color: "white" }
141
+ : undefined
142
+ }
143
+ >
144
+ <DynamicIcon name={item.iconName} className="size-3" />
145
+ </div>
146
+ ) : (
147
+ <div className="scale-90">{getStepIcon()}</div>
148
+ )}
149
+ </div>
150
+
151
+ {/* Step Content - Horizontal layout */}
152
+ <div className="flex items-center gap-1.5 min-w-0 flex-1">
153
+ <div className="min-w-0 flex-1">
154
+ <div className="flex items-center gap-1.5 flex-wrap">
155
+ <p
156
+ className={cn(
157
+ "text-xs font-medium whitespace-nowrap transition-all leading-tight",
158
+ isCompleted &&
159
+ "text-green-700 dark:text-green-300 font-semibold",
160
+ isCurrent && "text-blue-700 dark:text-blue-300 font-semibold",
161
+ isUpcoming && "text-muted-foreground opacity-60",
162
+ )}
163
+ >
164
+ {item.title}
165
+ </p>
166
+ {item.badgeText && (
167
+ <Badge
168
+ variant={
169
+ isCurrent ? "default" : isCompleted ? "outline" : "outline"
170
+ }
171
+ className={cn(
172
+ "text-[10px] px-1 py-0 h-3.5 transition-all leading-none",
173
+ isCompleted &&
174
+ "border-green-500 dark:border-green-400 text-green-700 dark:text-green-300",
175
+ isCurrent &&
176
+ "bg-blue-600 dark:bg-blue-500 text-white border-blue-600 dark:border-blue-500",
177
+ isUpcoming &&
178
+ "border-muted-foreground/30 text-muted-foreground opacity-50",
179
+ )}
180
+ >
181
+ {item.badgeText}
182
+ </Badge>
183
+ )}
184
+ </div>
185
+ {item.description && (
186
+ <p
187
+ className={cn(
188
+ "text-[10px] mt-0 line-clamp-1 transition-colors leading-tight",
189
+ isCompleted && "text-green-600/80 dark:text-green-400/80",
190
+ isCurrent && "text-blue-600/80 dark:text-blue-400/80",
191
+ isUpcoming && "text-muted-foreground opacity-50",
192
+ )}
193
+ >
194
+ {item.description}
195
+ </p>
196
+ )}
197
+ </div>
198
+ </div>
199
+ </div>
200
+ </div>
201
+ );
202
+ }
203
+
204
+ export function SharedActivityTimeline({
205
+ items,
206
+ className,
207
+ orientation = "vertical",
208
+ }: SharedActivityTimelineProps) {
209
+ // Horizontal layout - Compact
210
+ if (orientation === "horizontal") {
211
+ return (
212
+ <div className={cn("flex items-center gap-0.5", className)}>
213
+ {items.map((item, index) => (
214
+ <React.Fragment key={item.id}>
215
+ <HorizontalStep
216
+ item={item}
217
+ isLast={index === items.length - 1}
218
+ stepNumber={index + 1}
219
+ />
220
+ {/* Arrow between steps */}
221
+ {index < items.length - 1 && (
222
+ <div className="flex items-center justify-center shrink-0 w-3">
223
+ <ArrowRight
224
+ className={cn(
225
+ "h-3.5 w-3.5 transition-colors",
226
+ item.status === "completed"
227
+ ? "text-green-600 dark:text-green-400"
228
+ : "text-muted-foreground opacity-50",
229
+ )}
230
+ strokeWidth={2.5}
231
+ />
232
+ </div>
233
+ )}
234
+ </React.Fragment>
235
+ ))}
236
+ </div>
237
+ );
238
+ }
239
+
240
+ // Vertical layout (default)
241
+ return (
242
+ <Timeline className={cn("pl-2", className)}>
243
+ {items.map((item, index) => {
244
+ const isLast = index === items.length - 1;
245
+
246
+ // Map status to TimelineDot props
247
+ let dotStatus: "done" | "current" | "error" | undefined = undefined;
248
+ let dotClass = "";
249
+
250
+ if (item.status === "completed") {
251
+ dotStatus = "done";
252
+ } else if (item.status === "error") {
253
+ dotStatus = "error";
254
+ } else if (item.status === "current") {
255
+ dotStatus = "current";
256
+ } else {
257
+ // upcoming
258
+ dotClass =
259
+ "bg-muted text-muted-foreground border-muted-foreground/30";
260
+ }
261
+
262
+ const isCurrent = item.status === "current";
263
+ const isCompleted = item.status === "completed";
264
+ const isUpcoming = item.status === "upcoming" || !item.status;
265
+
266
+ return (
267
+ <TimelineItem
268
+ key={item.id}
269
+ status={item.status === "completed" ? "done" : "default"}
270
+ className={cn(
271
+ isCurrent &&
272
+ "bg-blue-50 dark:bg-blue-950/30 border-2 border-blue-500 dark:border-blue-400 rounded-lg px-3 py-2 -mx-1 shadow-sm",
273
+ )}
274
+ >
275
+ <div className="flex items-center gap-1.5">
276
+ {/* Step Number */}
277
+ <div
278
+ className={cn(
279
+ "shrink-0 w-5 h-5 rounded-full flex items-center justify-center text-[10px] font-bold transition-all",
280
+ isCompleted && "bg-green-500 dark:bg-green-400 text-white",
281
+ isCurrent && "bg-blue-500 dark:bg-blue-400 text-white",
282
+ isUpcoming && "bg-muted text-muted-foreground opacity-50",
283
+ )}
284
+ >
285
+ {index + 1}
286
+ </div>
287
+ <TimelineHeading
288
+ className={cn(
289
+ "text-sm font-medium transition-all flex-1 leading-tight",
290
+ isCompleted &&
291
+ "text-green-700 dark:text-green-300 font-semibold",
292
+ isCurrent && "text-blue-700 dark:text-blue-300 font-semibold",
293
+ isUpcoming && "text-muted-foreground opacity-60",
294
+ )}
295
+ >
296
+ {item.title}
297
+ </TimelineHeading>
298
+ </div>
299
+ {item.iconName ? (
300
+ <TimelineDot
301
+ customIconName={item.iconName}
302
+ customStatusName={item.status || "default"}
303
+ className={cn(
304
+ dotClass,
305
+ isCompleted &&
306
+ "bg-green-100 dark:bg-green-900/30 border-green-500 dark:border-green-400",
307
+ isCurrent &&
308
+ "ring-2 ring-blue-400 dark:ring-blue-500 ring-offset-2 ring-offset-background",
309
+ isUpcoming && "opacity-50",
310
+ )}
311
+ style={
312
+ item.fill
313
+ ? { backgroundColor: item.fill, color: "white" }
314
+ : undefined
315
+ }
316
+ iconClassName={cn(
317
+ "size-4 text-background p-0.5 transition-all",
318
+ isCurrent && "scale-105",
319
+ )}
320
+ />
321
+ ) : (
322
+ <TimelineDot
323
+ status={dotStatus}
324
+ className={cn(
325
+ dotClass,
326
+ isCompleted &&
327
+ "bg-green-100 dark:bg-green-900/30 border-green-500 dark:border-green-400",
328
+ isCurrent &&
329
+ "ring-2 ring-blue-400 dark:ring-blue-500 ring-offset-2 ring-offset-background",
330
+ isUpcoming && "opacity-50",
331
+ )}
332
+ />
333
+ )}
334
+ {/* Arrow outside of wrapper */}
335
+ {!isLast && (
336
+ <div className="relative col-start-2 col-end-3 z-20">
337
+ <TimelineLine
338
+ done={isCompleted}
339
+ className={cn(!isCompleted && "opacity-30")}
340
+ />
341
+ <ArrowDown
342
+ className={cn(
343
+ "absolute left-1/2 -translate-x-1/2 bottom-0 translate-y-1/2 h-3.5 w-3.5 z-20 transition-colors bg-background rounded-full p-0.5",
344
+ isCompleted
345
+ ? "text-green-600 dark:text-green-400"
346
+ : "text-muted-foreground opacity-50",
347
+ )}
348
+ strokeWidth={2.5}
349
+ />
350
+ </div>
351
+ )}
352
+ <TimelineContent className={cn("space-y-1 pb-2 transition-colors")}>
353
+ <div
354
+ className={cn(
355
+ "flex items-center text-sm transition-colors",
356
+ isCompleted && "text-green-600/80 dark:text-green-400/80",
357
+ isCurrent && "text-blue-600/80 dark:text-blue-400/80",
358
+ isUpcoming && "text-muted-foreground opacity-50",
359
+ )}
360
+ >
361
+ {item.date && (
362
+ <span>
363
+ {typeof item.date === "string"
364
+ ? item.date
365
+ : format(new Date(item.date), "dd/MM/yyyy HH:mm", {
366
+ locale: vi,
367
+ })}
368
+ </span>
369
+ )}
370
+ {item.badgeText && (
371
+ <Badge
372
+ variant={
373
+ isCurrent
374
+ ? "default"
375
+ : isCompleted
376
+ ? "outline"
377
+ : "outline"
378
+ }
379
+ className={cn(
380
+ "ms-1.5 text-[10px] px-1 py-0 h-3.5 transition-all leading-none",
381
+ isCompleted &&
382
+ "border-green-500 dark:border-green-400 text-green-700 dark:text-green-300",
383
+ isCurrent &&
384
+ "bg-blue-600 dark:bg-blue-500 text-white border-blue-600 dark:border-blue-500",
385
+ isUpcoming &&
386
+ "border-muted-foreground/30 text-muted-foreground opacity-50",
387
+ )}
388
+ >
389
+ {item.badgeText}
390
+ </Badge>
391
+ )}
392
+ </div>
393
+ {item.description && (
394
+ <p
395
+ className={cn(
396
+ "text-sm transition-colors leading-tight",
397
+ isCompleted && "text-green-600/80 dark:text-green-400/80",
398
+ isCurrent && "text-blue-600/80 dark:text-blue-400/80",
399
+ isUpcoming && "text-muted-foreground opacity-50",
400
+ )}
401
+ >
402
+ {item.description}
403
+ </p>
404
+ )}
405
+ <RenderAvatarStack assignedMembers={item.assignedMembers} />
406
+ </TimelineContent>
407
+ </TimelineItem>
408
+ );
409
+ })}
410
+ </Timeline>
411
+ );
412
+ }
@@ -0,0 +1,31 @@
1
+ "use client";
2
+
3
+ import React from "react";
4
+
5
+ import { cn } from "../utils";
6
+ import { SharedActivityTimeline } from "./activity-timeline";
7
+ import type { ActivityItem } from "./activity-timeline";
8
+
9
+ export type ApprovalStepStatus = ActivityItem["status"];
10
+
11
+ export interface ApprovalStep extends ActivityItem {
12
+ status?: ApprovalStepStatus;
13
+ }
14
+
15
+ interface ApprovalWorkflowProps {
16
+ steps: ApprovalStep[];
17
+ orientation?: "horizontal" | "vertical";
18
+ className?: string;
19
+ }
20
+
21
+ export function ApprovalWorkflow({
22
+ steps,
23
+ orientation = "horizontal",
24
+ className,
25
+ }: ApprovalWorkflowProps) {
26
+ return (
27
+ <div className={cn("w-full", className)}>
28
+ <SharedActivityTimeline items={steps} orientation={orientation} />
29
+ </div>
30
+ );
31
+ }
@@ -0,0 +1,2 @@
1
+ export * from "./activity-timeline";
2
+ export * from "./approval-workflow";