@tulip-systems/core 0.2.1 → 0.4.0

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 (368) hide show
  1. package/dist/auth/client.d.ts +3 -1
  2. package/dist/auth/client.js +4 -2
  3. package/dist/auth/server.d.ts +2 -2
  4. package/dist/auth/server.js +3 -3
  5. package/dist/auth.d.ts +2 -1
  6. package/dist/auth.js +3 -2
  7. package/dist/commands/client.d.ts +4 -4
  8. package/dist/commands/client.js +4 -4
  9. package/dist/components/client.d.ts +5 -3
  10. package/dist/components/client.js +6 -4
  11. package/dist/components/common/icons.d.ts +5 -2
  12. package/dist/components/common/icons.d.ts.map +1 -1
  13. package/dist/components/common/icons.js +4 -1
  14. package/dist/components/common/icons.js.map +1 -1
  15. package/dist/components/common/status.d.ts +3 -3
  16. package/dist/components/editor/components/content.client.d.ts +2 -2
  17. package/dist/components/editor/components/fixed-menu.client.d.ts +2 -2
  18. package/dist/components/editor/context.client.d.ts +2 -2
  19. package/dist/components/editor/index.client.d.ts +2 -2
  20. package/dist/components/editor/preview.client.d.ts +2 -2
  21. package/dist/components/header/back-button.client.d.ts +2 -2
  22. package/dist/components/header/bottom-bar.client.d.ts +5 -5
  23. package/dist/components/header/bottom-bar.client.d.ts.map +1 -1
  24. package/dist/components/header/bottom-bar.client.js.map +1 -1
  25. package/dist/components/header/breadcrumbs.client.d.ts +7 -7
  26. package/dist/components/header/breadcrumbs.client.js +1 -1
  27. package/dist/components/header/header.client.d.ts +2 -2
  28. package/dist/components/header/mobile-nav-switcher.client.d.ts +2 -2
  29. package/dist/components/header/top-bar.client.d.ts +4 -4
  30. package/dist/components/layouts/admin-content.client.d.ts +13 -0
  31. package/dist/components/layouts/admin-content.client.d.ts.map +1 -0
  32. package/dist/components/layouts/admin-layout.d.ts +8 -0
  33. package/dist/components/layouts/admin-layout.d.ts.map +1 -0
  34. package/dist/components/layouts/admin-layout.js +15 -0
  35. package/dist/components/layouts/admin-layout.js.map +1 -0
  36. package/dist/components/layouts/admin-loading.d.ts +3 -3
  37. package/dist/components/layouts/admin-loading.d.ts.map +1 -1
  38. package/dist/components/layouts/admin-loading.js +2 -2
  39. package/dist/components/layouts/admin-loading.js.map +1 -1
  40. package/dist/components/layouts/empty-page.d.ts +4 -4
  41. package/dist/components/layouts/list-layout.d.ts +2 -2
  42. package/dist/components/layouts/not-found-page.d.ts +4 -3
  43. package/dist/components/layouts/not-found-page.d.ts.map +1 -1
  44. package/dist/components/layouts/not-found-page.js +8 -2
  45. package/dist/components/layouts/not-found-page.js.map +1 -1
  46. package/dist/components/layouts/not-found-section.js +1 -1
  47. package/dist/components/layouts/not-found-section.js.map +1 -1
  48. package/dist/components/layouts/providers.client.d.ts +2 -2
  49. package/dist/components/layouts/root-layout.server.d.ts +2 -2
  50. package/dist/components/layouts/root-layout.server.d.ts.map +1 -1
  51. package/dist/components/layouts/root-loading.d.ts +7 -0
  52. package/dist/components/layouts/root-loading.d.ts.map +1 -0
  53. package/dist/components/layouts/root-loading.js +14 -0
  54. package/dist/components/layouts/root-loading.js.map +1 -0
  55. package/dist/components/layouts/tab-layout.d.ts +2 -2
  56. package/dist/components/lists/data-list.d.ts +5 -5
  57. package/dist/components/lists/data-stack.d.ts +8 -8
  58. package/dist/components/{layouts → navigation}/admin-sidebar-header.client.js +8 -9
  59. package/dist/components/navigation/admin-sidebar-header.client.js.map +1 -0
  60. package/dist/components/navigation/admin-sidebar-paths.client.d.ts +81 -0
  61. package/dist/components/navigation/admin-sidebar-paths.client.d.ts.map +1 -0
  62. package/dist/components/navigation/admin-sidebar-paths.client.js +143 -0
  63. package/dist/components/navigation/admin-sidebar-paths.client.js.map +1 -0
  64. package/dist/components/navigation/admin-sidebar-skeleton.js +31 -0
  65. package/dist/components/navigation/admin-sidebar-skeleton.js.map +1 -0
  66. package/dist/components/server.d.ts +1 -1
  67. package/dist/components/server.js +1 -1
  68. package/dist/components/ui/accordion.d.ts +5 -5
  69. package/dist/components/ui/alert-dialog.d.ts +12 -12
  70. package/dist/components/ui/alert.d.ts +6 -6
  71. package/dist/components/ui/aspect-ratio.d.ts +2 -2
  72. package/dist/components/ui/avatar.client.d.ts +4 -4
  73. package/dist/components/ui/badge.d.ts +4 -4
  74. package/dist/components/ui/breadcrumb.d.ts +8 -8
  75. package/dist/components/ui/button.d.ts +4 -4
  76. package/dist/components/ui/calendar.d.ts +2 -2
  77. package/dist/components/ui/card.d.ts +7 -7
  78. package/dist/components/ui/carousel.d.ts +6 -6
  79. package/dist/components/ui/chart.client.d.ts +5 -5
  80. package/dist/components/ui/checkbox.d.ts +2 -2
  81. package/dist/components/ui/collapsible.client.d.ts +4 -4
  82. package/dist/components/ui/color-picker.client.d.ts +2 -2
  83. package/dist/components/ui/combobox-dropdown.client.d.ts +2 -2
  84. package/dist/components/ui/combobox.client.d.ts +2 -2
  85. package/dist/components/ui/command.d.ts +10 -10
  86. package/dist/components/ui/context-menu.d.ts +16 -16
  87. package/dist/components/ui/dialog.client.d.ts +11 -11
  88. package/dist/components/ui/drawer.client.d.ts +11 -11
  89. package/dist/components/ui/dropdown-menu.d.ts +16 -16
  90. package/dist/components/ui/form.client.d.ts +8 -8
  91. package/dist/components/ui/hover-card.client.d.ts +4 -4
  92. package/dist/components/ui/input.d.ts +2 -2
  93. package/dist/components/ui/label.d.ts +2 -2
  94. package/dist/components/ui/navigation-menu.d.ts +11 -11
  95. package/dist/components/ui/pagination.d.ts +8 -8
  96. package/dist/components/ui/popover.d.ts +5 -5
  97. package/dist/components/ui/progress.client.d.ts +2 -2
  98. package/dist/components/ui/radio-group.d.ts +3 -3
  99. package/dist/components/ui/resizable.client.d.ts +4 -4
  100. package/dist/components/ui/scroll-area.d.ts +3 -3
  101. package/dist/components/ui/select.client.d.ts +11 -11
  102. package/dist/components/ui/separator.d.ts +2 -2
  103. package/dist/components/ui/sheet.client.d.ts +9 -9
  104. package/dist/components/ui/sidebar.client.d.ts +24 -24
  105. package/dist/components/ui/sidebar.client.js +1 -1
  106. package/dist/components/ui/skeleton.d.ts +2 -2
  107. package/dist/components/ui/slider.d.ts +2 -2
  108. package/dist/components/ui/sonner.client.d.ts +2 -2
  109. package/dist/components/ui/sonner.client.js +1 -1
  110. package/dist/components/ui/spinner.js +63 -0
  111. package/dist/components/ui/spinner.js.map +1 -0
  112. package/dist/components/ui/switch.d.ts +2 -2
  113. package/dist/components/ui/tabs.d.ts +5 -5
  114. package/dist/components/ui/textarea.d.ts +2 -2
  115. package/dist/components/ui/time-input.client.d.ts +2 -2
  116. package/dist/components/ui/toggle-group.client.d.ts +3 -3
  117. package/dist/components/ui/toggle.d.ts +4 -4
  118. package/dist/components/ui/tooltip.client.d.ts +5 -5
  119. package/dist/components/ui/tree.client.d.ts +5 -5
  120. package/dist/components/ui/tree.client.d.ts.map +1 -1
  121. package/dist/components.d.ts +6 -5
  122. package/dist/components.js +8 -7
  123. package/dist/config/server.d.ts +2 -3
  124. package/dist/config/server.js +3 -4
  125. package/dist/config.d.ts +3 -2
  126. package/dist/config.js +2 -1
  127. package/dist/lib/client.js +1 -1
  128. package/dist/lib/hooks/use-action.d.ts +2 -2
  129. package/dist/modules/auth/components/allowed.client.js +1 -1
  130. package/dist/modules/auth/components/allowed.client.js.map +1 -1
  131. package/dist/modules/auth/components/auth-layout.server.d.ts +3 -4
  132. package/dist/modules/auth/components/auth-layout.server.d.ts.map +1 -1
  133. package/dist/modules/auth/components/auth-layout.server.js +1 -4
  134. package/dist/modules/auth/components/auth-layout.server.js.map +1 -1
  135. package/dist/modules/auth/components/auth-loading.d.ts +7 -0
  136. package/dist/modules/auth/components/auth-loading.d.ts.map +1 -0
  137. package/dist/modules/auth/components/auth-loading.js +18 -0
  138. package/dist/modules/auth/components/auth-loading.js.map +1 -0
  139. package/dist/modules/auth/components/forget-password-page.client.js +2 -2
  140. package/dist/modules/auth/components/guard.client.js +16 -0
  141. package/dist/modules/auth/components/guard.client.js.map +1 -0
  142. package/dist/modules/auth/components/login-page.client.d.ts +1 -5
  143. package/dist/modules/auth/components/login-page.client.d.ts.map +1 -1
  144. package/dist/modules/auth/components/login-page.client.js +41 -31
  145. package/dist/modules/auth/components/login-page.client.js.map +1 -1
  146. package/dist/modules/auth/components/reset-password-page.client.js +3 -3
  147. package/dist/modules/auth/components/update-password-command.d.ts +1 -1
  148. package/dist/modules/auth/components/update-password-command.js +1 -1
  149. package/dist/modules/auth/handler/client.client.d.ts +48 -48
  150. package/dist/modules/auth/handler/client.client.d.ts.map +1 -1
  151. package/dist/modules/auth/handler/proxy.server.d.ts +13 -0
  152. package/dist/modules/auth/handler/proxy.server.d.ts.map +1 -0
  153. package/dist/modules/auth/handler/proxy.server.js +22 -0
  154. package/dist/modules/auth/handler/proxy.server.js.map +1 -0
  155. package/dist/modules/auth/handler/server.server.js +3 -1
  156. package/dist/modules/auth/handler/server.server.js.map +1 -1
  157. package/dist/modules/auth/hooks/use-permission.d.ts +26 -0
  158. package/dist/modules/auth/hooks/use-permission.d.ts.map +1 -0
  159. package/dist/modules/auth/hooks/{use-permission.client.js → use-permission.js} +5 -3
  160. package/dist/modules/auth/hooks/use-permission.js.map +1 -0
  161. package/dist/modules/auth/hooks/use-session.d.ts +34 -0
  162. package/dist/modules/auth/hooks/use-session.d.ts.map +1 -0
  163. package/dist/modules/auth/hooks/use-session.js +21 -0
  164. package/dist/modules/auth/hooks/use-session.js.map +1 -0
  165. package/dist/modules/auth/lib/helpers.server.d.ts +2 -2
  166. package/dist/modules/auth/lib/helpers.server.d.ts.map +1 -1
  167. package/dist/modules/auth/lib/helpers.server.js +3 -6
  168. package/dist/modules/auth/lib/helpers.server.js.map +1 -1
  169. package/dist/modules/auth/lib/validators.d.ts +4 -4
  170. package/dist/modules/auth/lib/validators.d.ts.map +1 -1
  171. package/dist/modules/auth/lib/validators.js +1 -1
  172. package/dist/modules/auth/lib/validators.js.map +1 -1
  173. package/dist/modules/commands/components/alert-dialog-command.client.d.ts +11 -11
  174. package/dist/modules/commands/components/alert-dialog-command.client.js +1 -1
  175. package/dist/modules/commands/components/click-command.client.d.ts +2 -2
  176. package/dist/modules/commands/components/command-trigger.client.d.ts +4 -4
  177. package/dist/modules/commands/components/command-trigger.client.js +1 -1
  178. package/dist/modules/commands/components/dialog-command.client.d.ts +8 -8
  179. package/dist/modules/commands/components/dialog-command.client.js +1 -1
  180. package/dist/modules/commands/components/dropdown-command.client.d.ts +5 -5
  181. package/dist/modules/commands/components/empty-command.client.d.ts +2 -2
  182. package/dist/modules/commands/components/form-dialog-command.client.d.ts +12 -12
  183. package/dist/modules/commands/components/form-dialog-command.client.js +1 -1
  184. package/dist/modules/commands/hooks/use-command-action.client.js +1 -1
  185. package/dist/modules/commands/hooks/use-command-menu.client.js +1 -1
  186. package/dist/modules/commands/hooks/use-command-menu.client.js.map +1 -1
  187. package/dist/modules/commands/menus/context-menu.client.d.ts +2 -2
  188. package/dist/modules/commands/menus/dropdown-menu.client.d.ts +8 -4
  189. package/dist/modules/commands/menus/dropdown-menu.client.d.ts.map +1 -1
  190. package/dist/modules/commands/menus/dropdown-menu.client.js +9 -1
  191. package/dist/modules/commands/menus/dropdown-menu.client.js.map +1 -1
  192. package/dist/modules/commands/menus/inline-menu.client.d.ts +8 -4
  193. package/dist/modules/commands/menus/inline-menu.client.d.ts.map +1 -1
  194. package/dist/modules/commands/menus/inline-menu.client.js +12 -1
  195. package/dist/modules/commands/menus/inline-menu.client.js.map +1 -1
  196. package/dist/modules/commands/menus/responsive-menu.client.d.ts +2 -1
  197. package/dist/modules/commands/menus/responsive-menu.client.d.ts.map +1 -1
  198. package/dist/modules/commands/menus/responsive-menu.client.js +6 -3
  199. package/dist/modules/commands/menus/responsive-menu.client.js.map +1 -1
  200. package/dist/modules/commands/utils/archive-command.client.d.ts +3 -3
  201. package/dist/modules/commands/utils/delete-command.client.d.ts +2 -2
  202. package/dist/modules/commands/utils/send-mail-command.client.d.ts +2 -2
  203. package/dist/modules/commands/utils/send-mail-command.client.d.ts.map +1 -1
  204. package/dist/modules/commands/utils/send-mail-command.client.js +3 -3
  205. package/dist/modules/config/db/helpers.d.ts +5 -5
  206. package/dist/modules/config/db/helpers.d.ts.map +1 -1
  207. package/dist/modules/config/lib/config.server.d.ts +0 -2
  208. package/dist/modules/config/lib/config.server.d.ts.map +1 -1
  209. package/dist/modules/config/lib/config.server.js.map +1 -1
  210. package/dist/modules/config/lib/resolve.server.d.ts +192 -190
  211. package/dist/modules/config/lib/resolve.server.d.ts.map +1 -1
  212. package/dist/modules/config/paths/helpers.d.ts +11 -0
  213. package/dist/modules/config/paths/helpers.d.ts.map +1 -0
  214. package/dist/modules/config/paths/helpers.js +9 -0
  215. package/dist/modules/config/paths/helpers.js.map +1 -0
  216. package/dist/modules/config/paths/types.d.ts +8 -37
  217. package/dist/modules/config/paths/types.d.ts.map +1 -1
  218. package/dist/modules/config/router/client.server.d.ts +4 -4
  219. package/dist/modules/config/router/client.server.d.ts.map +1 -1
  220. package/dist/modules/config/router/context.server.d.ts +194 -192
  221. package/dist/modules/config/router/context.server.d.ts.map +1 -1
  222. package/dist/modules/config/router/init.server.d.ts +200 -198
  223. package/dist/modules/config/router/init.server.d.ts.map +1 -1
  224. package/dist/modules/data-tables/components/cell/common.client.d.ts +5 -5
  225. package/dist/modules/data-tables/components/cell/common.client.d.ts.map +1 -1
  226. package/dist/modules/data-tables/components/column-header.d.ts +2 -2
  227. package/dist/modules/data-tables/components/filters/combobox.client.d.ts +2 -2
  228. package/dist/modules/data-tables/components/filters/combobox.client.d.ts.map +1 -1
  229. package/dist/modules/data-tables/components/filters/combobox.client.js +1 -1
  230. package/dist/modules/data-tables/components/filters/combobox.client.js.map +1 -1
  231. package/dist/modules/data-tables/components/filters/slider.client.d.ts +2 -2
  232. package/dist/modules/data-tables/components/filters/slider.client.d.ts.map +1 -1
  233. package/dist/modules/data-tables/components/header.d.ts +4 -4
  234. package/dist/modules/data-tables/components/layout.d.ts +2 -2
  235. package/dist/modules/data-tables/components/search-input.client.d.ts +2 -2
  236. package/dist/modules/data-tables/components/search-input.client.d.ts.map +1 -1
  237. package/dist/modules/data-tables/components/skeleton.d.ts +2 -2
  238. package/dist/modules/data-tables/components/table.d.ts +7 -7
  239. package/dist/modules/data-tables/components/toolbar.d.ts +3 -3
  240. package/dist/modules/data-tables/hooks/use-context.client.d.ts +2 -2
  241. package/dist/modules/data-tables/hooks/use-context.client.d.ts.map +1 -1
  242. package/dist/modules/data-tables/lib/filters/resolvers.d.ts +1 -1
  243. package/dist/modules/data-tables/lib/filters/resolvers.d.ts.map +1 -1
  244. package/dist/modules/data-tables/lib/search-params.d.ts +5 -5
  245. package/dist/modules/data-tables/lib/search-params.d.ts.map +1 -1
  246. package/dist/modules/data-tables/lib/types.d.ts +1 -1
  247. package/dist/modules/data-tables/tables/data-table/components/table.d.ts +2 -2
  248. package/dist/modules/data-tables/tables/inline-table/components/cells/common.d.ts +2 -2
  249. package/dist/modules/data-tables/tables/inline-table/components/cells/common.d.ts.map +1 -1
  250. package/dist/modules/data-tables/tables/inline-table/components/inputs/advanced-select.client.d.ts +2 -2
  251. package/dist/modules/data-tables/tables/inline-table/components/inputs/advanced-select.client.d.ts.map +1 -1
  252. package/dist/modules/data-tables/tables/inline-table/components/inputs/combobox.client.d.ts +2 -2
  253. package/dist/modules/data-tables/tables/inline-table/components/inputs/input.client.d.ts +3 -3
  254. package/dist/modules/data-tables/tables/inline-table/components/inputs/read-only.d.ts +2 -2
  255. package/dist/modules/data-tables/tables/inline-table/components/inputs/select.client.d.ts +2 -2
  256. package/dist/modules/data-tables/tables/inline-table/components/table.d.ts +2 -2
  257. package/dist/modules/data-tables/tables/inline-table/hooks/context.client.d.ts +2 -2
  258. package/dist/modules/data-tables/tables/inline-table/hooks/context.client.js +1 -1
  259. package/dist/modules/data-tables/tables/inline-table/hooks/use-input.client.js +1 -1
  260. package/dist/modules/data-tables/tables/inline-table/hooks/use-input.client.js.map +1 -1
  261. package/dist/modules/inline-edit/components/combobox-dropdown.client.d.ts +2 -2
  262. package/dist/modules/inline-edit/components/combobox-dropdown.client.d.ts.map +1 -1
  263. package/dist/modules/inline-edit/components/combobox.client.d.ts +2 -2
  264. package/dist/modules/inline-edit/components/combobox.client.d.ts.map +1 -1
  265. package/dist/modules/inline-edit/components/date-input.client.d.ts +2 -2
  266. package/dist/modules/inline-edit/components/date-input.client.js +1 -1
  267. package/dist/modules/inline-edit/components/date-picker.client.d.ts +2 -2
  268. package/dist/modules/inline-edit/components/editor.client.d.ts +2 -2
  269. package/dist/modules/inline-edit/components/input-toggle.client.d.ts +2 -2
  270. package/dist/modules/inline-edit/components/input.client.d.ts +3 -3
  271. package/dist/modules/inline-edit/components/richtext.client.d.ts +2 -2
  272. package/dist/modules/inline-edit/components/select.client.d.ts +6 -6
  273. package/dist/modules/inline-edit/components/switch.client.d.ts +2 -2
  274. package/dist/modules/inline-edit/components/toggle.client.d.ts +2 -2
  275. package/dist/modules/inline-edit/hooks/context.client.d.ts +2 -2
  276. package/dist/modules/inline-edit/hooks/use-inline.client.js +1 -1
  277. package/dist/modules/inline-edit/hooks/use-inline.client.js.map +1 -1
  278. package/dist/modules/uploads/components/button.client.d.ts +2 -2
  279. package/dist/modules/uploads/components/button.client.d.ts.map +1 -1
  280. package/dist/modules/uploads/components/images/grid.client.d.ts +2 -2
  281. package/dist/modules/uploads/components/images/grid.client.d.ts.map +1 -1
  282. package/dist/modules/uploads/components/images/grid.client.js +1 -1
  283. package/dist/modules/uploads/components/images/input.client.d.ts +2 -2
  284. package/dist/modules/uploads/components/images/input.client.d.ts.map +1 -1
  285. package/dist/modules/uploads/components/preview-dialog.client.d.ts +2 -2
  286. package/dist/modules/uploads/components/preview-dialog.client.d.ts.map +1 -1
  287. package/dist/modules/uploads/components/zone.client.d.ts +2 -2
  288. package/dist/modules/uploads/components/zone.client.d.ts.map +1 -1
  289. package/dist/modules/uploads/hooks/use-upload-image.client.js +1 -1
  290. package/dist/modules/uploads/lib/proxy.server.d.ts +1 -1
  291. package/dist/modules/uploads/lib/proxy.server.js +4 -4
  292. package/dist/modules/uploads/lib/route.server.d.ts +1 -1
  293. package/dist/modules/uploads/lib/route.server.js +2 -2
  294. package/dist/modules/uploads/lib/router.server.d.ts +965 -965
  295. package/dist/modules/uploads/lib/router.server.d.ts.map +1 -1
  296. package/dist/modules/uploads/lib/router.server.js +2 -2
  297. package/dist/modules/uploads/lib/schema.d.ts +61 -61
  298. package/dist/modules/uploads/lib/validators.d.ts +56 -56
  299. package/dist/modules/uploads/lib/validators.d.ts.map +1 -1
  300. package/package.json +5 -4
  301. package/src/components/common/icons.tsx +6 -0
  302. package/src/components/entry.client.ts +6 -0
  303. package/src/components/entry.server.ts +1 -1
  304. package/src/components/entry.ts +1 -0
  305. package/src/components/header/bottom-bar.client.tsx +2 -2
  306. package/src/components/layouts/admin-layout.tsx +13 -0
  307. package/src/components/layouts/admin-loading.tsx +1 -1
  308. package/src/components/layouts/not-found-page.tsx +9 -1
  309. package/src/components/layouts/not-found-section.tsx +10 -5
  310. package/src/components/layouts/root-loading.tsx +9 -0
  311. package/src/components/{layouts → navigation}/admin-sidebar-header.client.tsx +11 -6
  312. package/src/components/navigation/admin-sidebar-paths.client.tsx +190 -0
  313. package/src/components/navigation/admin-sidebar-skeleton.tsx +44 -0
  314. package/src/components/navigation/admin-sidebar.client.tsx +123 -0
  315. package/src/components/ui/spinner.tsx +51 -0
  316. package/src/modules/auth/components/allowed.client.tsx +1 -1
  317. package/src/modules/auth/components/auth-layout.server.tsx +2 -10
  318. package/src/modules/auth/components/auth-loading.tsx +11 -0
  319. package/src/modules/auth/components/guard.client.tsx +12 -0
  320. package/src/modules/auth/components/login-page.client.tsx +45 -37
  321. package/src/modules/auth/entry.client.ts +2 -0
  322. package/src/modules/auth/entry.server.ts +1 -1
  323. package/src/modules/auth/entry.ts +1 -0
  324. package/src/modules/auth/handler/proxy.server.ts +28 -0
  325. package/src/modules/auth/handler/server.server.ts +5 -0
  326. package/src/modules/auth/hooks/{use-permission.client.ts → use-permission.ts} +4 -2
  327. package/src/modules/auth/hooks/use-session.ts +18 -0
  328. package/src/modules/auth/lib/helpers.server.ts +1 -5
  329. package/src/modules/auth/lib/validators.ts +1 -1
  330. package/src/modules/commands/hooks/use-command-menu.client.tsx +6 -7
  331. package/src/modules/commands/menus/dropdown-menu.client.tsx +11 -2
  332. package/src/modules/commands/menus/inline-menu.client.tsx +14 -1
  333. package/src/modules/commands/menus/responsive-menu.client.tsx +11 -2
  334. package/src/modules/config/entry.server.ts +1 -2
  335. package/src/modules/config/entry.ts +3 -0
  336. package/src/modules/config/lib/config.server.ts +1 -2
  337. package/src/modules/config/lib/router.server.ts +64 -0
  338. package/src/modules/config/paths/helpers.ts +6 -0
  339. package/src/modules/config/paths/types.ts +26 -38
  340. package/src/modules/data-tables/components/filters/combobox.client.tsx +1 -1
  341. package/src/modules/data-tables/tables/inline-table/hooks/use-input.client.ts +1 -1
  342. package/src/modules/inline-edit/hooks/use-inline.client.tsx +1 -1
  343. package/dist/components/layouts/admin-layout.server.d.ts +0 -14
  344. package/dist/components/layouts/admin-layout.server.d.ts.map +0 -1
  345. package/dist/components/layouts/admin-layout.server.js +0 -26
  346. package/dist/components/layouts/admin-layout.server.js.map +0 -1
  347. package/dist/components/layouts/admin-sidebar-context.client.js +0 -28
  348. package/dist/components/layouts/admin-sidebar-context.client.js.map +0 -1
  349. package/dist/components/layouts/admin-sidebar-header.client.js.map +0 -1
  350. package/dist/components/layouts/admin-sidebar.client.js +0 -51
  351. package/dist/components/layouts/admin-sidebar.client.js.map +0 -1
  352. package/dist/modules/auth/components/allowed.server.d.ts +0 -23
  353. package/dist/modules/auth/components/allowed.server.d.ts.map +0 -1
  354. package/dist/modules/auth/components/allowed.server.js +0 -24
  355. package/dist/modules/auth/components/allowed.server.js.map +0 -1
  356. package/dist/modules/auth/hooks/use-permission.client.js.map +0 -1
  357. package/dist/modules/config/paths/create.server.d.ts +0 -14
  358. package/dist/modules/config/paths/create.server.d.ts.map +0 -1
  359. package/dist/modules/config/paths/create.server.js +0 -12
  360. package/dist/modules/config/paths/create.server.js.map +0 -1
  361. package/dist/modules/config/paths/get.server.js +0 -47
  362. package/dist/modules/config/paths/get.server.js.map +0 -1
  363. package/src/components/layouts/admin-layout.server.tsx +0 -31
  364. package/src/components/layouts/admin-sidebar-context.client.tsx +0 -34
  365. package/src/components/layouts/admin-sidebar.client.tsx +0 -88
  366. package/src/modules/auth/components/allowed.server.tsx +0 -41
  367. package/src/modules/config/paths/create.server.ts +0 -10
  368. package/src/modules/config/paths/get.server.ts +0 -69
