@carlonicora/nextjs-jsonapi 1.0.4 → 1.0.6

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 (283) hide show
  1. package/LICENSE +675 -0
  2. package/dist/{AbstractService-BKlpJA61.d.mts → AbstractService-B2n_JdiC.d.mts} +1 -1
  3. package/dist/{AbstractService-D9eSVKNa.d.ts → AbstractService-DtQTYovo.d.ts} +1 -1
  4. package/dist/{content.interface-Dg2lt_An.d.mts → AuthComponent-CPLvEerw.d.mts} +11 -15
  5. package/dist/{content.interface-BhyAiOFq.d.ts → AuthComponent-m6Qp4Hz6.d.ts} +11 -15
  6. package/dist/{BlockNoteEditor-UVO3VZZE.mjs → BlockNoteEditor-BLVXQPXV.mjs} +14 -18
  7. package/dist/{BlockNoteEditor-UVO3VZZE.mjs.map → BlockNoteEditor-BLVXQPXV.mjs.map} +1 -1
  8. package/dist/{BlockNoteEditor-VFWG6LXI.js → BlockNoteEditor-ZTDHULFT.js} +15 -19
  9. package/dist/BlockNoteEditor-ZTDHULFT.js.map +1 -0
  10. package/dist/JsonApiRequest-O7BGUMFO.mjs +23 -0
  11. package/dist/JsonApiRequest-VARLNKAF.js +23 -0
  12. package/dist/JsonApiRequest-VARLNKAF.js.map +1 -0
  13. package/dist/chunk-2LM6LCJW.mjs +1 -0
  14. package/dist/chunk-3APORDYP.mjs +7783 -0
  15. package/dist/chunk-3APORDYP.mjs.map +1 -0
  16. package/dist/{chunk-TMVHSY3Y.js → chunk-5ZEADNNP.js} +36 -17
  17. package/dist/chunk-5ZEADNNP.js.map +1 -0
  18. package/dist/{chunk-ECDTZBYO.mjs → chunk-74F6BBHH.mjs} +21 -2
  19. package/dist/chunk-74F6BBHH.mjs.map +1 -0
  20. package/dist/{chunk-GYWPEPOH.mjs → chunk-7C5RAEBO.mjs} +72 -68
  21. package/dist/chunk-7C5RAEBO.mjs.map +1 -0
  22. package/dist/chunk-A5DDIABK.js +1 -0
  23. package/dist/{chunk-TEGF6ZWG.js → chunk-AGAJMJ4T.js} +47 -9
  24. package/dist/chunk-AGAJMJ4T.js.map +1 -0
  25. package/dist/{chunk-CXQOWQSY.js → chunk-AYHKQWHH.js} +15 -2
  26. package/dist/chunk-AYHKQWHH.js.map +1 -0
  27. package/dist/{chunk-I2REI7OA.js → chunk-HMHGLXWC.js} +33 -15
  28. package/dist/chunk-HMHGLXWC.js.map +1 -0
  29. package/dist/chunk-IWFGEPAA.mjs +1 -0
  30. package/dist/chunk-JC3WJK65.js +1 -0
  31. package/dist/{chunk-L6EQEAXU.mjs → chunk-PYF2U6WG.mjs} +25 -7
  32. package/dist/chunk-PYF2U6WG.mjs.map +1 -0
  33. package/dist/{chunk-YDVTFM7X.mjs → chunk-RBIVEH2K.mjs} +42 -4
  34. package/dist/chunk-RBIVEH2K.mjs.map +1 -0
  35. package/dist/{chunk-V2JJPI7N.js → chunk-RZO2LOW4.js} +237 -233
  36. package/dist/chunk-RZO2LOW4.js.map +1 -0
  37. package/dist/{chunk-X4BIHJ2B.mjs → chunk-SM63SZCP.mjs} +15 -2
  38. package/dist/chunk-SM63SZCP.mjs.map +1 -0
  39. package/dist/chunk-WEC4YMOS.js +7783 -0
  40. package/dist/chunk-WEC4YMOS.js.map +1 -0
  41. package/dist/client/index.d.mts +21 -2
  42. package/dist/client/index.d.ts +21 -2
  43. package/dist/client/index.js +18 -245
  44. package/dist/client/index.js.map +1 -1
  45. package/dist/client/index.mjs +28 -255
  46. package/dist/client/index.mjs.map +1 -1
  47. package/dist/components/index.d.mts +293 -8
  48. package/dist/components/index.d.ts +293 -8
  49. package/dist/components/index.js +78 -2323
  50. package/dist/components/index.js.map +1 -1
  51. package/dist/components/index.mjs +172 -2417
  52. package/dist/components/index.mjs.map +1 -1
  53. package/dist/{config-hXufftVS.d.ts → config-BmnK65TD.d.mts} +1 -0
  54. package/dist/{config-hXufftVS.d.mts → config-BmnK65TD.d.ts} +1 -0
  55. package/dist/config-DQeAo9Kf.d.mts +49 -0
  56. package/dist/config-DQeAo9Kf.d.ts +49 -0
  57. package/dist/contexts/index.d.mts +109 -21
  58. package/dist/contexts/index.d.ts +109 -21
  59. package/dist/contexts/index.js +39 -7
  60. package/dist/contexts/index.js.map +1 -1
  61. package/dist/contexts/index.mjs +40 -8
  62. package/dist/core/index.d.mts +3 -4
  63. package/dist/core/index.d.ts +3 -4
  64. package/dist/core/index.js +3 -7
  65. package/dist/core/index.js.map +1 -1
  66. package/dist/core/index.mjs +4 -8
  67. package/dist/{d3.link.interface-QMdB22bC.d.mts → d3.link.interface-ClC4Irqp.d.mts} +2 -1
  68. package/dist/{d3.link.interface-QMdB22bC.d.ts → d3.link.interface-ClC4Irqp.d.ts} +2 -1
  69. package/dist/features/index.d.mts +17 -86
  70. package/dist/features/index.d.ts +17 -86
  71. package/dist/features/index.js +7 -16
  72. package/dist/features/index.js.map +1 -1
  73. package/dist/features/index.mjs +10 -19
  74. package/dist/hooks/index.d.mts +18 -43
  75. package/dist/hooks/index.d.ts +18 -43
  76. package/dist/hooks/index.js +20 -7
  77. package/dist/hooks/index.js.map +1 -1
  78. package/dist/hooks/index.mjs +19 -6
  79. package/dist/index.d.mts +10 -6
  80. package/dist/index.d.ts +10 -6
  81. package/dist/index.js +13 -10
  82. package/dist/index.js.map +1 -1
  83. package/dist/index.mjs +22 -19
  84. package/dist/interfaces/index.d.mts +2 -1
  85. package/dist/interfaces/index.d.ts +2 -1
  86. package/dist/notification.interface-BBgMUdLR.d.mts +14 -0
  87. package/dist/notification.interface-gyvT-Z2F.d.ts +14 -0
  88. package/dist/permissions/index.d.mts +2 -3
  89. package/dist/permissions/index.d.ts +2 -3
  90. package/dist/server/index.d.mts +38 -18
  91. package/dist/server/index.d.ts +38 -18
  92. package/dist/server/index.js +70 -2
  93. package/dist/server/index.js.map +1 -1
  94. package/dist/server/index.mjs +68 -0
  95. package/dist/server/index.mjs.map +1 -1
  96. package/dist/types-BUAlgqqh.d.ts +39 -0
  97. package/dist/{types-DluCaP1I.d.ts → types-Bl61ob-7.d.mts} +19 -2
  98. package/dist/{types-lQVA8d_P.d.mts → types-Bl61ob-7.d.ts} +19 -2
  99. package/dist/types-iVdVY7ba.d.mts +39 -0
  100. package/dist/useSocket-Cn7fB_B1.d.mts +25 -0
  101. package/dist/useSocket-DzMKRKCA.d.ts +25 -0
  102. package/dist/user.fields-CbdObSmS.d.mts +18 -0
  103. package/dist/user.fields-CbdObSmS.d.ts +18 -0
  104. package/dist/utils/index.d.mts +1 -2
  105. package/dist/utils/index.d.ts +1 -2
  106. package/package.json +5 -3
  107. package/src/client/index.ts +13 -0
  108. package/src/components/forms/index.ts +1 -0
  109. package/src/components/index.ts +5 -0
  110. package/src/components/tables/ContentListTable.tsx +1 -0
  111. package/src/contexts/CommonContext.tsx +52 -0
  112. package/src/contexts/SharedContext.tsx +2 -0
  113. package/src/contexts/SocketContext.tsx +65 -0
  114. package/src/contexts/index.ts +6 -1
  115. package/src/features/auth/components/containers/AuthContainer.tsx +32 -0
  116. package/src/features/auth/components/containers/index.ts +1 -0
  117. package/src/features/auth/components/details/LandingComponent.tsx +39 -0
  118. package/src/features/auth/components/details/index.ts +1 -0
  119. package/src/features/auth/components/forms/AcceptInvitation.tsx +136 -0
  120. package/src/features/auth/components/forms/ActivateAccount.tsx +75 -0
  121. package/src/features/auth/components/forms/Cookies.tsx +32 -0
  122. package/src/features/auth/components/forms/ForgotPassword.tsx +108 -0
  123. package/src/features/auth/components/forms/Login.tsx +118 -0
  124. package/src/features/auth/components/forms/Logout.tsx +19 -0
  125. package/src/features/auth/components/forms/RefreshUser.tsx +39 -0
  126. package/src/features/auth/components/forms/Register.tsx +150 -0
  127. package/src/features/auth/components/forms/ResetPassword.tsx +126 -0
  128. package/src/features/auth/components/forms/index.ts +9 -0
  129. package/src/features/auth/components/index.ts +3 -0
  130. package/src/features/auth/contexts/AuthContext.tsx +77 -0
  131. package/src/features/auth/contexts/index.ts +1 -0
  132. package/src/features/auth/enums/AuthComponent.ts +9 -0
  133. package/src/features/auth/enums/index.ts +1 -0
  134. package/src/features/auth/index.ts +2 -1
  135. package/src/features/auth/utils/AuthCookies.ts +134 -0
  136. package/src/features/auth/utils/index.ts +1 -0
  137. package/src/features/company/components/containers/AdminCompanyContainer.tsx +26 -0
  138. package/src/features/company/components/containers/CompanyContainer.tsx +17 -0
  139. package/src/features/company/components/containers/index.ts +2 -0
  140. package/src/features/company/components/details/CompanyDetails.tsx +26 -0
  141. package/src/features/company/components/details/index.ts +1 -0
  142. package/src/features/company/components/forms/CompanyConfigurationEditor.tsx +151 -0
  143. package/src/features/company/components/forms/CompanyConfigurationSecurityForm.tsx +97 -0
  144. package/src/features/company/components/forms/CompanyDeleter.tsx +121 -0
  145. package/src/features/company/components/forms/CompanyEditor.tsx +245 -0
  146. package/src/features/company/components/forms/CompanyLicense.tsx +213 -0
  147. package/src/features/company/components/forms/index.ts +5 -0
  148. package/src/features/company/components/index.ts +4 -0
  149. package/src/features/company/components/lists/CompaniesList.tsx +31 -0
  150. package/src/features/company/components/lists/index.ts +1 -0
  151. package/src/features/company/contexts/CompanyContext.tsx +99 -0
  152. package/src/features/company/contexts/index.ts +0 -0
  153. package/src/features/company/hooks/index.ts +1 -0
  154. package/src/features/company/hooks/useCompanyTableStructure.tsx +82 -0
  155. package/src/features/feature/components/forms/FormFeatures.tsx +141 -140
  156. package/src/features/feature/components/forms/index.ts +1 -0
  157. package/src/features/feature/components/index.ts +1 -1
  158. package/src/features/feature/index.ts +1 -2
  159. package/src/features/module/index.ts +1 -1
  160. package/src/features/notification/components/common/NotificationErrorBoundary.tsx +51 -0
  161. package/src/features/notification/components/common/index.ts +1 -0
  162. package/src/features/notification/components/containers/NotificationsListContainer.tsx +44 -0
  163. package/src/features/notification/components/containers/index.ts +1 -0
  164. package/src/features/notification/components/index.ts +5 -0
  165. package/src/features/notification/components/lists/NotificationsList.tsx +129 -0
  166. package/src/features/notification/components/lists/index.ts +1 -0
  167. package/src/features/notification/components/modals/NotificationModal.tsx +220 -0
  168. package/src/features/notification/components/modals/index.ts +1 -0
  169. package/src/features/notification/components/notifications/Notification.tsx +120 -0
  170. package/src/features/notification/components/notifications/PushNotificationProvider.tsx +9 -0
  171. package/src/features/notification/components/notifications/index.ts +2 -0
  172. package/src/features/notification/contexts/NotificationContext.tsx +187 -0
  173. package/src/features/notification/contexts/index.ts +1 -0
  174. package/src/features/notification/index.ts +1 -1
  175. package/src/features/push/index.ts +1 -1
  176. package/src/features/role/components/containers/RoleContainer.tsx +18 -0
  177. package/src/features/role/components/containers/index.ts +1 -0
  178. package/src/features/role/components/details/RoleDetails.tsx +21 -0
  179. package/src/features/role/components/details/index.ts +1 -0
  180. package/src/features/role/components/forms/FormRoles.tsx +82 -0
  181. package/src/features/role/components/forms/RemoveUserFromRole.tsx +108 -0
  182. package/src/features/role/components/forms/UserRoleAdd.tsx +128 -0
  183. package/src/features/role/components/forms/index.ts +3 -0
  184. package/src/features/role/components/index.ts +4 -0
  185. package/src/features/role/components/lists/RolesList.tsx +27 -0
  186. package/src/features/role/components/lists/UserRolesList.tsx +31 -0
  187. package/src/features/role/components/lists/index.ts +2 -0
  188. package/src/features/role/contexts/RoleContext.tsx +84 -0
  189. package/src/features/role/contexts/index.ts +1 -0
  190. package/src/features/role/hooks/index.ts +1 -0
  191. package/src/features/role/hooks/useRoleTableStructure.tsx +72 -0
  192. package/src/features/s3/index.ts +1 -1
  193. package/src/features/user/components/containers/UserContainer.tsx +23 -0
  194. package/src/features/user/components/containers/UserIndexContainer.tsx +12 -0
  195. package/src/features/user/components/containers/UsersListContainer.tsx +36 -0
  196. package/src/features/user/components/containers/index.ts +3 -0
  197. package/src/features/user/components/details/UserDetails.tsx +74 -0
  198. package/src/features/user/components/details/UserIndexDetails.tsx +28 -0
  199. package/src/features/user/components/details/index.ts +2 -0
  200. package/src/features/user/components/forms/RoleUserAdd.tsx +93 -0
  201. package/src/features/user/components/forms/UserAvatarEditor.tsx +78 -0
  202. package/src/features/user/components/forms/UserDeleter.tsx +49 -0
  203. package/src/features/user/components/forms/UserEditor.tsx +319 -0
  204. package/src/features/user/components/forms/UserMultiSelect.tsx +218 -0
  205. package/src/features/user/components/forms/UserReactivator.tsx +79 -0
  206. package/src/features/user/components/forms/UserResentInvitationEmail.tsx +88 -0
  207. package/src/features/user/components/forms/UserSelector.tsx +185 -0
  208. package/src/features/user/components/forms/index.ts +8 -0
  209. package/src/features/user/components/index.ts +3 -0
  210. package/src/features/user/components/lists/AdminUsersList.tsx +41 -0
  211. package/src/features/user/components/lists/CompanyUsersList.tsx +44 -0
  212. package/src/features/user/components/lists/RelevantUsersList.tsx +30 -0
  213. package/src/features/user/components/lists/RoleUsersList.tsx +31 -0
  214. package/src/features/user/components/lists/UserListInAdd.tsx +53 -0
  215. package/src/features/user/components/lists/UsersList.tsx +30 -0
  216. package/src/features/user/components/lists/UsersListByContentIds.tsx +30 -0
  217. package/src/features/user/components/lists/index.ts +7 -0
  218. package/src/features/user/components/widgets/UserAvatarList.tsx +31 -0
  219. package/src/features/user/components/widgets/UserSearchPopover.tsx +89 -0
  220. package/src/features/user/contexts/UserContext.tsx +106 -0
  221. package/src/features/user/contexts/index.ts +1 -0
  222. package/src/features/user/hooks/index.ts +2 -0
  223. package/src/features/user/hooks/useUserSearch.ts +53 -0
  224. package/src/features/user/hooks/useUserTableStructure.tsx +115 -0
  225. package/src/features/user/index.ts +0 -1
  226. package/src/hooks/index.ts +4 -0
  227. package/src/hooks/useCustomD3Graph.tsx +2 -0
  228. package/src/hooks/useNotificationSync.ts +20 -0
  229. package/src/hooks/usePageTracker.ts +69 -0
  230. package/src/hooks/usePushNotifications.ts +82 -0
  231. package/src/hooks/useSocket.ts +201 -0
  232. package/src/hooks/useTableGenerator.ts +6 -2
  233. package/src/i18n/config.ts +1 -0
  234. package/src/index.ts +4 -0
  235. package/src/interfaces/d3.link.interface.ts +2 -1
  236. package/src/server/ServerSession.ts +103 -0
  237. package/src/server/index.ts +2 -1
  238. package/src/unified/JsonApiRequest.ts +23 -0
  239. package/dist/ApiRequestDataTypeInterface-CUKFDBx2.d.mts +0 -20
  240. package/dist/ApiRequestDataTypeInterface-CUKFDBx2.d.ts +0 -20
  241. package/dist/BlockNoteEditor-VFWG6LXI.js.map +0 -1
  242. package/dist/JsonApiRequest-S3ICLM7B.mjs +0 -20
  243. package/dist/JsonApiRequest-ZZLSP26T.js +0 -20
  244. package/dist/JsonApiRequest-ZZLSP26T.js.map +0 -1
  245. package/dist/chunk-366S2JCC.mjs +0 -31
  246. package/dist/chunk-366S2JCC.mjs.map +0 -1
  247. package/dist/chunk-5W6AKZE6.mjs +0 -131
  248. package/dist/chunk-5W6AKZE6.mjs.map +0 -1
  249. package/dist/chunk-A3J3AAYM.mjs +0 -97
  250. package/dist/chunk-A3J3AAYM.mjs.map +0 -1
  251. package/dist/chunk-AWONBQQP.js +0 -97
  252. package/dist/chunk-AWONBQQP.js.map +0 -1
  253. package/dist/chunk-CXQOWQSY.js.map +0 -1
  254. package/dist/chunk-DKKMWBP4.mjs +0 -1
  255. package/dist/chunk-DKKMWBP4.mjs.map +0 -1
  256. package/dist/chunk-DO2HLAZO.js +0 -48
  257. package/dist/chunk-DO2HLAZO.js.map +0 -1
  258. package/dist/chunk-DZXDB3K2.mjs +0 -17
  259. package/dist/chunk-DZXDB3K2.mjs.map +0 -1
  260. package/dist/chunk-ECDTZBYO.mjs.map +0 -1
  261. package/dist/chunk-FY4SXJGU.js +0 -806
  262. package/dist/chunk-FY4SXJGU.js.map +0 -1
  263. package/dist/chunk-GYWPEPOH.mjs.map +0 -1
  264. package/dist/chunk-H6FMOA6B.js +0 -1
  265. package/dist/chunk-H6FMOA6B.js.map +0 -1
  266. package/dist/chunk-I2REI7OA.js.map +0 -1
  267. package/dist/chunk-J4Q36PMP.js +0 -31
  268. package/dist/chunk-J4Q36PMP.js.map +0 -1
  269. package/dist/chunk-L6EQEAXU.mjs.map +0 -1
  270. package/dist/chunk-MFO27OHB.mjs +0 -48
  271. package/dist/chunk-MFO27OHB.mjs.map +0 -1
  272. package/dist/chunk-RAF7PNLG.js +0 -131
  273. package/dist/chunk-RAF7PNLG.js.map +0 -1
  274. package/dist/chunk-RUR22SVM.js +0 -17
  275. package/dist/chunk-RUR22SVM.js.map +0 -1
  276. package/dist/chunk-TEGF6ZWG.js.map +0 -1
  277. package/dist/chunk-TMVHSY3Y.js.map +0 -1
  278. package/dist/chunk-V2JJPI7N.js.map +0 -1
  279. package/dist/chunk-WWWMJZEF.mjs +0 -806
  280. package/dist/chunk-WWWMJZEF.mjs.map +0 -1
  281. package/dist/chunk-X4BIHJ2B.mjs.map +0 -1
  282. package/dist/chunk-YDVTFM7X.mjs.map +0 -1
  283. /package/dist/{JsonApiRequest-S3ICLM7B.mjs.map → JsonApiRequest-O7BGUMFO.mjs.map} +0 -0
