@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,598 @@
1
+ // @goerp/core/hooks
2
+ // Core React hooks for GoERP platform
3
+ // Consolidated from packages/shared/hooks
4
+
5
+ "use client";
6
+
7
+ import {
8
+ useState,
9
+ useEffect,
10
+ useCallback,
11
+ useRef,
12
+ useContext,
13
+ createContext,
14
+ useMemo,
15
+ } from "react";
16
+ import type { ReactNode } from "react";
17
+ import {
18
+ useSearchParams,
19
+ useRouter,
20
+ usePathname,
21
+ useParams,
22
+ } from "next/navigation";
23
+ import { useMedia } from "react-use";
24
+ import { useDirection } from "@radix-ui/react-direction";
25
+
26
+ import { SettingsContext, defaultSettings } from "../providers";
27
+ import { remToPx } from "../utils";
28
+
29
+ const defaultSettingsValue = {
30
+ settings: defaultSettings,
31
+ updateSettings: () => {},
32
+ resetSettings: () => {},
33
+ };
34
+
35
+ // ============================================================================
36
+ // Debounce Hook
37
+ // ============================================================================
38
+
39
+ export function useDebounce<T>(value: T, delay: number): T {
40
+ const [debouncedValue, setDebouncedValue] = useState<T>(value);
41
+
42
+ useEffect(() => {
43
+ const timer = setTimeout(() => setDebouncedValue(value), delay);
44
+ return () => clearTimeout(timer);
45
+ }, [value, delay]);
46
+
47
+ return debouncedValue;
48
+ }
49
+
50
+ // ============================================================================
51
+ // Storage Hooks
52
+ // ============================================================================
53
+
54
+ export function useLocalStorage<T>(
55
+ key: string,
56
+ initialValue: T,
57
+ ): [T, (value: T | ((prev: T) => T)) => void] {
58
+ const [storedValue, setStoredValue] = useState<T>(initialValue);
59
+
60
+ useEffect(() => {
61
+ if (typeof window !== "undefined") {
62
+ try {
63
+ const item = window.localStorage.getItem(key);
64
+ if (item) {
65
+ setStoredValue(JSON.parse(item));
66
+ }
67
+ } catch (error) {
68
+ console.warn(`Error reading localStorage key "${key}":`, error);
69
+ }
70
+ }
71
+ }, [key]);
72
+
73
+ const setValue = useCallback(
74
+ (value: T | ((prev: T) => T)) => {
75
+ try {
76
+ const valueToStore =
77
+ value instanceof Function ? value(storedValue) : value;
78
+ setStoredValue(valueToStore);
79
+ if (typeof window !== "undefined") {
80
+ window.localStorage.setItem(key, JSON.stringify(valueToStore));
81
+ }
82
+ } catch (error) {
83
+ console.warn(`Error setting localStorage key "${key}":`, error);
84
+ }
85
+ },
86
+ [key, storedValue],
87
+ );
88
+
89
+ return [storedValue, setValue];
90
+ }
91
+
92
+ // ============================================================================
93
+ // State Hooks
94
+ // ============================================================================
95
+
96
+ export function usePrevious<T>(value: T): T | undefined {
97
+ const ref = useRef<T>(undefined);
98
+
99
+ useEffect(() => {
100
+ ref.current = value;
101
+ }, [value]);
102
+
103
+ return ref.current;
104
+ }
105
+
106
+ export function useToggle(
107
+ initialValue = false,
108
+ ): [boolean, () => void, (value: boolean) => void] {
109
+ const [value, setValue] = useState(initialValue);
110
+ const toggle = useCallback(() => setValue((v) => !v), []);
111
+ return [value, toggle, setValue];
112
+ }
113
+
114
+ export function useMounted(): boolean {
115
+ const [mounted, setMounted] = useState(false);
116
+
117
+ useEffect(() => {
118
+ setMounted(true);
119
+ }, []);
120
+
121
+ return mounted;
122
+ }
123
+
124
+ // ============================================================================
125
+ // Media Hooks
126
+ // ============================================================================
127
+
128
+ export function useMediaQuery(query: string): boolean {
129
+ const [matches, setMatches] = useState(false);
130
+
131
+ useEffect(() => {
132
+ if (typeof window !== "undefined") {
133
+ const media = window.matchMedia(query);
134
+ setMatches(media.matches);
135
+
136
+ const listener = (event: MediaQueryListEvent) =>
137
+ setMatches(event.matches);
138
+ media.addEventListener("change", listener);
139
+ return () => media.removeEventListener("change", listener);
140
+ }
141
+ }, [query]);
142
+
143
+ return matches;
144
+ }
145
+
146
+ export function useMobile(): boolean {
147
+ return useMediaQuery("(max-width: 768px)");
148
+ }
149
+
150
+ // ============================================================================
151
+ // Clipboard Hook
152
+ // ============================================================================
153
+
154
+ export function useCopyToClipboard(): [
155
+ string | null,
156
+ (text: string) => Promise<boolean>,
157
+ ] {
158
+ const [copiedText, setCopiedText] = useState<string | null>(null);
159
+
160
+ const copy = useCallback(async (text: string): Promise<boolean> => {
161
+ if (!navigator?.clipboard) {
162
+ console.warn("Clipboard not supported");
163
+ return false;
164
+ }
165
+
166
+ try {
167
+ await navigator.clipboard.writeText(text);
168
+ setCopiedText(text);
169
+ return true;
170
+ } catch (error) {
171
+ console.warn("Copy failed", error);
172
+ setCopiedText(null);
173
+ return false;
174
+ }
175
+ }, []);
176
+
177
+ return [copiedText, copy];
178
+ }
179
+
180
+ // ============================================================================
181
+ // Settings Hooks
182
+ // ============================================================================
183
+
184
+ export function useSettings() {
185
+ const context = useContext(SettingsContext);
186
+ if (!context) {
187
+ return defaultSettingsValue;
188
+ }
189
+ return context;
190
+ }
191
+
192
+ export function useIsVertical() {
193
+ const { settings } = useSettings();
194
+ return settings.layout === "vertical";
195
+ }
196
+
197
+ export function useIsDarkMode() {
198
+ const { settings } = useSettings();
199
+ const isDarkModePreferred = useMedia("(prefers-color-scheme: dark)");
200
+
201
+ let resolvedMode = settings.mode;
202
+ if (resolvedMode === "system") {
203
+ resolvedMode = isDarkModePreferred ? "dark" : "light";
204
+ }
205
+
206
+ return resolvedMode === "dark";
207
+ }
208
+
209
+ export function useRadius(asPx = true) {
210
+ const { settings } = useSettings();
211
+
212
+ let radius = Number(settings.radius);
213
+ if (asPx) {
214
+ radius = remToPx(radius);
215
+ }
216
+
217
+ return radius;
218
+ }
219
+
220
+ export function useDensity() {
221
+ const { settings } = useSettings();
222
+ return settings.density;
223
+ }
224
+
225
+ export function useIsRtl() {
226
+ const direction = useDirection();
227
+ return direction === "rtl";
228
+ }
229
+
230
+ // ============================================================================
231
+ // View Mode Hook
232
+ // ============================================================================
233
+
234
+ export function useViewMode(
235
+ storageKey: string,
236
+ defaultMode: "table" | "kanban" | "grid" = "table",
237
+ ) {
238
+ const [viewMode, setViewMode] = useState<"table" | "kanban" | "grid">(
239
+ defaultMode,
240
+ );
241
+
242
+ useEffect(() => {
243
+ const saved = localStorage.getItem(storageKey);
244
+ if (saved === "kanban" || saved === "table" || saved === "grid") {
245
+ setViewMode(saved as "table" | "kanban" | "grid");
246
+ }
247
+ }, [storageKey]);
248
+
249
+ const handleViewModeChange = (mode: "table" | "kanban" | "grid") => {
250
+ setViewMode(mode);
251
+ localStorage.setItem(storageKey, mode);
252
+ };
253
+
254
+ return [viewMode, handleViewModeChange] as const;
255
+ }
256
+
257
+ // ============================================================================
258
+ // Dictionary Hooks
259
+ // ============================================================================
260
+
261
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
262
+ export type DictionaryType = any;
263
+
264
+ const DictionaryContext = createContext<DictionaryType | null>(null);
265
+
266
+ interface DictionaryProviderProps {
267
+ dictionary: DictionaryType;
268
+ children: ReactNode;
269
+ }
270
+
271
+ export function DictionaryProvider({
272
+ dictionary,
273
+ children,
274
+ }: DictionaryProviderProps) {
275
+ return (
276
+ <DictionaryContext.Provider value={dictionary}>
277
+ {children}
278
+ </DictionaryContext.Provider>
279
+ );
280
+ }
281
+
282
+ export function useDictionary() {
283
+ const dictionary = useContext(DictionaryContext);
284
+
285
+ if (!dictionary) {
286
+ throw new Error("useDictionary must be used within DictionaryProvider");
287
+ }
288
+
289
+ return dictionary;
290
+ }
291
+
292
+ export function useDictionaryValue(key: string, fallback?: string): string {
293
+ const dictionary = useDictionary();
294
+
295
+ const keys = key.split(".");
296
+ let value: unknown = dictionary;
297
+
298
+ for (const k of keys) {
299
+ if (value && typeof value === "object" && k in value) {
300
+ value = (value as Record<string, unknown>)[k];
301
+ } else {
302
+ return fallback || key;
303
+ }
304
+ }
305
+
306
+ return typeof value === "string" ? value : fallback || key;
307
+ }
308
+
309
+ // ============================================================================
310
+ // CRUD Translations Hook
311
+ // ============================================================================
312
+
313
+ export function useCrudTranslations() {
314
+ const params = useParams();
315
+ const locale = (params.lang as string) || "en";
316
+
317
+ const t = useCallback(
318
+ async (
319
+ key: string,
320
+ _params?: Record<string, string | number>,
321
+ ): Promise<string> => {
322
+ return key;
323
+ },
324
+ [],
325
+ );
326
+
327
+ const tSync = useCallback(
328
+ (
329
+ translations: DictionaryType,
330
+ key: string,
331
+ params?: Record<string, string | number>,
332
+ ): string => {
333
+ const keys = key.split(".");
334
+ let value: unknown = translations;
335
+
336
+ for (const k of keys) {
337
+ if (value && typeof value === "object" && k in value) {
338
+ value = (value as Record<string, unknown>)[k];
339
+ } else {
340
+ return key;
341
+ }
342
+ }
343
+
344
+ if (typeof value === "string") {
345
+ if (params) {
346
+ return value.replace(/\{\{(\w+)\}\}/g, (_, param) => {
347
+ return String(params[param] || "");
348
+ });
349
+ }
350
+ return value;
351
+ }
352
+
353
+ return key;
354
+ },
355
+ [],
356
+ );
357
+
358
+ return {
359
+ t,
360
+ tSync,
361
+ locale,
362
+ };
363
+ }
364
+
365
+ // ============================================================================
366
+ // Prefetch Hook
367
+ // ============================================================================
368
+
369
+ export function usePrefetch() {
370
+ const router = useRouter();
371
+ const prefetchedRoutes = useRef<Set<string>>(new Set());
372
+ const prefetchedAPIs = useRef<Set<string>>(new Set());
373
+
374
+ const prefetchRoute = useCallback(
375
+ (route: string) => {
376
+ if (prefetchedRoutes.current.has(route)) {
377
+ return;
378
+ }
379
+
380
+ try {
381
+ router.prefetch(route);
382
+ prefetchedRoutes.current.add(route);
383
+ } catch (error) {
384
+ console.debug("Prefetch failed for route:", route, error);
385
+ }
386
+ },
387
+ [router],
388
+ );
389
+
390
+ const prefetchAPI = useCallback((endpoint: string, options?: RequestInit) => {
391
+ const url = endpoint;
392
+ const cacheKey = `${url}-${JSON.stringify(options || {})}`;
393
+
394
+ if (prefetchedAPIs.current.has(cacheKey)) {
395
+ return;
396
+ }
397
+
398
+ try {
399
+ fetch(url, {
400
+ method: "HEAD",
401
+ ...options,
402
+ headers: {
403
+ ...options?.headers,
404
+ "Cache-Control": "max-age=60",
405
+ },
406
+ })
407
+ .then(() => {
408
+ prefetchedAPIs.current.add(cacheKey);
409
+ })
410
+ .catch(() => {});
411
+ } catch (error) {
412
+ console.debug("API prefetch failed:", url, error);
413
+ }
414
+ }, []);
415
+
416
+ const prefetchCrudPage = useCallback(
417
+ (entity: string, apiEndpoint: string) => {
418
+ const route = `/crud/${entity}`;
419
+ prefetchRoute(route);
420
+
421
+ const apiUrl = `${apiEndpoint}?page=1&pageSize=10`;
422
+ prefetchAPI(apiUrl);
423
+ },
424
+ [prefetchRoute, prefetchAPI],
425
+ );
426
+
427
+ const prefetchNextPage = useCallback(
428
+ (
429
+ apiEndpoint: string,
430
+ currentPage: number,
431
+ pageSize: number,
432
+ queryParams?: Record<string, unknown>,
433
+ ) => {
434
+ const nextPage = currentPage + 1;
435
+ const params = new URLSearchParams({
436
+ page: String(nextPage),
437
+ pageSize: String(pageSize),
438
+ });
439
+
440
+ if (queryParams) {
441
+ Object.entries(queryParams).forEach(([k, v]) => {
442
+ if (v === undefined || v === null || v === "") {
443
+ return;
444
+ }
445
+
446
+ if (typeof v === "object" && !Array.isArray(v)) {
447
+ params.append(k, JSON.stringify(v));
448
+ } else if (Array.isArray(v)) {
449
+ params.append(k, JSON.stringify(v));
450
+ } else {
451
+ params.append(k, String(v));
452
+ }
453
+ });
454
+ }
455
+
456
+ const apiUrl = `${apiEndpoint}?${params.toString()}`;
457
+ prefetchAPI(apiUrl);
458
+ },
459
+ [prefetchAPI],
460
+ );
461
+
462
+ return {
463
+ prefetchRoute,
464
+ prefetchAPI,
465
+ prefetchCrudPage,
466
+ prefetchNextPage,
467
+ };
468
+ }
469
+
470
+ // ============================================================================
471
+ // URL Filters Hook
472
+ // ============================================================================
473
+
474
+ interface UseUrlFiltersOptions {
475
+ filters?: {
476
+ search?: string;
477
+ dateRange?: { from?: Date; to?: Date };
478
+ materialGroups?: string[];
479
+ suppliers?: string[];
480
+ statuses?: string[];
481
+ warehouses?: string[];
482
+ };
483
+ }
484
+
485
+ export function useUrlFilters(options: UseUrlFiltersOptions = {}) {
486
+ const router = useRouter();
487
+ const pathname = usePathname();
488
+ const searchParams = useSearchParams();
489
+
490
+ const currentFilters = useMemo(() => {
491
+ return {
492
+ search: searchParams.get("search") || "",
493
+ dateRange: {
494
+ from: searchParams.get("from")
495
+ ? new Date(searchParams.get("from")!)
496
+ : undefined,
497
+ to: searchParams.get("to")
498
+ ? new Date(searchParams.get("to")!)
499
+ : undefined,
500
+ },
501
+ materialGroups:
502
+ searchParams.get("materialGroups")?.split(",").filter(Boolean) || [],
503
+ suppliers:
504
+ searchParams.get("suppliers")?.split(",").filter(Boolean) || [],
505
+ statuses: searchParams.get("statuses")?.split(",").filter(Boolean) || [],
506
+ warehouses:
507
+ searchParams.get("warehouses")?.split(",").filter(Boolean) || [],
508
+ };
509
+ }, [searchParams]);
510
+
511
+ const updateFilters = useCallback(
512
+ (filters: UseUrlFiltersOptions["filters"]) => {
513
+ const params = new URLSearchParams(searchParams.toString());
514
+
515
+ if (filters?.search) {
516
+ params.set("search", filters.search);
517
+ } else {
518
+ params.delete("search");
519
+ }
520
+
521
+ if (filters?.dateRange?.from) {
522
+ params.set("from", filters.dateRange.from.toISOString());
523
+ } else {
524
+ params.delete("from");
525
+ }
526
+
527
+ if (filters?.dateRange?.to) {
528
+ params.set("to", filters.dateRange.to.toISOString());
529
+ } else {
530
+ params.delete("to");
531
+ }
532
+
533
+ if (filters?.materialGroups && filters.materialGroups.length > 0) {
534
+ params.set("materialGroups", filters.materialGroups.join(","));
535
+ } else {
536
+ params.delete("materialGroups");
537
+ }
538
+
539
+ if (filters?.suppliers && filters.suppliers.length > 0) {
540
+ params.set("suppliers", filters.suppliers.join(","));
541
+ } else {
542
+ params.delete("suppliers");
543
+ }
544
+
545
+ if (filters?.statuses && filters.statuses.length > 0) {
546
+ params.set("statuses", filters.statuses.join(","));
547
+ } else {
548
+ params.delete("statuses");
549
+ }
550
+
551
+ if (filters?.warehouses && filters.warehouses.length > 0) {
552
+ params.set("warehouses", filters.warehouses.join(","));
553
+ } else {
554
+ params.delete("warehouses");
555
+ }
556
+
557
+ router.push(`${pathname}?${params.toString()}`);
558
+ },
559
+ [router, pathname, searchParams],
560
+ );
561
+
562
+ const clearFilters = useCallback(() => {
563
+ router.push(pathname);
564
+ }, [router, pathname]);
565
+
566
+ return {
567
+ currentFilters,
568
+ updateFilters,
569
+ clearFilters,
570
+ };
571
+ }
572
+
573
+
574
+
575
+ // ============================================================================
576
+ // Tab Navigation Back Hook
577
+ // ============================================================================
578
+
579
+ // Re-export from layout for use in hooks
580
+ export { useTabNavigation } from "../ui/layout";
581
+
582
+ /**
583
+ * Hook to handle back navigation that integrates with the tab system.
584
+ */
585
+ export function useTabNavigationBack() {
586
+ // Import dynamically to avoid circular dependency issues at module level
587
+ const { useTabNavigation } = require("../ui/layout");
588
+ const { closeAndGoToParent } = useTabNavigation();
589
+
590
+ return {
591
+ navigateBack: closeAndGoToParent,
592
+ };
593
+ }
594
+ // ============================================================================
595
+ // Tenant Hooks
596
+ // ============================================================================
597
+
598
+ export * from "./use-tenant";
@@ -0,0 +1,30 @@
1
+ import { renderHook } from "@testing-library/react";
2
+ import { describe, it, expect } from "vitest";
3
+ import { useTenant } from "./use-tenant";
4
+ import { TenantProvider } from "../providers/tenant-provider";
5
+
6
+ describe("useTenant", () => {
7
+ const mockTenant = {
8
+ id: "test-tenant",
9
+ name: "Test Tenant",
10
+ branding: {
11
+ companyName: "Test Company",
12
+ },
13
+ };
14
+
15
+ it("should return tenant context when used within TenantProvider", () => {
16
+ const wrapper = ({ children }: { children: React.ReactNode }) => (
17
+ <TenantProvider tenant={mockTenant}>{children}</TenantProvider>
18
+ );
19
+
20
+ const { result } = renderHook(() => useTenant(), { wrapper });
21
+ expect(result.current.tenant).toEqual(mockTenant);
22
+ });
23
+
24
+ it("should throw error when used outside TenantProvider", () => {
25
+ // Assert that rendering the hook throws an error
26
+ expect(() => {
27
+ renderHook(() => useTenant());
28
+ }).toThrow("useTenantContext must be used within a TenantProvider");
29
+ });
30
+ });
@@ -0,0 +1,5 @@
1
+ import { useTenantContext } from "../providers/tenant-provider";
2
+
3
+ export function useTenant() {
4
+ return useTenantContext();
5
+ }
package/src/index.ts ADDED
@@ -0,0 +1,17 @@
1
+ // @goerp/core - Main Entry Point
2
+ // ERP Core Library for building enterprise applications
3
+
4
+ // Re-export core modules (non-conflicting)
5
+ export * from "./system";
6
+ export * from "./types";
7
+
8
+ // Note: Utils and Configs should be imported from their subpaths to avoid conflicts:
9
+ // import { cn, formatDate } from "@goerp/core/utils"
10
+ // import { i18n, crudConfig } from "@goerp/core/configs"
11
+
12
+ // React components are exported from their respective subpaths:
13
+ // import { AppShell } from "@goerp/core/layout"
14
+ // import { Button } from "@goerp/core/ui"
15
+ // import { useToast } from "@goerp/core/hooks"
16
+ // import { CrudPage } from "@goerp/core/crud"
17
+ // import { withAuth } from "@goerp/core/auth"