@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,92 @@
1
+ "use client";
2
+
3
+ import { format } from "date-fns";
4
+ import { vi } from "date-fns/locale";
5
+ import { TrendingUp, Zap } from "lucide-react";
6
+ import { cn } from "../utils";
7
+
8
+ interface WelcomeCardProps {
9
+ userName?: string;
10
+ className?: string;
11
+ }
12
+
13
+ function getGreeting(hour: number): { text: string; emoji: string } {
14
+ if (hour >= 5 && hour < 12) {
15
+ return { text: "Chào buổi sáng", emoji: "🌅" };
16
+ } else if (hour >= 12 && hour < 18) {
17
+ return { text: "Chào buổi chiều", emoji: "☀️" };
18
+ } else {
19
+ return { text: "Chào buổi tối", emoji: "🌙" };
20
+ }
21
+ }
22
+
23
+ export function WelcomeCard({ userName = "Bạn", className }: WelcomeCardProps) {
24
+ const now = new Date();
25
+ const hour = now.getHours();
26
+ const greeting = getGreeting(hour);
27
+
28
+ const formattedDate = format(now, "EEEE, dd/MM/yyyy", { locale: vi });
29
+ const formattedTime = format(now, "HH:mm");
30
+
31
+ return (
32
+ <div
33
+ className={cn(
34
+ "w-full bg-gradient-to-r from-blue-900 to-primary rounded-xl overflow-hidden shadow-lg group relative animate-in fade-in slide-in-from-bottom-4 duration-500",
35
+ className,
36
+ )}
37
+ >
38
+ {/* Background decoration */}
39
+ <div
40
+ className="absolute inset-0 opacity-20 pointer-events-none"
41
+ style={{
42
+ backgroundImage:
43
+ "radial-gradient(circle at 80% 20%, white 0%, transparent 40%), radial-gradient(circle at 10% 80%, white 0%, transparent 30%)",
44
+ }}
45
+ />
46
+
47
+ <div className="flex flex-col md:flex-row items-center relative overflow-hidden">
48
+ {/* Content Left */}
49
+ <div className="p-8 flex-1 z-10 w-full">
50
+ <div className="flex items-center gap-2 mb-4">
51
+ <span className="bg-white/20 text-white text-xs font-bold px-2 py-1 rounded uppercase tracking-wider backdrop-blur-sm">
52
+ {formattedDate}
53
+ </span>
54
+ </div>
55
+ <h2 className="text-white text-2xl md:text-3xl font-bold mb-3">
56
+ {greeting.text}, {userName}
57
+ </h2>
58
+ <div className="text-blue-100 mb-6 max-w-lg">
59
+ Chúc bạn một ngày làm việc hiệu quả. Hệ thống đang hoạt động ổn
60
+ định.
61
+ </div>
62
+ <div className="flex gap-3">
63
+ <div className="bg-white/10 text-white px-5 py-2.5 rounded-lg text-sm font-bold shadow-sm backdrop-blur-sm border border-white/20 flex items-center gap-2">
64
+ <span className="w-2 h-2 rounded-full bg-green-400 animate-pulse" />
65
+ <span>Online • {formattedTime}</span>
66
+ </div>
67
+ </div>
68
+ </div>
69
+
70
+ {/* Visuals Right (Hidden on mobile) */}
71
+ <div className="hidden md:flex flex-1 justify-end items-end h-full p-8 z-10 self-stretch">
72
+ <div className="grid grid-cols-2 gap-3 opacity-90 transform translate-y-4">
73
+ <div className="bg-white/10 backdrop-blur-md border border-white/20 p-3 rounded-lg flex items-center gap-3">
74
+ <TrendingUp className="text-white w-6 h-6" />
75
+ <div>
76
+ <div className="h-1.5 w-12 bg-white/40 rounded mb-1" />
77
+ <div className="h-1.5 w-8 bg-white/20 rounded" />
78
+ </div>
79
+ </div>
80
+ <div className="bg-white/10 backdrop-blur-md border border-white/20 p-3 rounded-lg flex items-center gap-3">
81
+ <Zap className="text-white w-6 h-6" />
82
+ <div>
83
+ <div className="h-1.5 w-12 bg-white/40 rounded mb-1" />
84
+ <div className="h-1.5 w-8 bg-white/20 rounded" />
85
+ </div>
86
+ </div>
87
+ </div>
88
+ </div>
89
+ </div>
90
+ </div>
91
+ );
92
+ }
@@ -0,0 +1,258 @@
1
+ "use client";
2
+
3
+ import { useState, useCallback } from "react";
4
+ import {
5
+ DndContext,
6
+ closestCenter,
7
+ KeyboardSensor,
8
+ PointerSensor,
9
+ useSensor,
10
+ useSensors,
11
+ } from "@dnd-kit/core";
12
+ import type { DragEndEvent } from "@dnd-kit/core";
13
+ import {
14
+ arrayMove,
15
+ SortableContext,
16
+ sortableKeyboardCoordinates,
17
+ rectSortingStrategy,
18
+ } from "@dnd-kit/sortable";
19
+ import { motion, AnimatePresence } from "framer-motion";
20
+ import { Plus, Eye, EyeOff, RotateCcw } from "lucide-react";
21
+ import { cn } from "../utils";
22
+ import type { WidgetConfig } from "./types";
23
+ import { DEFAULT_WIDGETS, WIDGET_LABELS } from "./types";
24
+ import { RevenueWidget } from "./widgets/revenue-widget";
25
+ import { OrdersWidget } from "./widgets/orders-widget";
26
+ import { CustomersWidget } from "./widgets/customers-widget";
27
+ import { StockWidget } from "./widgets/stock-widget";
28
+
29
+ interface WidgetContainerProps {
30
+ widgets: WidgetConfig[];
31
+ onWidgetsChange: (widgets: WidgetConfig[]) => void;
32
+ widgetData?: {
33
+ revenue?: { total: number; change?: number };
34
+ orders?: {
35
+ total: number;
36
+ pending?: number;
37
+ completed?: number;
38
+ change?: number;
39
+ };
40
+ customers?: { total: number; newToday?: number; change?: number };
41
+ stock?: {
42
+ totalItems: number;
43
+ lowStockCount: number;
44
+ lowStockItems?: Array<{
45
+ id: string;
46
+ name: string;
47
+ quantity: number;
48
+ minStock: number;
49
+ }>;
50
+ };
51
+ };
52
+ loading?: boolean;
53
+ className?: string;
54
+ }
55
+
56
+ export function WidgetContainer({
57
+ widgets,
58
+ onWidgetsChange,
59
+ widgetData,
60
+ loading = false,
61
+ className,
62
+ }: WidgetContainerProps) {
63
+ const [showHiddenWidgets, setShowHiddenWidgets] = useState(false);
64
+
65
+ const sensors = useSensors(
66
+ useSensor(PointerSensor, {
67
+ activationConstraint: {
68
+ distance: 8,
69
+ },
70
+ }),
71
+ useSensor(KeyboardSensor, {
72
+ coordinateGetter: sortableKeyboardCoordinates,
73
+ }),
74
+ );
75
+
76
+ const handleDragEnd = useCallback(
77
+ (event: DragEndEvent) => {
78
+ const { active, over } = event;
79
+
80
+ if (over && active.id !== over.id) {
81
+ const oldIndex = widgets.findIndex((w) => w.id === active.id);
82
+ const newIndex = widgets.findIndex((w) => w.id === over.id);
83
+
84
+ const newWidgets = arrayMove(widgets, oldIndex, newIndex).map(
85
+ (widget, index) => ({
86
+ ...widget,
87
+ position: index,
88
+ }),
89
+ );
90
+
91
+ onWidgetsChange(newWidgets);
92
+ }
93
+ },
94
+ [widgets, onWidgetsChange],
95
+ );
96
+
97
+ const handleToggleVisibility = useCallback(
98
+ (widgetId: string) => {
99
+ const newWidgets = widgets.map((widget) =>
100
+ widget.id === widgetId
101
+ ? { ...widget, visible: !widget.visible }
102
+ : widget,
103
+ );
104
+ onWidgetsChange(newWidgets);
105
+ },
106
+ [widgets, onWidgetsChange],
107
+ );
108
+
109
+ const handleRemoveWidget = useCallback(
110
+ (widgetId: string) => {
111
+ const newWidgets = widgets.map((widget) =>
112
+ widget.id === widgetId ? { ...widget, visible: false } : widget,
113
+ );
114
+ onWidgetsChange(newWidgets);
115
+ },
116
+ [widgets, onWidgetsChange],
117
+ );
118
+
119
+ const handleResetWidgets = useCallback(() => {
120
+ onWidgetsChange(DEFAULT_WIDGETS);
121
+ }, [onWidgetsChange]);
122
+
123
+ const visibleWidgets = widgets
124
+ .filter((w) => w.visible)
125
+ .sort((a, b) => a.position - b.position);
126
+
127
+ const hiddenWidgets = widgets.filter((w) => !w.visible);
128
+
129
+ const renderWidget = (config: WidgetConfig) => {
130
+ const commonProps = {
131
+ config,
132
+ loading,
133
+ onRemove: () => handleRemoveWidget(config.id),
134
+ onToggleVisibility: () => handleToggleVisibility(config.id),
135
+ };
136
+
137
+ switch (config.type) {
138
+ case "revenue":
139
+ return <RevenueWidget {...commonProps} data={widgetData?.revenue} />;
140
+ case "orders":
141
+ return <OrdersWidget {...commonProps} data={widgetData?.orders} />;
142
+ case "customers":
143
+ return (
144
+ <CustomersWidget {...commonProps} data={widgetData?.customers} />
145
+ );
146
+ case "stock":
147
+ return <StockWidget {...commonProps} data={widgetData?.stock} />;
148
+ default:
149
+ return null;
150
+ }
151
+ };
152
+
153
+ return (
154
+ <div className={cn("space-y-4", className)}>
155
+ {/* Header */}
156
+ <div className="flex items-center justify-between">
157
+ <h2 className="text-xl font-semibold text-foreground">
158
+ Widget Dashboard
159
+ </h2>
160
+ <div className="flex items-center gap-2">
161
+ {hiddenWidgets.length > 0 && (
162
+ <button
163
+ onClick={() => setShowHiddenWidgets(!showHiddenWidgets)}
164
+ className="inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-sm font-medium text-muted-foreground hover:bg-muted hover:text-foreground transition-colors"
165
+ >
166
+ {showHiddenWidgets ? (
167
+ <EyeOff className="h-4 w-4" />
168
+ ) : (
169
+ <Eye className="h-4 w-4" />
170
+ )}
171
+ <span>{hiddenWidgets.length} ẩn</span>
172
+ </button>
173
+ )}
174
+ <button
175
+ onClick={handleResetWidgets}
176
+ className="inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-sm font-medium text-muted-foreground hover:bg-muted hover:text-foreground transition-colors"
177
+ title="Khôi phục mặc định"
178
+ >
179
+ <RotateCcw className="h-4 w-4" />
180
+ </button>
181
+ </div>
182
+ </div>
183
+
184
+ {/* Hidden widgets panel */}
185
+ <AnimatePresence>
186
+ {showHiddenWidgets && hiddenWidgets.length > 0 && (
187
+ <motion.div
188
+ initial={{ opacity: 0, height: 0 }}
189
+ animate={{ opacity: 1, height: "auto" }}
190
+ exit={{ opacity: 0, height: 0 }}
191
+ // @ts-ignore className is valid for motion.div
192
+ className="overflow-hidden"
193
+ >
194
+ <div className="rounded-lg border border-dashed border-border bg-muted/30 p-4">
195
+ <p className="mb-3 text-sm font-medium text-muted-foreground">
196
+ Widget đã ẩn - Click để hiển thị lại
197
+ </p>
198
+ <div className="flex flex-wrap gap-2">
199
+ {hiddenWidgets.map((widget) => (
200
+ <button
201
+ key={widget.id}
202
+ onClick={() => handleToggleVisibility(widget.id)}
203
+ className="inline-flex items-center gap-1.5 rounded-lg bg-background px-3 py-2 text-sm font-medium shadow-sm border border-border hover:bg-muted transition-colors"
204
+ >
205
+ <Plus className="h-4 w-4" />
206
+ {WIDGET_LABELS[widget.type]}
207
+ </button>
208
+ ))}
209
+ </div>
210
+ </div>
211
+ </motion.div>
212
+ )}
213
+ </AnimatePresence>
214
+
215
+ {/* Sortable widget grid */}
216
+ <DndContext
217
+ sensors={sensors}
218
+ collisionDetection={closestCenter}
219
+ onDragEnd={handleDragEnd}
220
+ >
221
+ <SortableContext
222
+ items={visibleWidgets.map((w) => w.id)}
223
+ strategy={rectSortingStrategy}
224
+ >
225
+ <div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
226
+ <AnimatePresence mode="popLayout">
227
+ {visibleWidgets.map((widget) => (
228
+ <div key={widget.id}>{renderWidget(widget)}</div>
229
+ ))}
230
+ </AnimatePresence>
231
+ </div>
232
+ </SortableContext>
233
+ </DndContext>
234
+
235
+ {/* Empty state */}
236
+ {visibleWidgets.length === 0 && (
237
+ <div className="flex flex-col items-center justify-center rounded-lg border border-dashed border-border py-12 text-center">
238
+ <div className="rounded-full bg-muted p-3 mb-4">
239
+ <Plus className="h-6 w-6 text-muted-foreground" />
240
+ </div>
241
+ <h3 className="font-medium text-foreground mb-1">
242
+ Chưa có widget nào
243
+ </h3>
244
+ <p className="text-sm text-muted-foreground mb-4">
245
+ Thêm widget để theo dõi các chỉ số quan trọng
246
+ </p>
247
+ <button
248
+ onClick={handleResetWidgets}
249
+ className="inline-flex items-center gap-2 rounded-lg bg-primary px-4 py-2 text-sm font-medium text-primary-foreground hover:bg-primary/90 transition-colors"
250
+ >
251
+ <RotateCcw className="h-4 w-4" />
252
+ Khôi phục mặc định
253
+ </button>
254
+ </div>
255
+ )}
256
+ </div>
257
+ );
258
+ }
@@ -0,0 +1,200 @@
1
+ "use client";
2
+
3
+ import { motion } from "framer-motion";
4
+ import { useSortable } from "@dnd-kit/sortable";
5
+ import { CSS } from "@dnd-kit/utilities";
6
+ import { GripVertical, Settings, X, Eye, EyeOff } from "lucide-react";
7
+ import { cn } from "../../utils";
8
+ import type { WidgetConfig, WidgetSize } from "../types";
9
+
10
+ interface BaseWidgetProps {
11
+ config: WidgetConfig;
12
+ title: string;
13
+ children: React.ReactNode;
14
+ loading?: boolean;
15
+ error?: string;
16
+ onRemove?: () => void;
17
+ onToggleVisibility?: () => void;
18
+ onSettings?: () => void;
19
+ className?: string;
20
+ isDragging?: boolean;
21
+ }
22
+
23
+ const sizeClasses: Record<WidgetSize, string> = {
24
+ small: "col-span-1",
25
+ medium: "col-span-1 md:col-span-1",
26
+ large: "col-span-1 md:col-span-2",
27
+ };
28
+
29
+ export function BaseWidget({
30
+ config,
31
+ title,
32
+ children,
33
+ loading = false,
34
+ error,
35
+ onRemove,
36
+ onToggleVisibility,
37
+ onSettings,
38
+ className,
39
+ isDragging = false,
40
+ }: BaseWidgetProps) {
41
+ const {
42
+ attributes,
43
+ listeners,
44
+ setNodeRef,
45
+ transform,
46
+ transition,
47
+ isDragging: isSortableDragging,
48
+ } = useSortable({ id: config.id });
49
+
50
+ const style = {
51
+ transform: CSS.Transform.toString(transform),
52
+ transition,
53
+ };
54
+
55
+ const isCurrentlyDragging = isDragging || isSortableDragging;
56
+
57
+ if (!config.visible) {
58
+ return null;
59
+ }
60
+
61
+ return (
62
+ <motion.div
63
+ ref={setNodeRef}
64
+ style={style}
65
+ initial={{ opacity: 0, scale: 0.95 }}
66
+ animate={{ opacity: 1, scale: 1 }}
67
+ exit={{ opacity: 0, scale: 0.95 }}
68
+ // @ts-ignore className is valid for motion.div
69
+ className={cn(
70
+ "group relative flex flex-col justify-between gap-3 rounded-xl border border-slate-200 bg-white p-5 shadow-sm transition-all hover:border-primary/30 hover:shadow-md dark:border-slate-700 dark:bg-slate-800",
71
+ sizeClasses[config.size],
72
+ isCurrentlyDragging && "z-50 shadow-lg ring-2 ring-primary/20",
73
+ className,
74
+ )}
75
+ >
76
+ {/* Header */}
77
+ <div className="flex items-center justify-between mb-1">
78
+ <div className="flex items-center gap-2">
79
+ {/* Drag handle */}
80
+ <button
81
+ {...attributes}
82
+ {...listeners}
83
+ className="cursor-grab touch-none rounded text-slate-400 hover:text-slate-600 dark:hover:text-slate-300 active:cursor-grabbing opacity-0 group-hover:opacity-100 transition-opacity"
84
+ aria-label="Drag to reorder"
85
+ >
86
+ <GripVertical className="h-4 w-4" />
87
+ </button>
88
+ <h4 className="flex items-center gap-2 text-sm font-semibold text-slate-900 dark:text-white">
89
+ {title}
90
+ </h4>
91
+ </div>
92
+
93
+ {/* Actions */}
94
+ <div className="flex items-center gap-1 opacity-0 transition-opacity group-hover:opacity-100">
95
+ {onToggleVisibility && (
96
+ <button
97
+ onClick={onToggleVisibility}
98
+ className="rounded p-1 text-slate-400 hover:bg-slate-100 hover:text-slate-700 dark:hover:bg-slate-700 dark:hover:text-slate-300"
99
+ aria-label={config.visible ? "Hide widget" : "Show widget"}
100
+ >
101
+ {config.visible ? (
102
+ <Eye className="h-4 w-4" />
103
+ ) : (
104
+ <EyeOff className="h-4 w-4" />
105
+ )}
106
+ </button>
107
+ )}
108
+ {onSettings && (
109
+ <button
110
+ onClick={onSettings}
111
+ className="rounded p-1 text-slate-400 hover:bg-slate-100 hover:text-slate-700 dark:hover:bg-slate-700 dark:hover:text-slate-300"
112
+ aria-label="Widget settings"
113
+ >
114
+ <Settings className="h-4 w-4" />
115
+ </button>
116
+ )}
117
+ {onRemove && (
118
+ <button
119
+ onClick={onRemove}
120
+ className="rounded p-1 text-slate-400 hover:bg-red-50 hover:text-red-500 dark:hover:bg-red-900/20"
121
+ aria-label="Remove widget"
122
+ >
123
+ <X className="h-4 w-4" />
124
+ </button>
125
+ )}
126
+ </div>
127
+ </div>
128
+
129
+ {/* Content */}
130
+ <div className="flex-1">
131
+ {loading ? (
132
+ <div className="flex items-center justify-center py-4">
133
+ <div className="h-6 w-6 animate-spin rounded-full border-2 border-primary border-t-transparent" />
134
+ </div>
135
+ ) : error ? (
136
+ <div className="flex items-center justify-center py-4 text-xs text-red-500">
137
+ {error}
138
+ </div>
139
+ ) : (
140
+ children
141
+ )}
142
+ </div>
143
+ </motion.div>
144
+ );
145
+ }
146
+
147
+ // Stat display component for widgets
148
+ interface WidgetStatProps {
149
+ value: string | number;
150
+ label?: string;
151
+ change?: {
152
+ value: number;
153
+ type: "increase" | "decrease";
154
+ period?: string;
155
+ };
156
+ valueClassName?: string;
157
+ }
158
+
159
+ export function WidgetStat({
160
+ value,
161
+ label,
162
+ change,
163
+ valueClassName,
164
+ }: WidgetStatProps) {
165
+ return (
166
+ <div className="space-y-2">
167
+ <div className={cn("text-3xl font-bold text-foreground", valueClassName)}>
168
+ {value}
169
+ </div>
170
+ {label && <p className="text-sm text-muted-foreground">{label}</p>}
171
+ {change && (
172
+ <div
173
+ className={cn(
174
+ "inline-flex items-center gap-1 rounded-full px-2 py-0.5 text-xs font-medium",
175
+ change.type === "increase"
176
+ ? "bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-400"
177
+ : "bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-400",
178
+ )}
179
+ >
180
+ {change.type === "increase" ? (
181
+ <svg className="h-3 w-3" viewBox="0 0 12 12" fill="currentColor">
182
+ <path d="M6 2L10 7H2L6 2Z" />
183
+ </svg>
184
+ ) : (
185
+ <svg className="h-3 w-3" viewBox="0 0 12 12" fill="currentColor">
186
+ <path d="M6 10L2 5H10L6 10Z" />
187
+ </svg>
188
+ )}
189
+ <span>
190
+ {change.value > 0 ? "+" : ""}
191
+ {change.value}%
192
+ </span>
193
+ {change.period && (
194
+ <span className="text-muted-foreground">vs {change.period}</span>
195
+ )}
196
+ </div>
197
+ )}
198
+ </div>
199
+ );
200
+ }
@@ -0,0 +1,74 @@
1
+ "use client";
2
+
3
+ import { Users } from "lucide-react";
4
+ import { BaseWidget, WidgetStat } from "./base-widget";
5
+ import type { WidgetConfig } from "../types";
6
+
7
+ interface CustomersWidgetProps {
8
+ config: WidgetConfig;
9
+ data?: {
10
+ total: number;
11
+ newToday?: number;
12
+ change?: number;
13
+ changePeriod?: string;
14
+ };
15
+ loading?: boolean;
16
+ error?: string;
17
+ onRemove?: () => void;
18
+ onToggleVisibility?: () => void;
19
+ onSettings?: () => void;
20
+ }
21
+
22
+ export function CustomersWidget({
23
+ config,
24
+ data,
25
+ loading,
26
+ error,
27
+ onRemove,
28
+ onToggleVisibility,
29
+ onSettings,
30
+ }: CustomersWidgetProps) {
31
+ return (
32
+ <BaseWidget
33
+ config={config}
34
+ title="Khách hàng"
35
+ loading={loading}
36
+ error={error}
37
+ onRemove={onRemove}
38
+ onToggleVisibility={onToggleVisibility}
39
+ onSettings={onSettings}
40
+ >
41
+ <div className="flex items-start justify-between">
42
+ <div className="space-y-3">
43
+ <WidgetStat
44
+ value={data?.total ?? 0}
45
+ label="Tổng khách hàng"
46
+ change={
47
+ data?.change !== undefined
48
+ ? {
49
+ value: data.change,
50
+ type: data.change >= 0 ? "increase" : "decrease",
51
+ period: data.changePeriod || "tháng trước",
52
+ }
53
+ : undefined
54
+ }
55
+ valueClassName="text-purple-600 dark:text-purple-400"
56
+ />
57
+
58
+ {data?.newToday !== undefined && data.newToday > 0 && (
59
+ <div className="text-sm">
60
+ <span className="text-muted-foreground">Khách mới hôm nay: </span>
61
+ <span className="font-medium text-green-600 dark:text-green-400">
62
+ +{data.newToday}
63
+ </span>
64
+ </div>
65
+ )}
66
+ </div>
67
+
68
+ <div className="rounded-full bg-purple-100 p-3 dark:bg-purple-900/30">
69
+ <Users className="h-6 w-6 text-purple-600 dark:text-purple-400" />
70
+ </div>
71
+ </div>
72
+ </BaseWidget>
73
+ );
74
+ }
@@ -0,0 +1,6 @@
1
+ // Widget exports
2
+ export { BaseWidget, WidgetStat } from "./base-widget";
3
+ export { RevenueWidget } from "./revenue-widget";
4
+ export { OrdersWidget } from "./orders-widget";
5
+ export { CustomersWidget } from "./customers-widget";
6
+ export { StockWidget } from "./stock-widget";