@@ -0,0 +1,18 @@
1
+ "use client";
2
+
3
+ import { RoleUsersList } from "../../../user/components/lists/RoleUsersList";
4
+ import { useRoleContext } from "../../contexts";
5
+ import { RoleDetails } from "../details";
6
+
7
+ export function RoleContainer() {
8
+ const { role } = useRoleContext();
9
+
10
+ if (!role) return null;
11
+
12
+ return (
13
+ <>
14
+ <RoleDetails />
15
+ <RoleUsersList role={role} />
16
+ </>
17
+ );
18
+ }
@@ -0,0 +1 @@
1
+ export * from "./RoleContainer";
@@ -0,0 +1,21 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { AttributeElement } from "../../../../components";
5
+ import { Card, CardContent } from "../../../../shadcnui";
6
+ import { useRoleContext } from "../../contexts";
7
+
8
+ export function RoleDetails() {
9
+ const { role } = useRoleContext();
10
+ const t = useTranslations();
11
+
12
+ if (!role) return null;
13
+
14
+ return (
15
+ <Card className="w-full">
16
+ <CardContent className="p-4">
17
+ <AttributeElement title={t(`foundations.role.fields.description.label`)} value={role.description} />
18
+ </CardContent>
19
+ </Card>
20
+ );
21
+ }
@@ -0,0 +1 @@
1
+ export * from "./RoleDetails";
@@ -0,0 +1,82 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import {
5
+ Checkbox,
6
+ FormControl,
7
+ FormField,
8
+ FormItem,
9
+ FormLabel,
10
+ FormMessage,
11
+ Tooltip,
12
+ TooltipContent,
13
+ TooltipTrigger,
14
+ } from "../../../../shadcnui";
15
+ import { UserInterface } from "../../../user";
16
+ import { useCurrentUserContext } from "../../../user/contexts";
17
+ import { RoleInterface } from "../../data";
18
+
19
+ type FormRolesProps = {
20
+ form: any;
21
+ id: string;
22
+ name: string;
23
+ roles: RoleInterface[];
24
+ };
25
+
26
+ export function FormRoles({ form, id, name, roles }: FormRolesProps) {
27
+ const t = useTranslations();
28
+ const { hasAccesToFeature } = useCurrentUserContext<UserInterface>();
29
+
30
+ return (
31
+ <div className="flex w-full flex-col">
32
+ <FormField
33
+ control={form.control}
34
+ name={id}
35
+ render={({ field }) => (
36
+ <FormItem className={`${name ? "mb-5" : "mb-1"}`}>
37
+ <FormControl>
38
+ <div>
39
+ <div className="text-sm font-semibold">{name}</div>
40
+ {roles
41
+ .filter((role: RoleInterface) => role.isSelectable)
42
+ .sort((a: RoleInterface, b: RoleInterface) => a.name.localeCompare(b.name))
43
+ .map((role: RoleInterface) => {
44
+ if (role.requiredFeature && !hasAccesToFeature(role.requiredFeature.id)) return null;
45
+
46
+ return (
47
+ <div key={role.id}>
48
+ <Checkbox
49
+ defaultChecked={(field.value as string[]).some((roleId: string) => roleId === role.id)}
50
+ onCheckedChange={(checked) => {
51
+ if (checked) {
52
+ form.setValue(id, [...(field.value as string[]), role.id]);
53
+ } else {
54
+ form.setValue(
55
+ id,
56
+ (field.value as string[]).filter((roleId: string) => roleId !== role.id),
57
+ );
58
+ }
59
+ }}
60
+ />
61
+ <Tooltip>
62
+ <TooltipTrigger asChild>
63
+ <FormLabel className="ml-3 font-normal">
64
+ {t(`foundations.role.roles`, { role: role.id.replaceAll(`-`, ``) })}
65
+ </FormLabel>
66
+ </TooltipTrigger>
67
+ <TooltipContent>
68
+ {t(`foundations.role.roles_descriptions`, { role: role.id.replaceAll(`-`, ``) })}
69
+ </TooltipContent>
70
+ </Tooltip>
71
+ </div>
72
+ );
73
+ })}
74
+ </div>
75
+ </FormControl>
76
+ <FormMessage />
77
+ </FormItem>
78
+ )}
79
+ />
80
+ </div>
81
+ );
82
+ }
@@ -0,0 +1,108 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { useEffect, useState } from "react";
5
+ import { errorToast } from "../../../../components";
6
+ import { getRoleId } from "../../../../roles";
7
+ import {
8
+ Button,
9
+ Dialog,
10
+ DialogContent,
11
+ DialogDescription,
12
+ DialogHeader,
13
+ DialogTitle,
14
+ DialogTrigger,
15
+ } from "../../../../shadcnui";
16
+ import { UserInterface, UserService } from "../../../user";
17
+ import { RoleInterface, RoleService } from "../../data";
18
+
19
+ type RemoveUserFromRoleProps = {
20
+ user: UserInterface;
21
+ role: RoleInterface;
22
+ refresh: () => Promise<void>;
23
+ };
24
+
25
+ export function RemoveUserFromRole({ role, user, refresh }: RemoveUserFromRoleProps) {
26
+ const [open, setOpen] = useState<boolean>(false);
27
+ const [canRemove, setCanRemove] = useState<boolean>(false);
28
+ const t = useTranslations();
29
+
30
+ useEffect(() => {
31
+ async function checkCompanyAdminDeletability(): Promise<void> {
32
+ const roleUsers = await UserService.findAllUsersByRole({
33
+ roleId: role.id,
34
+ });
35
+
36
+ if (roleUsers.length > 1) setCanRemove(true);
37
+ }
38
+
39
+ if (role.id !== getRoleId().CompanyAdministrator) {
40
+ setCanRemove(true);
41
+ return;
42
+ }
43
+
44
+ checkCompanyAdminDeletability();
45
+ }, [role]);
46
+
47
+ const remove = async () => {
48
+ try {
49
+ await RoleService.removeUserFromRole({
50
+ roleId: role.id,
51
+ userId: user.id,
52
+ });
53
+
54
+ setOpen(false);
55
+ refresh();
56
+ } catch (error) {
57
+ errorToast({ title: t(`generic.errors.error`), error: error });
58
+ }
59
+ };
60
+
61
+ const roleName = t(`foundations.role.roles`, { role: role.id.replaceAll(`-`, ``) });
62
+
63
+ return (
64
+ <Dialog open={open} onOpenChange={setOpen}>
65
+ <DialogTrigger
66
+ onClick={(e) => {
67
+ e.stopPropagation();
68
+ e.preventDefault();
69
+ setOpen(true);
70
+ }}
71
+ >
72
+ <span className="hover:text-destructive cursor-pointer">{t(`foundations.role.remove_user.title`)}</span>
73
+ </DialogTrigger>
74
+ <DialogContent className={`flex max-h-[70vh] max-w-3xl flex-col overflow-y-auto`}>
75
+ <DialogHeader>
76
+ <DialogTitle>{t(`foundations.role.remove_user.title`)}</DialogTitle>
77
+ <DialogDescription>
78
+ {canRemove
79
+ ? t(`foundations.role.remove_user.subtitle_allowed`)
80
+ : t(`foundations.role.remove_user.subtitle_not_allowed`)}
81
+ </DialogDescription>
82
+ </DialogHeader>
83
+ {canRemove ? (
84
+ <>
85
+ {t(`foundations.role.remove_user.description_allowed`, { role: roleName, user: user.name })}
86
+ <div className="flex justify-end">
87
+ <Button className="mr-2" variant={"outline"} type={`button`} onClick={() => setOpen(false)}>
88
+ {t(`generic.buttons.cancel`)}
89
+ </Button>
90
+ <Button
91
+ type="submit"
92
+ onClick={(e) => {
93
+ e.preventDefault();
94
+ remove();
95
+ }}
96
+ variant={"destructive"}
97
+ >
98
+ {t(`generic.buttons.confirm_delete`)}
99
+ </Button>
100
+ </div>
101
+ </>
102
+ ) : (
103
+ <>{t(`foundations.role.remove_user.description_not_allowed`, { role: roleName, user: user.name })}</>
104
+ )}
105
+ </DialogContent>
106
+ </Dialog>
107
+ );
108
+ }
@@ -0,0 +1,128 @@
1
+ "use client";
2
+
3
+ import { PlusCircle } from "lucide-react";
4
+ import { useTranslations } from "next-intl";
5
+ import { useCallback, useEffect, useRef, useState } from "react";
6
+ import { toast } from "sonner";
7
+ import { useDebounce } from "../../../../hooks";
8
+ import {
9
+ Button,
10
+ Command,
11
+ CommandDialog,
12
+ CommandEmpty,
13
+ CommandInput,
14
+ CommandItem,
15
+ CommandList,
16
+ DialogDescription,
17
+ DialogTitle,
18
+ } from "../../../../shadcnui";
19
+ import { UserInterface } from "../../../user";
20
+ import { RoleInterface, RoleService } from "../../data";
21
+
22
+ type UserRoleAddProps = {
23
+ user: UserInterface;
24
+ refresh: () => Promise<void>;
25
+ };
26
+
27
+ export function UserRoleAdd({ user, refresh }: UserRoleAddProps) {
28
+ const [open, setOpen] = useState<boolean>(false);
29
+ const inputRef = useRef<HTMLInputElement>(null);
30
+ const [searchTerm, setSearchTerm] = useState<string>("");
31
+ const [roles, setRoles] = useState<RoleInterface[]>([]);
32
+ const t = useTranslations();
33
+
34
+ const addUserToRole = async (role: RoleInterface) => {
35
+ await RoleService.addUserToRole({
36
+ roleId: role.id,
37
+ userId: user.id,
38
+ });
39
+ setRoles(roles.filter((u) => u.id !== role.id));
40
+
41
+ toast.message(
42
+ t(`generic.association.label`, {
43
+ source: t(`types.roles`, { count: 1 }),
44
+ destination: t(`types.users`, { count: 1 }),
45
+ }),
46
+ {
47
+ description: t(`generic.association.success`, {
48
+ source: t(`types.roles`, { count: 1 }),
49
+ destination: t(`types.users`, { count: 1 }),
50
+ source_name: role.name,
51
+ destination_name: user.name,
52
+ }),
53
+ },
54
+ );
55
+
56
+ refresh();
57
+ };
58
+
59
+ const searchRoles = useCallback(
60
+ async (term: string) => {
61
+ setRoles(
62
+ await RoleService.findAllRolesUserNotIn({
63
+ search: term,
64
+ userId: user.id,
65
+ }),
66
+ );
67
+ },
68
+ [searchTerm, user],
69
+ );
70
+
71
+ const updateSearchTerm = useDebounce(searchRoles, 500);
72
+
73
+ useEffect(() => {
74
+ if (open) updateSearchTerm(searchTerm);
75
+ }, [open, searchTerm]);
76
+
77
+ useEffect(() => {
78
+ if (open) searchRoles("");
79
+ }, [open]);
80
+
81
+ return (
82
+ <>
83
+ <Button size="sm" onClick={() => setOpen(true)}>
84
+ <PlusCircle className="mr-3 h-3.5 w-3.5" />
85
+ {t(`generic.association.label`, {
86
+ source: t(`types.roles`, { count: 1 }),
87
+ destination: t(`types.users`, { count: 1 }),
88
+ })}
89
+ </Button>
90
+ <CommandDialog open={open} onOpenChange={setOpen}>
91
+ <DialogTitle>
92
+ {t(`generic.association.label`, {
93
+ source: t(`types.roles`, { count: 1 }),
94
+ destination: t(`types.users`, { count: 1 }),
95
+ })}
96
+ </DialogTitle>
97
+ <DialogDescription>
98
+ {t(`generic.association.description`, {
99
+ source: t(`types.roles`, { count: 1 }),
100
+ destination: t(`types.users`, { count: 1 }),
101
+ destination_name: user.name,
102
+ })}
103
+ </DialogDescription>
104
+ <Command shouldFilter={false}>
105
+ <CommandInput
106
+ placeholder={t(`generic.search.placeholder`, { type: t(`types.roles`, { count: 1 }) })}
107
+ value={searchTerm}
108
+ onValueChange={setSearchTerm}
109
+ ref={inputRef}
110
+ />
111
+ <CommandList className="mt-3 h-auto max-h-96 min-h-96 max-w-full overflow-x-hidden overflow-y-auto">
112
+ <CommandEmpty>{t(`generic.search.no_results`, { type: t(`types.roles`, { count: 1 }) })}</CommandEmpty>
113
+ {roles.map((role: RoleInterface) => (
114
+ <CommandItem
115
+ className="cursor-pointer"
116
+ key={role.id}
117
+ onSelect={() => addUserToRole(role)}
118
+ onClick={() => addUserToRole(role)}
119
+ >
120
+ {t(`foundations.role.roles`, { role: role.id.replaceAll(`-`, ``) })}
121
+ </CommandItem>
122
+ ))}
123
+ </CommandList>
124
+ </Command>
125
+ </CommandDialog>
126
+ </>
127
+ );
128
+ }
@@ -0,0 +1,3 @@
1
+ export * from "./FormRoles";
2
+ export * from "./RemoveUserFromRole";
3
+ export * from "./UserRoleAdd";
@@ -0,0 +1,4 @@
1
+ export * from "./containers";
2
+ export * from "./details";
3
+ export * from "./forms";
4
+ export * from "./lists";
@@ -0,0 +1,27 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { ContentListTable } from "../../../../components";
5
+ import { Modules } from "../../../../core";
6
+ import { DataListRetriever, useDataListRetriever } from "../../../../hooks";
7
+ import { RoleFields, RoleInterface, RoleService } from "../../data";
8
+ import "../../hooks/useRoleTableStructure";
9
+
10
+ export function RolesList() {
11
+ const t = useTranslations();
12
+
13
+ const data: DataListRetriever<RoleInterface> = useDataListRetriever({
14
+ retriever: (params) => RoleService.findAllRoles(params),
15
+ retrieverParams: {},
16
+ module: Modules.Role,
17
+ });
18
+
19
+ return (
20
+ <ContentListTable
21
+ data={data}
22
+ fields={[RoleFields.name, RoleFields.description]}
23
+ tableGeneratorType={Modules.Role}
24
+ title={t(`types.roles`, { count: 2 })}
25
+ />
26
+ );
27
+ }
@@ -0,0 +1,31 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { ContentListTable } from "../../../../components";
5
+ import { Modules } from "../../../../core";
6
+ import { DataListRetriever, useDataListRetriever } from "../../../../hooks";
7
+ import { UserInterface } from "../../../user";
8
+ import { RoleFields, RoleInterface, RoleService } from "../../data";
9
+
10
+ type UserRolesListProps = {
11
+ user: UserInterface;
12
+ };
13
+
14
+ export function UserRolesList({ user }: UserRolesListProps) {
15
+ const t = useTranslations();
16
+
17
+ const data: DataListRetriever<RoleInterface> = useDataListRetriever({
18
+ retriever: (params) => RoleService.findAllRolesByUser(params),
19
+ retrieverParams: { userId: user.id },
20
+ module: Modules.Role,
21
+ });
22
+
23
+ return (
24
+ <ContentListTable
25
+ data={data}
26
+ fields={[RoleFields.name, RoleFields.description]}
27
+ tableGeneratorType={Modules.Role}
28
+ title={t(`types.roles`, { count: 2 })}
29
+ />
30
+ );
31
+ }
@@ -0,0 +1,2 @@
1
+ export * from "./RolesList";
2
+ export * from "./UserRolesList";
@@ -0,0 +1,84 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { createContext, ReactNode, useContext, useState } from "react";
5
+ import { SharedProvider } from "../../../contexts";
6
+ import { JsonApiHydratedDataInterface, Modules, rehydrate } from "../../../core";
7
+ import { usePageUrlGenerator } from "../../../hooks";
8
+ import { BreadcrumbItemData } from "../../../interfaces";
9
+ import { RoleInterface } from "../data";
10
+
11
+ interface RoleContextType {
12
+ role: RoleInterface | undefined;
13
+ setRole: (value: RoleInterface | undefined) => void;
14
+ }
15
+
16
+ const RoleContext = createContext<RoleContextType | undefined>(undefined);
17
+
18
+ type RoleProviderProps = {
19
+ children: ReactNode;
20
+ dehydratedRole?: JsonApiHydratedDataInterface;
21
+ };
22
+
23
+ export const RoleProvider = ({ children, dehydratedRole }: RoleProviderProps) => {
24
+ const generateUrl = usePageUrlGenerator();
25
+ const t = useTranslations();
26
+
27
+ const [role, setRole] = useState<RoleInterface | undefined>(
28
+ dehydratedRole ? rehydrate<RoleInterface>(Modules.Role, dehydratedRole) : undefined,
29
+ );
30
+
31
+ const breadcrumb = () => {
32
+ const response: BreadcrumbItemData[] = [];
33
+
34
+ response.push({
35
+ name: t(`types.roles`, { count: 2 }),
36
+ href: generateUrl({ page: Modules.Role }),
37
+ });
38
+
39
+ if (role)
40
+ response.push({
41
+ name: role.name,
42
+ href: generateUrl({ page: Modules.Role, id: role.id }),
43
+ });
44
+
45
+ return response;
46
+ };
47
+
48
+ const title = () => {
49
+ const response: any = {
50
+ type: t(`types.roles`, { count: role ? 1 : 2 }),
51
+ };
52
+
53
+ const functions: ReactNode[] = [];
54
+
55
+ if (role) {
56
+ response.element = role.name;
57
+ }
58
+
59
+ if (functions.length > 0) response.functions = functions;
60
+
61
+ return response;
62
+ };
63
+
64
+ return (
65
+ <SharedProvider value={{ breadcrumbs: breadcrumb(), title: title() }}>
66
+ <RoleContext.Provider
67
+ value={{
68
+ role: role,
69
+ setRole: setRole,
70
+ }}
71
+ >
72
+ {children}
73
+ </RoleContext.Provider>
74
+ </SharedProvider>
75
+ );
76
+ };
77
+
78
+ export const useRoleContext = (): RoleContextType => {
79
+ const context = useContext(RoleContext);
80
+ if (context === undefined) {
81
+ throw new Error("useRoleContext must be used within a RoleProvider");
82
+ }
83
+ return context;
84
+ };
@@ -0,0 +1 @@
1
+ export * from "./RoleContext";
@@ -0,0 +1 @@
1
+ export * from "./useRoleTableStructure";
@@ -0,0 +1,72 @@
1
+ "use client";
2
+
3
+ import { ColumnDef, Row } from "@tanstack/react-table";
4
+ import { useTranslations } from "next-intl";
5
+ import { useMemo } from "react";
6
+ import { cellDate, cellId } from "../../../components";
7
+ import { Modules } from "../../../core";
8
+ import { TableContent, usePageUrlGenerator, UseTableStructureHook } from "../../../hooks";
9
+ import { Link } from "../../../shadcnui";
10
+ import { RoleFields, RoleInterface } from "../data";
11
+
12
+ export const useRoleTableStructure: UseTableStructureHook<RoleInterface, RoleFields> = (params) => {
13
+ const t = useTranslations();
14
+ const generateUrl = usePageUrlGenerator();
15
+
16
+ // Memoize tableData to prevent infinite re-renders
17
+ const tableData = useMemo(() => {
18
+ return params.data.map((role: RoleInterface) => {
19
+ const entry: TableContent<RoleInterface> = {
20
+ jsonApiData: role,
21
+ };
22
+ entry[RoleFields.roleId] = role.id;
23
+ params.fields.forEach((field) => {
24
+ entry[field] = role[field as keyof RoleInterface];
25
+ });
26
+ return entry;
27
+ });
28
+ }, [params.data, params.fields]);
29
+
30
+ const fieldColumnMap: Partial<Record<RoleFields, () => any>> = {
31
+ [RoleFields.roleId]: () =>
32
+ cellId({
33
+ name: "roleId",
34
+ checkedIds: params.checkedIds,
35
+ toggleId: params.toggleId,
36
+ }),
37
+ [RoleFields.name]: () => ({
38
+ id: "name",
39
+ accessorKey: "name",
40
+ header: t(`foundations.role.fields.name.label`),
41
+ cell: ({ row }: { row: Row<TableContent<RoleInterface>> }) => {
42
+ const role = row.original.jsonApiData as RoleInterface;
43
+ return <Link href={generateUrl({ page: Modules.Role, id: role.id })}>{row.getValue("name")}</Link>;
44
+ },
45
+ enableSorting: false,
46
+ enableHiding: false,
47
+ }),
48
+ [RoleFields.description]: () => ({
49
+ id: "description",
50
+ accessorKey: "description",
51
+ header: t(`foundations.role.fields.description.label`),
52
+ cell: ({ row }: { row: Row<TableContent<RoleInterface>> }) => <>{row.getValue("description")}</>,
53
+ enableSorting: false,
54
+ enableHiding: false,
55
+ }),
56
+ [RoleFields.createdAt]: () =>
57
+ cellDate({
58
+ name: "createdAt",
59
+ title: t(`generic.date.create`),
60
+ }),
61
+ };
62
+
63
+ // Memoize columns to prevent infinite re-renders
64
+ const columns = useMemo(() => {
65
+ return params.fields.map((field) => fieldColumnMap[field]?.()).filter((col) => col !== undefined) as ColumnDef<
66
+ TableContent<RoleInterface>
67
+ >[];
68
+ }, [params.fields, fieldColumnMap, t, generateUrl]);
69
+
70
+ // Memoize the return object to prevent infinite re-renders
71
+ return useMemo(() => ({ data: tableData, columns: columns }), [tableData, columns]);
72
+ };
@@ -1,2 +1,2 @@
1
- export * from "./data";
2
1
  export * from "./s3.module";
