@goplusvn/core 0.1.0 → 0.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (591) hide show
  1. package/package.json +31 -175
  2. package/src/assets/erp_wallpaper.png +0 -0
  3. package/src/assets/goeat_logo.png +0 -0
  4. package/src/audit/audit-manager.ts +139 -0
  5. package/src/audit/index.ts +11 -0
  6. package/src/audit/memory-audit-logger.ts +86 -0
  7. package/src/audit/types.ts +50 -0
  8. package/src/auth/auth-service.ts +97 -0
  9. package/src/auth/index.ts +266 -0
  10. package/src/code-generation/index.ts +69 -0
  11. package/src/configs/auth-routes.ts +17 -0
  12. package/src/configs/crud.ts +136 -0
  13. package/src/configs/data/navigations.ts +781 -0
  14. package/src/configs/data/oauth-links.ts +10 -0
  15. package/src/configs/entities/material-categories.config.ts +125 -0
  16. package/src/configs/i18n.ts +12 -0
  17. package/src/configs/index.ts +26 -0
  18. package/src/configs/status.ts +25 -0
  19. package/src/configs/themes.ts +100 -0
  20. package/src/crud/components/crud-bulk-actions.tsx +91 -0
  21. package/src/crud/components/crud-card-view.tsx +241 -0
  22. package/src/crud/components/crud-context.tsx +122 -0
  23. package/src/crud/components/crud-delete-dialog.tsx +145 -0
  24. package/src/crud/components/crud-dialog.tsx +406 -0
  25. package/src/crud/components/crud-empty-state.tsx +104 -0
  26. package/src/crud/components/crud-export-button.tsx +170 -0
  27. package/src/crud/components/crud-field-renderer.tsx +653 -0
  28. package/src/crud/components/crud-filter-chips.tsx +102 -0
  29. package/src/crud/components/crud-filters/checkbox-filter.tsx +97 -0
  30. package/src/crud/components/crud-filters/datetime-filter.tsx +83 -0
  31. package/src/crud/components/crud-filters/filter-builder.tsx +66 -0
  32. package/src/crud/components/crud-filters/index.tsx +76 -0
  33. package/src/crud/components/crud-filters/radio-filter.tsx +86 -0
  34. package/src/crud/components/crud-filters/select-filter.tsx +141 -0
  35. package/src/crud/components/crud-filters/text-filter.tsx +86 -0
  36. package/src/crud/components/crud-form.tsx +642 -0
  37. package/src/crud/components/crud-import-dialog.tsx +440 -0
  38. package/src/crud/components/crud-infinite-scroll.tsx +116 -0
  39. package/src/crud/components/crud-page.tsx +1017 -0
  40. package/src/crud/components/crud-provider.tsx +277 -0
  41. package/src/crud/components/crud-row-actions.tsx +189 -0
  42. package/src/crud/components/crud-search.tsx +82 -0
  43. package/src/crud/components/crud-sheet.tsx +336 -0
  44. package/src/crud/components/crud-table-skeleton.tsx +26 -0
  45. package/src/crud/components/crud-table-toolbar.tsx +91 -0
  46. package/src/crud/components/crud-table.tsx +352 -0
  47. package/src/crud/components/crud-virtual-table.tsx +55 -0
  48. package/src/crud/components/index.tsx +20 -0
  49. package/src/crud/crud-filters/checkbox-filter.tsx +87 -0
  50. package/src/crud/crud-filters/datetime-filter.tsx +82 -0
  51. package/src/crud/crud-filters/filter-builder.tsx +64 -0
  52. package/src/crud/crud-filters/index.tsx +78 -0
  53. package/src/crud/crud-filters/radio-filter.tsx +79 -0
  54. package/src/crud/crud-filters/select-filter.tsx +148 -0
  55. package/src/crud/crud-filters/text-filter.tsx +81 -0
  56. package/src/crud/index.ts +43 -0
  57. package/src/crud/lib/crud-service.test.ts +334 -0
  58. package/src/crud/lib/crud-service.ts +358 -0
  59. package/src/crud/lib/crud-utils.test.ts +354 -0
  60. package/src/crud/lib/crud-utils.ts +299 -0
  61. package/src/crud/lib/crud-validator.ts +247 -0
  62. package/src/crud/lib/data-loader.ts +234 -0
  63. package/src/crud/lib/field-calculator.ts +241 -0
  64. package/src/crud/lib/field-formatter.ts +240 -0
  65. package/src/crud/lib/import-export-service.test.ts +290 -0
  66. package/src/crud/lib/import-export-service.ts +352 -0
  67. package/src/crud/lib/import-server-utils.ts +109 -0
  68. package/src/crud/lib/lazy-loader.ts +241 -0
  69. package/src/crud/lib/parse-filters.ts +85 -0
  70. package/src/crud/lib/permissions.ts +52 -0
  71. package/src/crud/lib/serialize-config.ts +60 -0
  72. package/src/crud/lib/stream-loader.ts +145 -0
  73. package/src/crud/lib/translate-config.ts +335 -0
  74. package/src/crud/lib/types.ts +11 -0
  75. package/src/crud/pages/entity-crud-page.tsx +144 -0
  76. package/src/crud/server.ts +8 -0
  77. package/src/home/constants.tsx +142 -0
  78. package/src/home/feature-showcase.tsx +171 -0
  79. package/src/home/home-page.tsx +191 -0
  80. package/src/home/hooks/index.ts +1 -0
  81. package/src/home/hooks/useWidgetPreferences.ts +167 -0
  82. package/src/home/index.ts +33 -0
  83. package/src/home/quick-access-dialog.tsx +271 -0
  84. package/src/home/quick-access-menu.tsx +267 -0
  85. package/src/home/types.ts +140 -0
  86. package/src/home/welcome-card.tsx +92 -0
  87. package/src/home/widget-container.tsx +258 -0
  88. package/src/home/widgets/base-widget.tsx +200 -0
  89. package/src/home/widgets/customers-widget.tsx +74 -0
  90. package/src/home/widgets/index.ts +6 -0
  91. package/src/home/widgets/orders-widget.tsx +87 -0
  92. package/src/home/widgets/revenue-widget.tsx +71 -0
  93. package/src/home/widgets/stock-widget.tsx +109 -0
  94. package/src/hooks/index.tsx +598 -0
  95. package/src/hooks/use-tenant.test.tsx +30 -0
  96. package/src/hooks/use-tenant.ts +5 -0
  97. package/src/index.ts +17 -0
  98. package/src/infrastructure/__tests__/architecture-verification.spec.ts +103 -0
  99. package/src/infrastructure/api-service.ts +317 -0
  100. package/src/infrastructure/cache/cache-manager.ts +107 -0
  101. package/src/infrastructure/cache/cache.ts +120 -0
  102. package/src/infrastructure/cache/index.ts +8 -0
  103. package/src/infrastructure/cache/types.ts +48 -0
  104. package/src/infrastructure/cron/cron-manager.ts +239 -0
  105. package/src/infrastructure/cron/index.ts +6 -0
  106. package/src/infrastructure/cron/types.ts +41 -0
  107. package/src/infrastructure/event-bus/event-bus.ts +145 -0
  108. package/src/infrastructure/event-bus/index.ts +2 -0
  109. package/src/infrastructure/event-bus/types.ts +22 -0
  110. package/src/infrastructure/index.ts +32 -0
  111. package/src/infrastructure/lock/decorators.ts +67 -0
  112. package/src/infrastructure/lock/index.ts +2 -0
  113. package/src/infrastructure/lock/lock-manager.ts +33 -0
  114. package/src/infrastructure/logger/index.ts +2 -0
  115. package/src/infrastructure/logger/logger.ts +96 -0
  116. package/src/infrastructure/logger/types.ts +25 -0
  117. package/src/layout/index.tsx +185 -0
  118. package/src/navigation/index.ts +91 -0
  119. package/src/notification/index.ts +14 -0
  120. package/src/notification/notification-service.ts +120 -0
  121. package/src/notification/storage/in-memory.ts +56 -0
  122. package/src/notification/storage/index.ts +1 -0
  123. package/src/notification/types.ts +51 -0
  124. package/src/organization/branch-service.ts +299 -0
  125. package/src/organization/branches.config.ts +154 -0
  126. package/src/organization/index.ts +5 -0
  127. package/src/plugin/apps-registry.ts +97 -0
  128. package/src/plugin/index.ts +5 -0
  129. package/src/plugin/types.ts +41 -0
  130. package/src/providers/index.tsx +109 -0
  131. package/src/providers/tenant-provider.tsx +45 -0
  132. package/src/rbac/components/roles/role-card.tsx +158 -0
  133. package/src/rbac/components/roles/role-stats-cards.tsx +29 -0
  134. package/src/rbac/components/roles/role-toolbar.tsx +123 -0
  135. package/src/rbac/hooks/use-role-operations.ts +159 -0
  136. package/src/rbac/hooks/use-roles-data.ts +59 -0
  137. package/src/rbac/index.ts +297 -0
  138. package/src/rbac/lib/permission-helpers.ts +63 -0
  139. package/src/rbac/pages/action-list-page.tsx +25 -0
  140. package/src/rbac/pages/resource-list-page.tsx +25 -0
  141. package/src/rbac/pages/role-list-page.tsx +378 -0
  142. package/src/rbac/permission-service.ts +140 -0
  143. package/src/rbac/permissions.ts +135 -0
  144. package/src/rbac/resource-service.ts +115 -0
  145. package/src/rbac/resource-validator.ts +119 -0
  146. package/src/rbac/role-service.ts +165 -0
  147. package/src/rbac/server.ts +16 -0
  148. package/src/rbac/types.ts +38 -0
  149. package/src/schemas/action.schema.ts +66 -0
  150. package/src/schemas/branch.schema.ts +52 -0
  151. package/src/schemas/coming-soon-schema.ts +9 -0
  152. package/src/schemas/company.schema.ts +44 -0
  153. package/src/schemas/forgot-passward-schema.ts +9 -0
  154. package/src/schemas/index.ts +30 -0
  155. package/src/schemas/material-category.schema.ts +43 -0
  156. package/src/schemas/material-pricing.schema.ts +74 -0
  157. package/src/schemas/material.schema.ts +76 -0
  158. package/src/schemas/materials.ts +52 -0
  159. package/src/schemas/new-passward-schema.ts +15 -0
  160. package/src/schemas/partner-company.schema.ts +149 -0
  161. package/src/schemas/register-schema.ts +36 -0
  162. package/src/schemas/resource.schema.ts +133 -0
  163. package/src/schemas/role.schema.ts +11 -0
  164. package/src/schemas/sign-in-schema.ts +24 -0
  165. package/src/schemas/supplier-pricing.schema.ts +15 -0
  166. package/src/schemas/supplier.schema.ts +120 -0
  167. package/src/schemas/system-category-group.schema.ts +67 -0
  168. package/src/schemas/system-category.schema.ts +77 -0
  169. package/src/schemas/system-config.schema.ts +118 -0
  170. package/src/schemas/uom.schema.ts +75 -0
  171. package/src/schemas/user-supplier.schema.ts +179 -0
  172. package/src/schemas/user.schema.ts +18 -0
  173. package/src/schemas/verify-email-schema.ts +9 -0
  174. package/src/schemas/warehouse.schema.ts +49 -0
  175. package/src/system/components/categories/category-list.tsx +529 -0
  176. package/src/system/components/categories/category-manager.tsx +89 -0
  177. package/src/system/components/categories/group-sidebar.tsx +308 -0
  178. package/src/system/components/settings/setting-dialogs.tsx +197 -0
  179. package/src/system/components/settings/setting-field.tsx +291 -0
  180. package/src/system/components/settings/setting-form-dialog.tsx +308 -0
  181. package/src/system/components/settings/settings-groups.ts +80 -0
  182. package/src/system/components/settings/settings-search.tsx +71 -0
  183. package/src/system/components/settings/settings-section.tsx +74 -0
  184. package/src/system/components/settings/settings-sidebar.tsx +81 -0
  185. package/src/system/constants.ts +3 -0
  186. package/src/system/index.ts +150 -0
  187. package/src/system/job-manager.ts +176 -0
  188. package/src/system/pages/components/categories/category-list.tsx +537 -0
  189. package/src/system/pages/components/categories/category-manager.tsx +90 -0
  190. package/src/system/pages/components/categories/group-sidebar.tsx +311 -0
  191. package/src/system/pages/components/settings/sales-rules-settings.tsx +222 -0
  192. package/src/system/pages/components/settings/setting-dialogs.tsx +197 -0
  193. package/src/system/pages/components/settings/setting-field.tsx +292 -0
  194. package/src/system/pages/components/settings/setting-form-dialog.tsx +308 -0
  195. package/src/system/pages/components/settings/settings-groups.ts +87 -0
  196. package/src/system/pages/components/settings/settings-page.tsx +372 -0
  197. package/src/system/pages/components/settings/settings-search.tsx +71 -0
  198. package/src/system/pages/components/settings/settings-section.tsx +74 -0
  199. package/src/system/pages/components/settings/settings-sidebar.tsx +81 -0
  200. package/src/system/pages/components/settings/system-settings.tsx +244 -0
  201. package/src/system/pages/system-category-page.tsx +15 -0
  202. package/src/system/pages/system-settings-page.tsx +380 -0
  203. package/src/system/schemas/system-category-group.schema.ts +46 -0
  204. package/src/system/schemas/system-category.schema.ts +56 -0
  205. package/src/system/services/settings-service.ts +127 -0
  206. package/src/system/services/system-category-service.ts +63 -0
  207. package/src/system/types.ts +45 -0
  208. package/src/types/index.ts +703 -0
  209. package/src/ui/auth/auth-layout.tsx +135 -0
  210. package/src/ui/auth/forgot-password-form.tsx +98 -0
  211. package/src/ui/auth/index.tsx +7 -0
  212. package/src/ui/auth/new-password-form.tsx +107 -0
  213. package/src/ui/auth/oauth-links.tsx +30 -0
  214. package/src/ui/auth/register-form.tsx +202 -0
  215. package/src/ui/auth/sign-in-form.tsx +238 -0
  216. package/src/ui/auth/verify-email-form.tsx +104 -0
  217. package/src/ui/crud/index.tsx +10 -0
  218. package/src/ui/data-display/accordion.tsx +65 -0
  219. package/src/ui/data-display/aspect-ratio.tsx +11 -0
  220. package/src/ui/data-display/avatar.tsx +163 -0
  221. package/src/ui/data-display/bento-grid.tsx +77 -0
  222. package/src/ui/data-display/carousel.tsx +249 -0
  223. package/src/ui/data-display/chart.tsx +363 -0
  224. package/src/ui/data-display/code-block-highlight.tsx +54 -0
  225. package/src/ui/data-display/collapsible.tsx +42 -0
  226. package/src/ui/data-display/compact-stat-bar.tsx +149 -0
  227. package/src/ui/data-display/data-table/data-table-context.tsx +255 -0
  228. package/src/ui/data-display/data-table/data-table-empty-state.tsx +133 -0
  229. package/src/ui/data-display/data-table/data-table-skeleton.tsx +145 -0
  230. package/src/ui/data-display/data-table/data-table-toolbar.tsx +353 -0
  231. package/src/ui/data-display/data-table/data-table.tsx +597 -0
  232. package/src/ui/data-display/data-table/index.ts +44 -0
  233. package/src/ui/data-display/data-table-column-header.tsx +75 -0
  234. package/src/ui/data-display/data-table-pagination.tsx +130 -0
  235. package/src/ui/data-display/data-table-view-options.tsx +59 -0
  236. package/src/ui/data-display/formatted-number-input.tsx +210 -0
  237. package/src/ui/data-display/highlight.tsx +20 -0
  238. package/src/ui/data-display/hover-card.tsx +48 -0
  239. package/src/ui/data-display/index.tsx +50 -0
  240. package/src/ui/data-display/iphone-15-pro.tsx +114 -0
  241. package/src/ui/data-display/kanban/index.ts +4 -0
  242. package/src/ui/data-display/kanban/kanban-board.tsx +192 -0
  243. package/src/ui/data-display/kanban/kanban-column.tsx +74 -0
  244. package/src/ui/data-display/kanban/kanban-item.tsx +50 -0
  245. package/src/ui/data-display/kanban/kanban-types.ts +21 -0
  246. package/src/ui/data-display/kpi-card.tsx +68 -0
  247. package/src/ui/data-display/media-grid.tsx +110 -0
  248. package/src/ui/data-display/safari.tsx +175 -0
  249. package/src/ui/data-display/show-more-text.tsx +55 -0
  250. package/src/ui/data-display/tabs.tsx +68 -0
  251. package/src/ui/data-display/timeline.tsx +256 -0
  252. package/src/ui/feedback/alert.tsx +60 -0
  253. package/src/ui/feedback/context-menu.tsx +245 -0
  254. package/src/ui/feedback/drawer.tsx +132 -0
  255. package/src/ui/feedback/error-dialog.tsx +273 -0
  256. package/src/ui/feedback/index.tsx +183 -0
  257. package/src/ui/feedback/progress.tsx +32 -0
  258. package/src/ui/feedback/sheet.tsx +148 -0
  259. package/src/ui/feedback/sonner.tsx +36 -0
  260. package/src/ui/forms/command.tsx +157 -0
  261. package/src/ui/forms/date-picker.tsx +73 -0
  262. package/src/ui/forms/date-range-picker.tsx +76 -0
  263. package/src/ui/forms/date-time-picker.tsx +109 -0
  264. package/src/ui/forms/editor/editor-menu-bar.tsx +394 -0
  265. package/src/ui/forms/editor/index.tsx +130 -0
  266. package/src/ui/forms/editor/multi-select-example.tsx +1234 -0
  267. package/src/ui/forms/emoji-picker.tsx +109 -0
  268. package/src/ui/forms/file-dropzone.tsx +169 -0
  269. package/src/ui/forms/file-thumbnail.tsx +29 -0
  270. package/src/ui/forms/index.tsx +201 -0
  271. package/src/ui/forms/input-file.tsx +99 -0
  272. package/src/ui/forms/input-group.tsx +46 -0
  273. package/src/ui/forms/input-otp.tsx +81 -0
  274. package/src/ui/forms/input-phone.tsx +172 -0
  275. package/src/ui/forms/input-spin.tsx +116 -0
  276. package/src/ui/forms/input-tags.tsx +219 -0
  277. package/src/ui/forms/input-time.tsx +42 -0
  278. package/src/ui/forms/multi-select.tsx +629 -0
  279. package/src/ui/forms/multiple-date-picker.tsx +74 -0
  280. package/src/ui/forms/radio-group.tsx +42 -0
  281. package/src/ui/forms/rating.tsx +158 -0
  282. package/src/ui/forms/time-picker.tsx +57 -0
  283. package/src/ui/index.tsx +17 -0
  284. package/src/ui/layout/animated-list.tsx +77 -0
  285. package/src/ui/layout/animated-sidebar.tsx +294 -0
  286. package/src/ui/layout/command-menu.tsx +355 -0
  287. package/src/ui/layout/customizer.tsx +324 -0
  288. package/src/ui/layout/footer.tsx +43 -0
  289. package/src/ui/layout/full-screen-toggle.tsx +52 -0
  290. package/src/ui/layout/header-breadcrumb.tsx +77 -0
  291. package/src/ui/layout/horizontal-layout-header.tsx +83 -0
  292. package/src/ui/layout/horizontal-layout.tsx +50 -0
  293. package/src/ui/layout/index.tsx +25 -0
  294. package/src/ui/layout/language-dropdown.tsx +103 -0
  295. package/src/ui/layout/logo.tsx +63 -0
  296. package/src/ui/layout/main-layout.tsx +57 -0
  297. package/src/ui/layout/mode-dropdown.tsx +58 -0
  298. package/src/ui/layout/notification-dropdown.tsx +127 -0
  299. package/src/ui/layout/page-tabs.tsx +306 -0
  300. package/src/ui/layout/route-cache.tsx +214 -0
  301. package/src/ui/layout/sidebar-group-icon-menu.tsx +195 -0
  302. package/src/ui/layout/sidebar.tsx +279 -0
  303. package/src/ui/layout/tab-content-cache.tsx +201 -0
  304. package/src/ui/layout/tab-navigation-provider.tsx +536 -0
  305. package/src/ui/layout/toggle-mobile-sidebar.tsx +33 -0
  306. package/src/ui/layout/top-bar-header-menubar.tsx +412 -0
  307. package/src/ui/layout/user-dropdown.tsx +188 -0
  308. package/src/ui/layout/vertical-layout-header.tsx +65 -0
  309. package/src/ui/layout/vertical-layout.tsx +47 -0
  310. package/src/ui/management/audit-log-page.tsx +209 -0
  311. package/src/ui/management/cache-management.tsx +349 -0
  312. package/src/ui/management/index.ts +3 -0
  313. package/src/ui/management/job-management.tsx +308 -0
  314. package/src/ui/pages/not-found.tsx +30 -0
  315. package/src/ui/primitives/badge.tsx +66 -0
  316. package/src/ui/primitives/breadcrumb.tsx +103 -0
  317. package/src/ui/primitives/button.tsx +129 -0
  318. package/src/ui/primitives/calendar.tsx +74 -0
  319. package/src/ui/primitives/card.tsx +86 -0
  320. package/src/ui/primitives/checkbox.tsx +31 -0
  321. package/src/ui/primitives/client.ts +30 -0
  322. package/src/ui/primitives/combobox.tsx +290 -0
  323. package/src/ui/primitives/dialog.tsx +121 -0
  324. package/src/ui/primitives/dropdown-menu.tsx +239 -0
  325. package/src/ui/primitives/dynamic-icon.tsx +24 -0
  326. package/src/ui/primitives/index.tsx +134 -0
  327. package/src/ui/primitives/input-number.tsx +131 -0
  328. package/src/ui/primitives/input.tsx +22 -0
  329. package/src/ui/primitives/keyboard.tsx +23 -0
  330. package/src/ui/primitives/label.tsx +24 -0
  331. package/src/ui/primitives/menubar.tsx +262 -0
  332. package/src/ui/primitives/navigation-menu.tsx +157 -0
  333. package/src/ui/primitives/pagination.tsx +118 -0
  334. package/src/ui/primitives/popover.tsx +56 -0
  335. package/src/ui/primitives/prefetch-link.tsx +60 -0
  336. package/src/ui/primitives/resizable.tsx +59 -0
  337. package/src/ui/primitives/scroll-area.tsx +63 -0
  338. package/src/ui/primitives/select.tsx +172 -0
  339. package/src/ui/primitives/separator.tsx +51 -0
  340. package/src/ui/primitives/sidebar.tsx +844 -0
  341. package/src/ui/primitives/slider.tsx +27 -0
  342. package/src/ui/primitives/status-badge.tsx +47 -0
  343. package/src/ui/primitives/sticky-layout.tsx +50 -0
  344. package/src/ui/primitives/switch.tsx +29 -0
  345. package/src/ui/primitives/table.tsx +116 -0
  346. package/src/ui/primitives/tabs.tsx +55 -0
  347. package/src/ui/primitives/toggle-group.tsx +70 -0
  348. package/src/ui/primitives/toggle.tsx +47 -0
  349. package/src/ui/primitives/tooltip.tsx +59 -0
  350. package/src/user/components/dangerous-zone.tsx +34 -0
  351. package/src/user/components/delete-account-form.tsx +40 -0
  352. package/src/user/components/index.ts +4 -0
  353. package/src/user/components/profile-info-form.tsx +390 -0
  354. package/src/user/components/profile-info.tsx +32 -0
  355. package/src/user/components/unified-profile-dialog.tsx +1019 -0
  356. package/src/user/components/user-stats.tsx +27 -0
  357. package/src/user/components/user-toolbar.tsx +137 -0
  358. package/src/user/components/users-card-view.tsx +253 -0
  359. package/src/user/index.ts +11 -0
  360. package/src/user/pages/user-list-page.tsx +234 -0
  361. package/src/user/pages/users-client-page.tsx +385 -0
  362. package/src/user/profile-page.tsx +19 -0
  363. package/src/user/schemas.ts +68 -0
  364. package/src/user/types.ts +34 -0
  365. package/src/user/user-service.ts +538 -0
  366. package/src/utils/index.ts +906 -0
  367. package/src/workflow/activity-timeline.tsx +412 -0
  368. package/src/workflow/approval-workflow.tsx +31 -0
  369. package/src/workflow/index.ts +2 -0
  370. package/dist/audit/index.d.mts +0 -115
  371. package/dist/audit/index.d.ts +0 -115
  372. package/dist/audit/index.js +0 -204
  373. package/dist/audit/index.js.map +0 -1
  374. package/dist/audit/index.mjs +0 -200
  375. package/dist/audit/index.mjs.map +0 -1
  376. package/dist/auth/index.d.mts +0 -86
  377. package/dist/auth/index.d.ts +0 -86
  378. package/dist/auth/index.js +0 -210
  379. package/dist/auth/index.js.map +0 -1
  380. package/dist/auth/index.mjs +0 -198
  381. package/dist/auth/index.mjs.map +0 -1
  382. package/dist/button-1dWvP9Ib.d.mts +0 -30
  383. package/dist/button-1dWvP9Ib.d.ts +0 -30
  384. package/dist/calendar-2QzdEo1z.d.mts +0 -20
  385. package/dist/calendar-2QzdEo1z.d.ts +0 -20
  386. package/dist/code-generation/index.d.mts +0 -30
  387. package/dist/code-generation/index.d.ts +0 -30
  388. package/dist/code-generation/index.js +0 -31
  389. package/dist/code-generation/index.js.map +0 -1
  390. package/dist/code-generation/index.mjs +0 -28
  391. package/dist/code-generation/index.mjs.map +0 -1
  392. package/dist/configs/index.d.mts +0 -175
  393. package/dist/configs/index.d.ts +0 -175
  394. package/dist/configs/index.js +0 -254
  395. package/dist/configs/index.js.map +0 -1
  396. package/dist/configs/index.mjs +0 -233
  397. package/dist/configs/index.mjs.map +0 -1
  398. package/dist/crud/index.d.mts +0 -646
  399. package/dist/crud/index.d.ts +0 -646
  400. package/dist/crud/index.js +0 -11772
  401. package/dist/crud/index.js.map +0 -1
  402. package/dist/crud/index.mjs +0 -11665
  403. package/dist/crud/index.mjs.map +0 -1
  404. package/dist/crud/server.d.mts +0 -20
  405. package/dist/crud/server.d.ts +0 -20
  406. package/dist/crud/server.js +0 -123
  407. package/dist/crud/server.js.map +0 -1
  408. package/dist/crud/server.mjs +0 -120
  409. package/dist/crud/server.mjs.map +0 -1
  410. package/dist/data-table-skeleton-12NA8Mjx.d.mts +0 -39
  411. package/dist/data-table-skeleton-12NA8Mjx.d.ts +0 -39
  412. package/dist/dialog-bKfjZMTd.d.mts +0 -22
  413. package/dist/dialog-bKfjZMTd.d.ts +0 -22
  414. package/dist/dynamic-icon-DrGIiu2N.d.mts +0 -10
  415. package/dist/dynamic-icon-DrGIiu2N.d.ts +0 -10
  416. package/dist/home/index.d.mts +0 -269
  417. package/dist/home/index.d.ts +0 -269
  418. package/dist/home/index.js +0 -1678
  419. package/dist/home/index.js.map +0 -1
  420. package/dist/home/index.mjs +0 -1635
  421. package/dist/home/index.mjs.map +0 -1
  422. package/dist/hooks/index.d.mts +0 -7
  423. package/dist/hooks/index.d.ts +0 -7
  424. package/dist/hooks/index.js +0 -8316
  425. package/dist/hooks/index.js.map +0 -1
  426. package/dist/hooks/index.mjs +0 -8255
  427. package/dist/hooks/index.mjs.map +0 -1
  428. package/dist/index-50hpiPrV.d.ts +0 -116
  429. package/dist/index-B9zQVEVi.d.mts +0 -116
  430. package/dist/index.d.mts +0 -5
  431. package/dist/index.d.ts +0 -5
  432. package/dist/index.js +0 -123
  433. package/dist/index.js.map +0 -1
  434. package/dist/index.mjs +0 -118
  435. package/dist/index.mjs.map +0 -1
  436. package/dist/infrastructure/index.d.mts +0 -423
  437. package/dist/infrastructure/index.d.ts +0 -423
  438. package/dist/infrastructure/index.js +0 -633
  439. package/dist/infrastructure/index.js.map +0 -1
  440. package/dist/infrastructure/index.mjs +0 -619
  441. package/dist/infrastructure/index.mjs.map +0 -1
  442. package/dist/label-DWTEkNPo.d.ts +0 -226
  443. package/dist/label-LPpdcoBx.d.mts +0 -226
  444. package/dist/layout/index.d.mts +0 -48
  445. package/dist/layout/index.d.ts +0 -48
  446. package/dist/layout/index.js +0 -117
  447. package/dist/layout/index.js.map +0 -1
  448. package/dist/layout/index.mjs +0 -90
  449. package/dist/layout/index.mjs.map +0 -1
  450. package/dist/navigation/index.d.mts +0 -16
  451. package/dist/navigation/index.d.ts +0 -16
  452. package/dist/navigation/index.js +0 -53
  453. package/dist/navigation/index.js.map +0 -1
  454. package/dist/navigation/index.mjs +0 -50
  455. package/dist/navigation/index.mjs.map +0 -1
  456. package/dist/notification/index.d.mts +0 -105
  457. package/dist/notification/index.d.ts +0 -105
  458. package/dist/notification/index.js +0 -278
  459. package/dist/notification/index.js.map +0 -1
  460. package/dist/notification/index.mjs +0 -274
  461. package/dist/notification/index.mjs.map +0 -1
  462. package/dist/organization/index.d.mts +0 -99
  463. package/dist/organization/index.d.ts +0 -99
  464. package/dist/organization/index.js +0 -360
  465. package/dist/organization/index.js.map +0 -1
  466. package/dist/organization/index.mjs +0 -352
  467. package/dist/organization/index.mjs.map +0 -1
  468. package/dist/plugin/index.d.mts +0 -83
  469. package/dist/plugin/index.d.ts +0 -83
  470. package/dist/plugin/index.js +0 -86
  471. package/dist/plugin/index.js.map +0 -1
  472. package/dist/plugin/index.mjs +0 -84
  473. package/dist/plugin/index.mjs.map +0 -1
  474. package/dist/providers/index.d.mts +0 -25
  475. package/dist/providers/index.d.ts +0 -25
  476. package/dist/providers/index.js +0 -84
  477. package/dist/providers/index.js.map +0 -1
  478. package/dist/providers/index.mjs +0 -77
  479. package/dist/providers/index.mjs.map +0 -1
  480. package/dist/rbac/index.d.mts +0 -226
  481. package/dist/rbac/index.d.ts +0 -226
  482. package/dist/rbac/index.js +0 -4784
  483. package/dist/rbac/index.js.map +0 -1
  484. package/dist/rbac/index.mjs +0 -4722
  485. package/dist/rbac/index.mjs.map +0 -1
  486. package/dist/rbac/permissions.d.mts +0 -26
  487. package/dist/rbac/permissions.d.ts +0 -26
  488. package/dist/rbac/permissions.js +0 -94
  489. package/dist/rbac/permissions.js.map +0 -1
  490. package/dist/rbac/permissions.mjs +0 -90
  491. package/dist/rbac/permissions.mjs.map +0 -1
  492. package/dist/rbac/server.d.mts +0 -1
  493. package/dist/rbac/server.d.ts +0 -1
  494. package/dist/rbac/server.js +0 -128
  495. package/dist/rbac/server.js.map +0 -1
  496. package/dist/rbac/server.mjs +0 -124
  497. package/dist/rbac/server.mjs.map +0 -1
  498. package/dist/schemas/index.d.mts +0 -1257
  499. package/dist/schemas/index.d.ts +0 -1257
  500. package/dist/schemas/index.js +0 -572
  501. package/dist/schemas/index.js.map +0 -1
  502. package/dist/schemas/index.mjs +0 -523
  503. package/dist/schemas/index.mjs.map +0 -1
  504. package/dist/server-QuYCTa89.d.mts +0 -83
  505. package/dist/server-QuYCTa89.d.ts +0 -83
  506. package/dist/sonner-C74GlRDQ.d.mts +0 -71
  507. package/dist/sonner-C74GlRDQ.d.ts +0 -71
  508. package/dist/status-BOXZgIqX.d.mts +0 -12
  509. package/dist/status-BOXZgIqX.d.ts +0 -12
  510. package/dist/system/index.d.mts +0 -77
  511. package/dist/system/index.d.ts +0 -77
  512. package/dist/system/index.js +0 -102
  513. package/dist/system/index.js.map +0 -1
  514. package/dist/system/index.mjs +0 -100
  515. package/dist/system/index.mjs.map +0 -1
  516. package/dist/tabs-C6FfBwPY.d.mts +0 -18
  517. package/dist/tabs-C6FfBwPY.d.ts +0 -18
  518. package/dist/tenant-provider-B8eC_Wpb.d.mts +0 -27
  519. package/dist/tenant-provider-B8eC_Wpb.d.ts +0 -27
  520. package/dist/types/index.d.mts +0 -469
  521. package/dist/types/index.d.ts +0 -469
  522. package/dist/types/index.js +0 -25
  523. package/dist/types/index.js.map +0 -1
  524. package/dist/types/index.mjs +0 -21
  525. package/dist/types/index.mjs.map +0 -1
  526. package/dist/ui/auth.d.mts +0 -39
  527. package/dist/ui/auth.d.ts +0 -39
  528. package/dist/ui/auth.js +0 -4941
  529. package/dist/ui/auth.js.map +0 -1
  530. package/dist/ui/auth.mjs +0 -4896
  531. package/dist/ui/auth.mjs.map +0 -1
  532. package/dist/ui/crud.d.mts +0 -2
  533. package/dist/ui/crud.d.ts +0 -2
  534. package/dist/ui/crud.js +0 -4
  535. package/dist/ui/crud.js.map +0 -1
  536. package/dist/ui/crud.mjs +0 -3
  537. package/dist/ui/crud.mjs.map +0 -1
  538. package/dist/ui/data-display.d.mts +0 -596
  539. package/dist/ui/data-display.d.ts +0 -596
  540. package/dist/ui/data-display.js +0 -5307
  541. package/dist/ui/data-display.js.map +0 -1
  542. package/dist/ui/data-display.mjs +0 -5212
  543. package/dist/ui/data-display.mjs.map +0 -1
  544. package/dist/ui/feedback.d.mts +0 -55
  545. package/dist/ui/feedback.d.ts +0 -55
  546. package/dist/ui/feedback.js +0 -2608
  547. package/dist/ui/feedback.js.map +0 -1
  548. package/dist/ui/feedback.mjs +0 -2526
  549. package/dist/ui/feedback.mjs.map +0 -1
  550. package/dist/ui/forms.d.mts +0 -309
  551. package/dist/ui/forms.d.ts +0 -309
  552. package/dist/ui/forms.js +0 -4656
  553. package/dist/ui/forms.js.map +0 -1
  554. package/dist/ui/forms.mjs +0 -4571
  555. package/dist/ui/forms.mjs.map +0 -1
  556. package/dist/ui/index.d.mts +0 -331
  557. package/dist/ui/index.d.ts +0 -331
  558. package/dist/ui/index.js +0 -16953
  559. package/dist/ui/index.js.map +0 -1
  560. package/dist/ui/index.mjs +0 -16598
  561. package/dist/ui/index.mjs.map +0 -1
  562. package/dist/ui/primitives/client.d.mts +0 -61
  563. package/dist/ui/primitives/client.d.ts +0 -61
  564. package/dist/ui/primitives/client.js +0 -3408
  565. package/dist/ui/primitives/client.js.map +0 -1
  566. package/dist/ui/primitives/client.mjs +0 -3256
  567. package/dist/ui/primitives/client.mjs.map +0 -1
  568. package/dist/ui/primitives.d.mts +0 -113
  569. package/dist/ui/primitives.d.ts +0 -113
  570. package/dist/ui/primitives.js +0 -3356
  571. package/dist/ui/primitives.js.map +0 -1
  572. package/dist/ui/primitives.mjs +0 -3227
  573. package/dist/ui/primitives.mjs.map +0 -1
  574. package/dist/user/index.d.mts +0 -228
  575. package/dist/user/index.d.ts +0 -228
  576. package/dist/user/index.js +0 -4306
  577. package/dist/user/index.js.map +0 -1
  578. package/dist/user/index.mjs +0 -4260
  579. package/dist/user/index.mjs.map +0 -1
  580. package/dist/utils/index.d.mts +0 -205
  581. package/dist/utils/index.d.ts +0 -205
  582. package/dist/utils/index.js +0 -574
  583. package/dist/utils/index.js.map +0 -1
  584. package/dist/utils/index.mjs +0 -514
  585. package/dist/utils/index.mjs.map +0 -1
  586. package/dist/workflow/index.d.mts +0 -40
  587. package/dist/workflow/index.d.ts +0 -40
  588. package/dist/workflow/index.js +0 -3710
  589. package/dist/workflow/index.js.map +0 -1
  590. package/dist/workflow/index.mjs +0 -3677
  591. package/dist/workflow/index.mjs.map +0 -1
@@ -0,0 +1,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";