@@ -14,19 +14,19 @@ import {
14
14
  } from "@/components/ui/dropdown-menu";
15
15
  import { cn } from "@/lib/utils/cn";
16
16
  import { authClient } from "@/modules/auth/handler/client.client";
17
+ import { useSession } from "@/modules/auth/hooks/use-session";
17
18
  import { ChevronsUpDownIcon, LaptopIcon, LogOutIcon, MoonIcon, SunIcon } from "lucide-react";
18
19
  import { useTheme } from "next-themes";
19
20
  import { useRouter } from "next/navigation";
20
- import { useContext } from "react";
21
21
  import { SidebarHeader, useSidebar } from "../entry.client";
22
- import { AdminSidebarContext } from "./admin-sidebar-context.client";
23
22
 
24
23
  export function AdminSidebarHeader() {
25
24
  const router = useRouter();
26
25
  const { theme, setTheme } = useTheme();
27
26
 
28
27
  const { setOpenMobile } = useSidebar();
29
- const { session } = useContext(AdminSidebarContext);
28
+
29
+ const sessionQuery = useSession();
30
30
 
31
31
  return (
32
32
  <SidebarHeader>
@@ -35,10 +35,15 @@ export function AdminSidebarHeader() {
35
35
  <Button variant="outline" className="items-center justify-between py-3.5">
36
36
  <div className="flex items-center gap-4">
37
37
  <Avatar className="h-6 w-6 text-sm">
38
- <AvatarImage src={session.user.image ?? undefined} alt={session.user.name} />
39
- <AvatarFallback>{session.user.name?.[0] ?? ""}</AvatarFallback>
38
+ <AvatarImage
39
+ src={sessionQuery.data?.user.image ?? undefined}
40
+ alt={sessionQuery.data?.user.name}
41
+ />
42
+ <AvatarFallback>{sessionQuery.data?.user.name?.[0] ?? ""}</AvatarFallback>
40
43
  </Avatar>
41
- <span className="text-md text-muted-foreground font-bold">{session.user.name}</span>
44
+ <span className="text-md text-muted-foreground font-bold">
45
+ {sessionQuery.data?.user.name}
46
+ </span>
42
47
  </div>
43
48
 
44
49
  <ChevronsUpDownIcon className="text-muted-foreground h-4 w-4" />
@@ -0,0 +1,190 @@
1
+ "use client";
2
+
3
+ import {
4
+ Sidebar,
5
+ SidebarContent,
6
+ SidebarGroup,
7
+ SidebarGroupContent,
8
+ SidebarGroupLabel,
9
+ SidebarMenu,
10
+ SidebarMenuButton,
11
+ SidebarMenuItem,
12
+ SidebarMenuSub,
13
+ SidebarMenuSubButton,
14
+ SidebarMenuSubItem,
15
+ SidebarRail,
16
+ useSidebar,
17
+ } from "@/components/ui/sidebar.client";
18
+ import { Allowed } from "@/modules/auth/components/allowed.client";
19
+ import { Permission } from "@/modules/auth/entry";
20
+ import { Route } from "next";
21
+ import Link from "next/link";
22
+ import { useSelectedLayoutSegments } from "next/navigation";
23
+ import * as React from "react";
24
+ import { Suspense } from "react";
25
+ import { AdminSidebarHeader } from "./admin-sidebar-header.client";
26
+ import {
27
+ AdminSidebarHeaderLoading,
28
+ AdminSidebarMenuContentLoading,
29
+ } from "./admin-sidebar-skeleton";
30
+
31
+ /**
32
+ * Paths
33
+ */
34
+ export function Paths({ children, ...props }: React.ComponentProps<typeof Sidebar>) {
35
+ return (
36
+ <Sidebar {...props}>
37
+ <Suspense fallback={<AdminSidebarHeaderLoading />}>
38
+ <AdminSidebarHeader />
39
+ </Suspense>
40
+
41
+ <SidebarContent>
42
+ <Suspense fallback={<AdminSidebarMenuContentLoading />}>
43
+ <SidebarMenu>{children}</SidebarMenu>
44
+ </Suspense>
45
+ </SidebarContent>
46
+ <SidebarRail />
47
+ </Sidebar>
48
+ );
49
+ }
50
+
51
+ /**
52
+ * Path Group
53
+ */
54
+ export function PathGroup(props: React.ComponentProps<"div">) {
55
+ return <SidebarGroup {...props} />;
56
+ }
57
+
58
+ /**
59
+ * Path Group Title
60
+ */
61
+ export function PathGroupTitle(props: React.ComponentProps<"div">) {
62
+ return <SidebarGroupLabel {...props} />;
63
+ }
64
+
65
+ /**
66
+ * Path Group Items
67
+ */
68
+ export function PathGroupItems(props: React.PropsWithChildren) {
69
+ return (
70
+ <SidebarGroupContent>
71
+ <SidebarMenu>{props.children}</SidebarMenu>
72
+ </SidebarGroupContent>
73
+ );
74
+ }
75
+
76
+ /**
77
+ * Path Context
78
+ */
79
+ const PathContext = React.createContext({} as { isActive: boolean });
80
+
81
+ /**
82
+ * Path
83
+ */
84
+ type PathProps = React.ComponentProps<"li"> & {
85
+ permission?: Permission;
86
+ isActive?: boolean;
87
+ segment?: string;
88
+ };
89
+
90
+ export function Path({ permission, children, isActive, segment, ...props }: PathProps) {
91
+ const currentSegments = useSelectedLayoutSegments();
92
+
93
+ if (!permission) {
94
+ return (
95
+ <PathContext value={{ isActive: isActive || segment === currentSegments?.[0] }}>
96
+ <SidebarMenuItem {...props}>{children}</SidebarMenuItem>
97
+ </PathContext>
98
+ );
99
+ }
100
+
101
+ return (
102
+ <Allowed permission={permission}>
103
+ <PathContext value={{ isActive: isActive || segment === currentSegments?.[0] }}>
104
+ <SidebarMenuItem {...props}>{children}</SidebarMenuItem>
105
+ </PathContext>
106
+ </Allowed>
107
+ );
108
+ }
109
+
110
+ /**
111
+ * Path Button
112
+ */
113
+ export function PathLink({ href, children }: React.PropsWithChildren<{ href: Route }>) {
114
+ const { isMobile, toggleSidebar } = useSidebar();
115
+ const { isActive } = React.useContext(PathContext);
116
+
117
+ return (
118
+ <SidebarMenuButton asChild isActive={isActive}>
119
+ <Link href={href} onClick={isMobile ? toggleSidebar : undefined}>
120
+ {children}
121
+ </Link>
122
+ </SidebarMenuButton>
123
+ );
124
+ }
125
+
126
+ /**
127
+ * Path Items
128
+ */
129
+ export function PathItems(props: React.ComponentProps<"ul">) {
130
+ const { isActive } = React.useContext(PathContext);
131
+ if (!isActive) return null;
132
+
133
+ return <SidebarMenuSub {...props} />;
134
+ }
135
+
136
+ /**
137
+ * Path Sub Item Context
138
+ */
139
+ const PathSubItemContext = React.createContext({} as { isActive: boolean });
140
+
141
+ /**
142
+ * Path Sub Item
143
+ */
144
+ type PathSubItemProps = React.ComponentProps<"li"> & {
145
+ permission?: Permission;
146
+ isActive?: boolean;
147
+ segment?: string;
148
+ };
149
+
150
+ export function PathSubItem({
151
+ permission,
152
+ isActive,
153
+ segment,
154
+ children,
155
+ ...props
156
+ }: PathSubItemProps) {
157
+ const currentSegments = useSelectedLayoutSegments();
158
+
159
+ if (!permission) {
160
+ return (
161
+ <PathSubItemContext value={{ isActive: isActive || segment === currentSegments?.[1] }}>
162
+ <SidebarMenuSubItem {...props}>{children}</SidebarMenuSubItem>
163
+ </PathSubItemContext>
164
+ );
165
+ }
166
+
167
+ return (
168
+ <Allowed permission={permission}>
169
+ <PathSubItemContext value={{ isActive: isActive || segment === currentSegments?.[1] }}>
170
+ <SidebarMenuSubItem {...props}>{children}</SidebarMenuSubItem>
171
+ </PathSubItemContext>
172
+ </Allowed>
173
+ );
174
+ }
175
+
176
+ /**
177
+ * Path Button
178
+ */
179
+ export function PathSubItemLink({ href, children }: React.PropsWithChildren<{ href: Route }>) {
180
+ const { isMobile, toggleSidebar } = useSidebar();
181
+ const { isActive } = React.useContext(PathSubItemContext);
182
+
183
+ return (
184
+ <SidebarMenuSubButton asChild isActive={isActive}>
185
+ <Link href={href} onClick={isMobile ? toggleSidebar : undefined}>
186
+ {children}
187
+ </Link>
188
+ </SidebarMenuSubButton>
189
+ );
190
+ }
@@ -0,0 +1,44 @@
1
+ import { Button } from "@/components/ui/button";
2
+ import { DropdownMenu, DropdownMenuTrigger } from "@/components/ui/dropdown-menu";
3
+ import {
4
+ SidebarHeader,
5
+ SidebarMenu,
6
+ SidebarMenuItem,
7
+ SidebarMenuSkeleton,
8
+ } from "@/components/ui/sidebar.client";
9
+ import { Skeleton } from "@/components/ui/skeleton";
10
+ import { Suspense } from "react";
11
+
12
+ /**
13
+ * SidebarMenuContentLoading
14
+ */
15
+ export function AdminSidebarMenuContentLoading() {
16
+ return (
17
+ <SidebarMenu>
18
+ {Array.from({ length: 20 }).map((_, index) => (
19
+ <SidebarMenuItem key={index}>
20
+ <Suspense>
21
+ <SidebarMenuSkeleton />
22
+ </Suspense>
23
+ </SidebarMenuItem>
24
+ ))}
25
+ </SidebarMenu>
26
+ );
27
+ }
28
+
29
+ /**
30
+ *
31
+ */
32
+ export function AdminSidebarHeaderLoading() {
33
+ return (
34
+ <SidebarHeader>
35
+ <DropdownMenu>
36
+ <DropdownMenuTrigger asChild>
37
+ <Button variant="outline" disabled>
38
+ <Skeleton className="h-5 w-full" />
39
+ </Button>
40
+ </DropdownMenuTrigger>
41
+ </DropdownMenu>
42
+ </SidebarHeader>
43
+ );
44
+ }
@@ -0,0 +1,123 @@
1
+ "use client";
2
+
3
+ import {
4
+ Sidebar,
5
+ SidebarContent,
6
+ SidebarGroup,
7
+ SidebarGroupContent,
8
+ SidebarGroupLabel,
9
+ SidebarMenu,
10
+ SidebarMenuButton,
11
+ SidebarMenuItem,
12
+ SidebarMenuSub,
13
+ SidebarMenuSubButton,
14
+ SidebarMenuSubItem,
15
+ SidebarRail,
16
+ useSidebar,
17
+ } from "@/components/ui/sidebar.client";
18
+ import { Allowed } from "@/modules/auth/entry.client";
19
+ import { Paths } from "@/modules/config/entry";
20
+ import Link from "next/link";
21
+ import { useSelectedLayoutSegments } from "next/navigation";
22
+ import { Suspense } from "react";
23
+ import { Fragment } from "react/jsx-runtime";
24
+ import { AdminSidebarHeader } from "./admin-sidebar-header.client";
25
+ import {
26
+ AdminSidebarHeaderLoading,
27
+ AdminSidebarMenuContentLoading,
28
+ } from "./admin-sidebar-skeleton";
29
+
30
+ /**
31
+ * AdminSidebar
32
+ */
33
+ export function AdminSidebar({
34
+ paths,
35
+ ...props
36
+ }: React.ComponentProps<typeof Sidebar> & { paths: Paths }) {
37
+ const currentSegments = useSelectedLayoutSegments();
38
+
39
+ const { isMobile, toggleSidebar } = useSidebar();
40
+
41
+ return (
42
+ <Sidebar {...props}>
43
+ <Suspense fallback={<AdminSidebarHeaderLoading />}>
44
+ <AdminSidebarHeader />
45
+ </Suspense>
46
+
47
+ <SidebarContent>
48
+ <Suspense fallback={<AdminSidebarMenuContentLoading />}>
49
+ <SidebarMenu>
50
+ {paths.map((group, index) => (
51
+ <SidebarGroup key={index}>
52
+ {group.name && <SidebarGroupLabel>{group.name}</SidebarGroupLabel>}
53
+
54
+ <SidebarGroupContent>
55
+ <SidebarMenu>
56
+ {group.items.map((item) => {
57
+ const isActive = item.isActive || item.segment === currentSegments?.[0];
58
+ const Icon = item.icon;
59
+
60
+ const children = (
61
+ <SidebarMenuItem>
62
+ <SidebarMenuButton asChild isActive={isActive}>
63
+ <Link href={item.href} onClick={isMobile ? toggleSidebar : undefined}>
64
+ {Icon && <Icon />}
65
+ {item.name}
66
+ </Link>
67
+ </SidebarMenuButton>
68
+
69
+ {isActive && !!item.items?.length && (
70
+ <SidebarMenuSub>
71
+ {item.items.map((subItem) => {
72
+ const subChildren = (
73
+ <SidebarMenuSubItem key={subItem.name}>
74
+ <SidebarMenuSubButton
75
+ asChild
76
+ isActive={subItem.segment === currentSegments?.[1]}
77
+ >
78
+ <Link
79
+ href={subItem.href}
80
+ onClick={isMobile ? toggleSidebar : undefined}
81
+ >
82
+ {subItem.name}
83
+ </Link>
84
+ </SidebarMenuSubButton>
85
+ </SidebarMenuSubItem>
86
+ );
87
+
88
+ if (!subItem.permission) {
89
+ return <Fragment key={subItem.name}>{subChildren}</Fragment>;
90
+ }
91
+
92
+ return (
93
+ <Allowed key={subItem.name} permission={subItem.permission}>
94
+ {subChildren}
95
+ </Allowed>
96
+ );
97
+ })}
98
+ </SidebarMenuSub>
99
+ )}
100
+ </SidebarMenuItem>
101
+ );
102
+
103
+ if (!item.permission) {
104
+ return <Fragment key={item.name}>{children}</Fragment>;
105
+ }
106
+
107
+ return (
108
+ <Allowed key={item.name} permission={item.permission}>
109
+ {children}
110
+ </Allowed>
111
+ );
112
+ })}
113
+ </SidebarMenu>
114
+ </SidebarGroupContent>
115
+ </SidebarGroup>
116
+ ))}
117
+ </SidebarMenu>
118
+ </Suspense>
119
+ </SidebarContent>
120
+ <SidebarRail />
121
+ </Sidebar>
122
+ );
123
+ }
@@ -0,0 +1,51 @@
1
+ export function Spinner() {
2
+ return (
3
+ <svg viewBox="0 0 120.09 141.73" className="w-16">
4
+ <title>Tulip Logo Icon</title>
5
+ <defs>
6
+ <linearGradient
7
+ id="tulip-logo-icon-gradient"
8
+ x1="60.5"
9
+ y1="0"
10
+ x2="60.5"
11
+ y2="141.73"
12
+ gradientUnits="userSpaceOnUse"
13
+ >
14
+ <stop offset="0" stopColor="#ffc754" />
15
+ <stop offset="1" stopColor="#ff1800" />
16
+ </linearGradient>
17
+ <linearGradient
18
+ id="tulip-logo-icon-gradient-2"
19
+ x1="72.4"
20
+ y1="5.81"
21
+ x2="72.4"
22
+ y2="141.73"
23
+ xlinkHref="#tulip-logo-icon-gradient"
24
+ />
25
+ <linearGradient
26
+ id="tulip-logo-icon-gradient-3"
27
+ x1="47.69"
28
+ y1="5.81"
29
+ x2="47.69"
30
+ y2="141.73"
31
+ xlinkHref="#tulip-logo-icon-gradient"
32
+ />
33
+ </defs>
34
+
35
+ <g>
36
+ <path
37
+ d="M107.88,93.09c0-35.95-47.38-93.09-47.38-93.09,0,0-47.39,57.14-47.38,93.09,0,26.17,21.21,48.64,47.38,48.64,26.17,0,47.39-22.47,47.38-48.64Z"
38
+ className="animate-pulse fill-[url(#tulip-logo-icon-gradient)] opacity-90"
39
+ />
40
+ <path
41
+ d="M116.68,109.5c12.3-33.78-12.69-103.68-12.69-103.68,0,0-64.07,37.49-76.37,71.27-8.95,24.59,3.3,52.97,27.89,61.92,24.59,8.95,52.21-4.91,61.16-29.5Z"
42
+ className="animate-pulse fill-[url(#tulip-logo-icon-gradient)] opacity-90"
43
+ />
44
+ <path
45
+ d="M92.46,77.09C80.16,43.3,16.09,5.81,16.09,5.81c0,0-24.98,69.9-12.69,103.68,8.95,24.59,36.57,38.45,61.16,29.5,24.59-8.95,36.84-37.32,27.89-61.92Z"
46
+ className="animate-pulse fill-[url(#tulip-logo-icon-gradient)] opacity-90"
47
+ />
48
+ </g>
49
+ </svg>
50
+ );
51
+ }
@@ -1,7 +1,7 @@
1
1
  "use client";
2
2
 
3
3
  import { JSX } from "react";
4
- import { usePermission } from "../hooks/use-permission.client";
4
+ import { usePermission } from "../hooks/use-permission";
5
5
  import { type Permission } from "../lib/permissions";
6
6
 
7
7
  /**
@@ -1,23 +1,15 @@
1
- import "server-cli-only";
2
-
3
- import { TDatabaseSchema } from "@/modules/config/db/types";
4
- import { getTulip, GetTulipParams } from "@/modules/config/lib/resolve.server";
5
1
  import Image from "next/image";
6
2
  import Link from "next/link";
7
3
  import { PropsWithChildren } from "react";
8
4
 
9
- export async function AuthLayout<TSchema extends TDatabaseSchema>(
10
- props: PropsWithChildren<GetTulipParams<TSchema>>,
11
- ) {
12
- const { general } = await getTulip(props);
13
-
5
+ export async function AuthLayout(props: PropsWithChildren<{ name: string }>) {
14
6
  return (
15
7
  <div className="container relative mx-auto grid h-[100dvh] flex-col items-center justify-center lg:max-w-none lg:grid-cols-2 lg:px-0">
16
8
  <div className="bg-muted relative hidden h-full flex-col p-10 text-white lg:flex dark:border-r">
17
9
  <Image src="/images/auth-background.jpeg" alt="Background" className="object-cover" fill />
18
10
 
19
11
  <Link href="/" className="relative z-20 flex items-center text-lg font-bold">
20
- {general.name}
12
+ {props.name}
21
13
  </Link>
22
14
 
23
15
  <div className="relative z-20 mt-auto">
@@ -0,0 +1,11 @@
1
+ import { Skeleton } from "@/components/ui/skeleton";
2
+
3
+ export function AuthLoading() {
4
+ return (
5
+ <div className="mx-auto flex w-full flex-col gap-4 sm:w-[350px]">
6
+ <Skeleton className="h-7 w-full" />
7
+ <Skeleton className="h-48 w-full" />
8
+ <Skeleton className="h-12 w-full" />
9
+ </div>
10
+ );
11
+ }
@@ -0,0 +1,12 @@
1
+ "use client";
2
+
3
+ import { redirect } from "next/navigation";
4
+ import { PropsWithChildren } from "react";
5
+ import { useSession } from "../hooks/use-session";
6
+
7
+ export function AuthGuard({ children }: PropsWithChildren) {
8
+ const session = useSession();
9
+ if (!session.data) redirect("/auth/login");
10
+
11
+ return <>{children}</>;
12
+ }
@@ -5,21 +5,61 @@ import { Button } from "@/components/ui/button";
5
5
  import { Form, FormControl, FormField, FormItem, FormMessage } from "@/components/ui/form.client";
6
6
  import { Input } from "@/components/ui/input";
7
7
  import { standardSchemaResolver } from "@hookform/resolvers/standard-schema";
8
+ import { useMutation, useQueryClient } from "@tanstack/react-query";
8
9
  import Link from "next/link";
9
10
  import { useRouter } from "next/navigation";
10
- import React, { use } from "react";
11
+ import { parseAsString, useQueryState } from "nuqs";
11
12
  import { useForm } from "react-hook-form";
12
13
  import { toast } from "sonner";
13
14
  import * as z from "zod";
14
15
  import { authClient } from "../handler/client.client";
15
16
  import { loginSchema } from "../lib/validators";
16
17
 
17
- export function LoginPage(props: { searchParams: Promise<{ callbackURL?: string }> }) {
18
- const { callbackURL } = use(props.searchParams);
18
+ export function LoginPage() {
19
+ const queryClient = useQueryClient();
20
+ const [callbackURL] = useQueryState("callbackURL", parseAsString);
19
21
 
20
22
  const router = useRouter();
21
23
 
22
- const [isPending, startTransition] = React.useTransition();
24
+ const { mutate, isPending } = useMutation({
25
+ mutationFn: async (data: z.infer<typeof loginSchema>) => {
26
+ const result = await authClient.signIn.email(data);
27
+ if (result.error) throw result.error;
28
+
29
+ const session = await authClient.getSession();
30
+ if (session.error) throw session.error;
31
+
32
+ queryClient.setQueryData(["session"], session.data);
33
+
34
+ return result.data;
35
+ },
36
+ onSuccess: () => {
37
+ router.push(callbackURL ?? "/admin");
38
+ toast.success("Succesvol ingelogd");
39
+ },
40
+ onError: (err) => {
41
+ console.error("Login error: ", err);
42
+
43
+ if ("status" in err) {
44
+ if (err.status === 401) {
45
+ toast.error("Ongeldige email of wachtwoord");
46
+ return;
47
+ }
48
+
49
+ if (err.status === 403) {
50
+ toast.error("Account is geblokkeerd");
51
+ return;
52
+ }
53
+
54
+ if (err.status === 429) {
55
+ toast.error("Te veel login pogingen");
56
+ return;
57
+ }
58
+ }
59
+
60
+ toast.error("Er is iets misgegaan");
61
+ },
62
+ });
23
63
 
24
64
  const form = useForm<z.infer<typeof loginSchema>>({
25
65
  resolver: standardSchemaResolver(loginSchema),
@@ -37,39 +77,7 @@ export function LoginPage(props: { searchParams: Promise<{ callbackURL?: string
37
77
  </div>
38
78
 
39
79
  <Form {...form}>
40
- <form
41
- className="space-y-6"
42
- onSubmit={form.handleSubmit((data) =>
43
- startTransition(async () => {
44
- await authClient.signIn.email(data, {
45
- onSuccess: () => {
46
- router.replace(callbackURL ?? "/");
47
- toast.success("Succesvol ingelogd");
48
- },
49
- onError: ({ error }) => {
50
- console.error("Login error: ", error);
51
-
52
- if (error.status === 401) {
53
- toast.error("Ongeldige email of wachtwoord");
54
- return;
55
- }
56
-
57
- if (error.status === 403) {
58
- toast.error("Account is geblokkeerd");
59
- return;
60
- }
61
-
62
- if (error.status === 429) {
63
- toast.error("Te veel login pogingen");
64
- return;
65
- }
66
-
67
- toast.error("Er is iets misgegaan");
68
- },
69
- });
70
- }),
71
- )}
72
- >
80
+ <form className="space-y-6" onSubmit={form.handleSubmit((data) => mutate(data))}>
73
81
  <div className="flex flex-col gap-3">
74
82
  <FormField
75
83
  control={form.control}
@@ -4,3 +4,5 @@ export * from "./components/login-page.client";
4
4
  export * from "./components/reset-password-page.client";
5
5
  export * from "./components/update-password-command";
6
6
  export * from "./handler/client.client";
7
+ export * from "./hooks/use-permission";
8
+ export * from "./hooks/use-session";
@@ -1,4 +1,4 @@
1
- export * from "./components/allowed.server";
2
1
  export * from "./components/auth-layout.server";
2
+ export * from "./handler/proxy.server";
3
3
  export * from "./handler/route.server";
4
4
  export * from "./lib/helpers.server";
@@ -1,3 +1,4 @@
1
+ export * from "./components/auth-loading";
1
2
  export * from "./db/schema";
2
3
  export * from "./lib/permissions";
3
4
  export * from "./lib/validators";
@@ -0,0 +1,28 @@
1
+ import { getSessionCookie } from "better-auth/cookies";
2
+ import { NextRequest, NextResponse } from "next/server";
3
+ import { URLPattern } from "urlpattern-polyfill";
4
+ import { authCookiePrefix } from "./server.server";
5
+
6
+ /**
7
+ * Create auth proxy
8
+ */
9
+ export function createAuthProxy(props: { publicRoutes: string[] }) {
10
+ return (req: NextRequest) => {
11
+ // 1. Check if the current route is protected or public
12
+ const pathname = req.nextUrl.pathname;
13
+
14
+ const isPublicRoute = props.publicRoutes.some((route) =>
15
+ new URLPattern({ pathname: route }).test({ pathname }),
16
+ );
17
+
18
+ // 2. Check if a session cookie is present
19
+ const sessionCookie = getSessionCookie(req, { cookiePrefix: authCookiePrefix });
20
+
21
+ // 3. Redirect to /auth/login if the user is not authenticated
22
+ if (!isPublicRoute && !sessionCookie) {
23
+ return NextResponse.redirect(new URL("/auth/login", req.nextUrl));
24
+ }
25
+
26
+ return NextResponse.next();
27
+ };
28
+ }