2
+ export * from "./data";
@@ -0,0 +1,23 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { useUserContext } from "../../contexts";
5
+ import { UserDetails } from "../details";
6
+
7
+ export function UserContainer() {
8
+ const { user } = useUserContext();
9
+ if (!user) return null;
10
+
11
+ const t = useTranslations();
12
+
13
+ return (
14
+ <div className="flex w-full gap-x-4">
15
+ <div className="w-2xl flex h-[calc(100vh-theme(spacing.20))] flex-col justify-between border-r pr-4">
16
+ <div className="flex h-full overflow-y-auto">
17
+ <UserDetails user={user} />
18
+ </div>
19
+ </div>
20
+ <div className="flex w-full flex-col gap-y-4"></div>
21
+ </div>
22
+ );
23
+ }
@@ -0,0 +1,12 @@
1
+ "use client";
2
+
3
+ import { PageContentContainer } from "../../../../components";
4
+ import { useUserContext } from "../../contexts";
5
+ import { UserIndexDetails } from "../details";
6
+
7
+ export function UserIndexContainer() {
8
+ const { user } = useUserContext();
9
+ if (!user) return null;
10
+
11
+ return <PageContentContainer details={<UserIndexDetails />} />;
12
+ }
@@ -0,0 +1,36 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { CompanyUsersList, Tab, TabsContainer } from "../../../../components";
5
+ import { Modules } from "../../../../core";
6
+ import { Action } from "../../../../permissions";
7
+ import { useCurrentUserContext } from "../../contexts";
8
+ import { UserInterface } from "../../data";
9
+
10
+ function UsersListContainerInternal() {
11
+ const { hasPermissionToModule } = useCurrentUserContext<UserInterface>();
12
+ const t = useTranslations();
13
+
14
+ if (!hasPermissionToModule({ module: Modules.User, action: Action.Delete })) return <CompanyUsersList />;
15
+
16
+ const tabs: Tab[] = [
17
+ {
18
+ label: t(`types.users`, { count: 2 }),
19
+ content: <CompanyUsersList />,
20
+ modules: [Modules.Company],
21
+ action: Action.Read,
22
+ },
23
+ {
24
+ label: t(`foundations.user.deleted`),
25
+ content: <CompanyUsersList isDeleted={true} />,
26
+ modules: [Modules.Company],
27
+ action: Action.Update,
28
+ },
29
+ ];
30
+
31
+ return <TabsContainer tabs={tabs} />;
32
+ }
33
+
34
+ export function UsersListContainer() {
35
+ return <UsersListContainerInternal />;
36
+ }