@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,30 @@
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 { UserFields, UserInterface, UserService } from "../../data";
8
+
9
+ type RelevantUsersListProps = {
10
+ id: string;
11
+ };
12
+
13
+ export function RelevantUsersList({ id }: RelevantUsersListProps) {
14
+ const t = useTranslations();
15
+
16
+ const data: DataListRetriever<UserInterface> = useDataListRetriever({
17
+ module: Modules.User,
18
+ retriever: (params) => UserService.findRelevant(params),
19
+ retrieverParams: { id: id },
20
+ }) as DataListRetriever<UserInterface>;
21
+
22
+ return (
23
+ <ContentListTable
24
+ data={data}
25
+ fields={[UserFields.name, UserFields.email, UserFields.relevance]}
26
+ tableGeneratorType={Modules.User}
27
+ title={t(`generic.relevant_users`)}
28
+ />
29
+ );
30
+ }
@@ -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 { RoleInterface } from "../../../role";
8
+ import { UserFields, UserInterface, UserService } from "../../data";
9
+
10
+ type RoleUsersListProps = {
11
+ role: RoleInterface;
12
+ };
13
+
14
+ export function RoleUsersList({ role }: RoleUsersListProps) {
15
+ const t = useTranslations();
16
+
17
+ const data: DataListRetriever<UserInterface> = useDataListRetriever({
18
+ retriever: (params) => UserService.findAllUsersByRole(params),
19
+ retrieverParams: { roleId: role.id },
20
+ module: Modules.User,
21
+ }) as DataListRetriever<UserInterface>;
22
+
23
+ return (
24
+ <ContentListTable
25
+ data={data}
26
+ fields={[UserFields.name, UserFields.email]}
27
+ tableGeneratorType={Modules.User}
28
+ title={t(`types.users`, { count: 2 })}
29
+ />
30
+ );
31
+ }
@@ -0,0 +1,53 @@
1
+ "use client";
2
+
3
+ import React from "react";
4
+ import { DataListRetriever } from "../../../../hooks";
5
+ import { CommandItem } from "../../../../shadcnui";
6
+ import { UserInterface } from "../../data";
7
+ import { UserAvatar } from "../widgets";
8
+
9
+ type UserListInAddProps = {
10
+ data: DataListRetriever<UserInterface>;
11
+ existingUsers: UserInterface[] | null;
12
+ setSelectedUser: (user: UserInterface) => void;
13
+ setLevelOpen?: (open: boolean) => void;
14
+ };
15
+
16
+ export function UserListInAdd({ data, existingUsers, setSelectedUser, setLevelOpen }: UserListInAddProps) {
17
+ return (
18
+ <>
19
+ {data.data !== undefined &&
20
+ (data.data as UserInterface[])
21
+ .filter(
22
+ (user: UserInterface) =>
23
+ existingUsers && !existingUsers.find((existingUser: UserInterface) => existingUser.id === user.id),
24
+ )
25
+ .map((user: UserInterface) => {
26
+ return (
27
+ <React.Fragment key={user.id}>
28
+ <CommandItem
29
+ className="cursor-pointer"
30
+ key={user.id}
31
+ onClick={(e) => {
32
+ setSelectedUser(user);
33
+ setLevelOpen?.(true);
34
+ }}
35
+ onSelect={(e) => {
36
+ setSelectedUser(user);
37
+ setLevelOpen?.(true);
38
+ }}
39
+ >
40
+ <div className="flex w-full flex-row items-center justify-between px-4 py-1">
41
+ <UserAvatar user={user} />
42
+ <div className="ml-5 flex w-full flex-col">
43
+ <div className="text-sm font-medium">{user.name}</div>
44
+ <div className="text-xs font-normal">{user.email}</div>
45
+ </div>
46
+ </div>
47
+ </CommandItem>
48
+ </React.Fragment>
49
+ );
50
+ })}
51
+ </>
52
+ );
53
+ }
@@ -0,0 +1,30 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { ReactElement } from "react";
5
+ import { ContentListTable } from "../../../../components";
6
+ import { Modules } from "../../../../core";
7
+ import { DataListRetriever } from "../../../../hooks";
8
+ import { UserFields, UserInterface } from "../../data";
9
+
10
+ type UsersListProps = {
11
+ data: DataListRetriever<UserInterface>;
12
+ optionComponents?: ReactElement<any>[];
13
+ removeFunction?: (user: UserInterface) => Promise<void>;
14
+ hideOptions?: boolean;
15
+ showRelevance?: boolean;
16
+ restrictToJoinRequests?: boolean;
17
+ };
18
+
19
+ export function UsersList({ data, optionComponents, removeFunction, hideOptions, showRelevance }: UsersListProps) {
20
+ const t = useTranslations();
21
+
22
+ return (
23
+ <ContentListTable
24
+ data={data}
25
+ fields={[UserFields.name, UserFields.email]}
26
+ tableGeneratorType={Modules.User}
27
+ title={t(`types.users`, { count: 2 })}
28
+ />
29
+ );
30
+ }
@@ -0,0 +1,30 @@
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 { UserFields, UserInterface, UserService } from "../../data";
8
+
9
+ type UsersListByContentIdsProps = {
10
+ contentIds: string[];
11
+ };
12
+
13
+ export function UsersListByContentIds({ contentIds }: UsersListByContentIdsProps) {
14
+ const t = useTranslations();
15
+
16
+ const data: DataListRetriever<UserInterface> = useDataListRetriever({
17
+ module: Modules.User,
18
+ retriever: (params) => UserService.findManyByContentIds(params),
19
+ retrieverParams: { contentIds: contentIds },
20
+ });
21
+
22
+ return (
23
+ <ContentListTable
24
+ data={data}
25
+ fields={[UserFields.name, UserFields.email]}
26
+ tableGeneratorType={Modules.User}
27
+ title={t(`generic.relevant_users`)}
28
+ />
29
+ );
30
+ }
@@ -1 +1,8 @@
1
+ export * from "./AdminUsersList";
2
+ export * from "./CompanyUsersList";
1
3
  export * from "./ContributorsList";
4
+ export * from "./RelevantUsersList";
5
+ export * from "./RoleUsersList";
6
+ export * from "./UserListInAdd";
7
+ export * from "./UsersList";
8
+ export * from "./UsersListByContentIds";
@@ -0,0 +1,31 @@
1
+ "use client";
2
+
3
+ import { Modules } from "../../../../core";
4
+ import { usePageUrlGenerator } from "../../../../hooks";
5
+ import { Link } from "../../../../shadcnui";
6
+ import { UserInterface } from "../../data";
7
+ import { UserAvatar } from "./UserAvatar";
8
+
9
+ type UserAvatarListProps = {
10
+ users: UserInterface[];
11
+ };
12
+
13
+ export function UserAvatarList({ users }: UserAvatarListProps) {
14
+ const generateUrl = usePageUrlGenerator();
15
+
16
+ return (
17
+ <div className="flex flex-row items-center">
18
+ <div className="flex flex-row-reverse justify-end -space-x-1 space-x-reverse">
19
+ {users.map((user: UserInterface) => (
20
+ <Link
21
+ key={user.id}
22
+ href={generateUrl({ page: Modules.User, id: user.id })}
23
+ onClick={(e) => e.stopPropagation()}
24
+ >
25
+ <UserAvatar user={user} className="h-5 w-5" />
26
+ </Link>
27
+ ))}
28
+ </div>
29
+ </div>
30
+ );
31
+ }
@@ -0,0 +1,89 @@
1
+ "use client";
2
+
3
+ import { RefreshCwIcon, SearchIcon, XIcon } from "lucide-react";
4
+ import { useTranslations } from "next-intl";
5
+ import { ReactNode, useEffect, useState } from "react";
6
+ import {
7
+ Command,
8
+ CommandItem,
9
+ CommandList,
10
+ Input,
11
+ Popover,
12
+ PopoverContent,
13
+ PopoverTrigger,
14
+ } from "../../../../shadcnui";
15
+ import { useUserSearch } from "../../hooks";
16
+ import { UserAvatar } from "./UserAvatar";
17
+
18
+ type UserSearchPopoverProps = {
19
+ children: ReactNode;
20
+ onSelect: (userId: string) => void;
21
+ align?: "start" | "center" | "end";
22
+ className?: string;
23
+ };
24
+
25
+ export const UserSearchPopover = ({ children, onSelect, align = "start", className }: UserSearchPopoverProps) => {
26
+ const t = useTranslations();
27
+ const [isOpen, setIsOpen] = useState(false);
28
+ const { users, searchQuery, setSearchQuery, isLoading, loadUsers, clearSearch } = useUserSearch();
29
+
30
+ useEffect(() => {
31
+ if (isOpen && users.length === 0 && !searchQuery) {
32
+ loadUsers("");
33
+ }
34
+ }, [isOpen, users.length, searchQuery, loadUsers]);
35
+
36
+ const handleSelectUser = (userId: string) => {
37
+ onSelect(userId);
38
+ setIsOpen(false);
39
+ clearSearch();
40
+ };
41
+
42
+ return (
43
+ <Popover open={isOpen} onOpenChange={setIsOpen}>
44
+ <PopoverTrigger asChild>{children}</PopoverTrigger>
45
+ <PopoverContent align={align} onClick={(e) => e.stopPropagation()} className={className ?? "w-80"}>
46
+ <Command shouldFilter={false}>
47
+ <div className="relative mb-2 w-full">
48
+ <SearchIcon className="text-muted-foreground absolute top-2.5 left-2.5 h-4 w-4" />
49
+ <Input
50
+ placeholder={t(`generic.search.placeholder`, { type: t(`types.users`, { count: 1 }) })}
51
+ type="text"
52
+ className="w-full pr-8 pl-8"
53
+ onChange={(e) => setSearchQuery(e.target.value)}
54
+ value={searchQuery}
55
+ onClick={(e) => e.stopPropagation()}
56
+ />
57
+ {isLoading ? (
58
+ <RefreshCwIcon className="text-muted-foreground absolute top-2.5 right-2.5 h-4 w-4 animate-spin" />
59
+ ) : searchQuery ? (
60
+ <XIcon
61
+ className="text-muted-foreground hover:text-foreground absolute top-2.5 right-2.5 h-4 w-4 cursor-pointer"
62
+ onClick={(e) => {
63
+ e.stopPropagation();
64
+ clearSearch();
65
+ }}
66
+ />
67
+ ) : null}
68
+ </div>
69
+ <CommandList>
70
+ {users.length > 0 ? (
71
+ users.map((user) => (
72
+ <CommandItem key={user.id} className="cursor-pointer" onSelect={() => handleSelectUser(user.id)}>
73
+ <UserAvatar user={user} className="mr-2 h-4 w-4" showLink={false} />
74
+ <span>{user.name}</span>
75
+ </CommandItem>
76
+ ))
77
+ ) : (
78
+ <div className="text-muted-foreground py-6 text-center text-sm">
79
+ {isLoading
80
+ ? t(`generic.loading`)
81
+ : t(`generic.search.no_results`, { type: t(`types.users`, { count: 1 }) })}
82
+ </div>
83
+ )}
84
+ </CommandList>
85
+ </Command>
86
+ </PopoverContent>
87
+ </Popover>
88
+ );
89
+ };
@@ -0,0 +1,106 @@
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 { Action } from "../../../permissions";
10
+ import { UserDeleter, UserEditor, UserReactivator, UserResentInvitationEmail } from "../components/forms";
11
+ import { UserInterface } from "../data";
12
+ import { useCurrentUserContext } from "./CurrentUserContext";
13
+
14
+ interface UserContextType {
15
+ user: UserInterface | undefined;
16
+ setUser: (value: UserInterface | undefined) => void;
17
+ }
18
+
19
+ const UserContext = createContext<UserContextType | undefined>(undefined);
20
+
21
+ type UserProviderProps = {
22
+ children: ReactNode;
23
+ dehydratedUser?: JsonApiHydratedDataInterface;
24
+ };
25
+
26
+ export const UserProvider = ({ children, dehydratedUser }: UserProviderProps) => {
27
+ const generateUrl = usePageUrlGenerator();
28
+ const { hasPermissionToModule } = useCurrentUserContext<UserInterface>();
29
+ const t = useTranslations();
30
+
31
+ const [user, setUser] = useState<UserInterface | undefined>(
32
+ dehydratedUser ? rehydrate<UserInterface>(Modules.User, dehydratedUser) : undefined,
33
+ );
34
+
35
+ const breadcrumb = () => {
36
+ const response: BreadcrumbItemData[] = [];
37
+
38
+ if (hasPermissionToModule({ module: Modules.User, action: Action.Update })) {
39
+ response.push({
40
+ name: t(`generic.settings`),
41
+ href: generateUrl({ page: `/settings` }),
42
+ });
43
+
44
+ response.push({
45
+ name: t(`types.users`, { count: 2 }),
46
+ href: generateUrl({ page: `/settings`, id: Modules.User.pageUrl }),
47
+ });
48
+ }
49
+
50
+ if (user)
51
+ response.push({
52
+ name: `${user.name}${user.isDeleted ? ` (${t(`foundations.user.deleted`)})` : ""}`,
53
+ href: generateUrl({ page: Modules.User, id: user.id }),
54
+ });
55
+
56
+ return response;
57
+ };
58
+
59
+ const title = () => {
60
+ const response: any = {
61
+ type: t(`types.users`, { count: user ? 1 : 2 }),
62
+ };
63
+
64
+ const functions: ReactNode[] = [];
65
+
66
+ if (user) {
67
+ response.element = `${user.name}${user.isDeleted ? ` (${t(`foundations.user.deleted`)})` : ""}`;
68
+
69
+ if (user.isDeleted) {
70
+ functions.push(<UserReactivator key={`UserReactivator`} user={user} propagateChanges={setUser} />);
71
+ } else {
72
+ if (!user.isActivated)
73
+ functions.push(<UserResentInvitationEmail key={`UserResentInvitationEmail`} user={user} />);
74
+
75
+ functions.push(<UserDeleter key={`UserDeleter`} user={user} />);
76
+ }
77
+
78
+ functions.push(<UserEditor key={`UserEditor`} user={user} propagateChanges={setUser} />);
79
+ }
80
+
81
+ if (functions.length > 0) response.functions = functions;
82
+
83
+ return response;
84
+ };
85
+
86
+ return (
87
+ <SharedProvider value={{ breadcrumbs: breadcrumb(), title: title() }}>
88
+ <UserContext.Provider
89
+ value={{
90
+ user: user,
91
+ setUser: setUser,
92
+ }}
93
+ >
94
+ {children}
95
+ </UserContext.Provider>
96
+ </SharedProvider>
97
+ );
98
+ };
99
+
100
+ export const useUserContext = (): UserContextType => {
101
+ const context = useContext(UserContext);
102
+ if (context === undefined) {
103
+ throw new Error("useUserContext must be used within a UserProvider");
104
+ }
105
+ return context;
106
+ };
@@ -1 +1,2 @@
1
1
  export * from "./CurrentUserContext";
2
+ export * from "./UserContext";
@@ -0,0 +1,2 @@
1
+ export * from "./useUserSearch";
2
+ export * from "./useUserTableStructure";
@@ -0,0 +1,53 @@
1
+ "use client";
2
+
3
+ import { useCallback, useEffect, useRef, useState } from "react";
4
+ import { useDebounce } from "../../../hooks/useDebounce";
5
+ import { UserInterface, UserService } from "../data";
6
+
7
+ export const useUserSearch = () => {
8
+ const [users, setUsers] = useState<UserInterface[]>([]);
9
+ const [searchQuery, setSearchQuery] = useState("");
10
+ const [isLoading, setIsLoading] = useState(false);
11
+ const searchQueryRef = useRef("");
12
+
13
+ const loadUsers = useCallback(
14
+ async (search: string) => {
15
+ try {
16
+ if (search === searchQueryRef.current && users.length > 0) return;
17
+ setIsLoading(true);
18
+ searchQueryRef.current = search;
19
+ const fetchedUsers = await UserService.findMany({ search, fetchAll: true });
20
+ setUsers(fetchedUsers);
21
+ } catch (error) {
22
+ console.error("Failed to load users:", error);
23
+ } finally {
24
+ setIsLoading(false);
25
+ }
26
+ },
27
+ [users.length],
28
+ );
29
+
30
+ const debouncedLoadUsers = useDebounce(loadUsers, 500);
31
+
32
+ useEffect(() => {
33
+ if (searchQuery !== searchQueryRef.current) {
34
+ setIsLoading(true);
35
+ debouncedLoadUsers(searchQuery);
36
+ }
37
+ }, [searchQuery, debouncedLoadUsers]);
38
+
39
+ const clearSearch = () => {
40
+ setSearchQuery("");
41
+ searchQueryRef.current = "";
42
+ };
43
+
44
+ return {
45
+ users,
46
+ searchQuery,
47
+ setSearchQuery,
48
+ isLoading,
49
+ loadUsers,
50
+ clearSearch,
51
+ searchQueryRef,
52
+ };
53
+ };
@@ -0,0 +1,115 @@
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, UserAvatar } from "../../../components";
7
+ import { Modules } from "../../../core";
8
+ import { TableContent, UseTableStructureHook } from "../../../hooks/types";
9
+ import { usePageUrlGenerator } from "../../../hooks/usePageUrlGenerator";
10
+ import { Link } from "../../../shadcnui";
11
+ import { cn } from "../../../utils";
12
+ import { UserFields, UserInterface } from "../data";
13
+
14
+ export const useUserTableStructure: UseTableStructureHook<UserInterface, UserFields> = (params) => {
15
+ const t = useTranslations();
16
+ const generateUrl = usePageUrlGenerator();
17
+
18
+ // Memoize tableData to prevent infinite re-renders
19
+ const tableData = useMemo(() => {
20
+ return params.data.map((user: UserInterface) => {
21
+ const entry: TableContent<UserInterface> = {
22
+ jsonApiData: user,
23
+ };
24
+ entry[UserFields.userId] = user.id;
25
+ params.fields.forEach((field) => {
26
+ entry[field] = user[field as keyof UserInterface];
27
+ });
28
+ return entry;
29
+ });
30
+ }, [params.data, params.fields]);
31
+
32
+ const fieldColumnMap: Partial<Record<UserFields, () => any>> = {
33
+ [UserFields.userId]: () =>
34
+ cellId({
35
+ name: "userId",
36
+ checkedIds: params.checkedIds,
37
+ toggleId: params.toggleId,
38
+ }),
39
+ [UserFields.name]: () => ({
40
+ id: "name",
41
+ accessorKey: "name",
42
+ header: t(`foundations.user.fields.name.label`),
43
+ cell: ({ row }: { row: Row<TableContent<UserInterface>> }) => {
44
+ const user = row.original.jsonApiData as UserInterface;
45
+ return (
46
+ <Link
47
+ href={generateUrl({ page: Modules.User, id: user.id })}
48
+ className={cn(
49
+ `flex items-center justify-start gap-2`,
50
+ user.isDeleted || !user.isActivated ? "text-muted-foreground italic" : "",
51
+ )}
52
+ >
53
+ <UserAvatar user={user} />
54
+ {user.name}
55
+ {user.isDeleted ? ` - ${t("foundations.user.errors.deleted")}` : ""}
56
+ {!user.isActivated ? ` - ${t("foundations.user.errors.inactive")}` : ""}
57
+ </Link>
58
+ );
59
+ },
60
+ enableSorting: false,
61
+ enableHiding: false,
62
+ }),
63
+ [UserFields.email]: () => ({
64
+ id: "email",
65
+ accessorKey: "email",
66
+ header: t(`generic.fields.email.label`),
67
+ cell: ({ row }: { row: Row<TableContent<UserInterface>> }) => <>{row.getValue("email")}</>,
68
+ enableSorting: false,
69
+ enableHiding: false,
70
+ }),
71
+ [UserFields.relevance]: () => ({
72
+ id: "relevance",
73
+ accessorKey: "relevance",
74
+ header: t(`generic.relevance`),
75
+ cell: ({ row }: { row: TableContent<UserInterface> }) => {
76
+ const user: UserInterface = row.original.jsonApiData;
77
+
78
+ if (!user.relevance) return <></>;
79
+
80
+ const response = `${user.relevance.toFixed(0)}%`;
81
+
82
+ return (
83
+ <div className="relative flex h-5 w-20 items-center justify-center overflow-hidden rounded border text-center">
84
+ <div
85
+ className={`bg-accent absolute top-0 left-0 h-full opacity-${Math.round(user.relevance)}`}
86
+ style={{ width: `${user.relevance}%` }}
87
+ ></div>
88
+ <span
89
+ className={`relative text-xs ${user.relevance < 40 ? "text-muted-foreground" : "text-accent-foreground font-semibold"}`}
90
+ >
91
+ {response}
92
+ </span>
93
+ </div>
94
+ );
95
+ },
96
+ enableSorting: false,
97
+ enableHiding: false,
98
+ }),
99
+ [UserFields.createdAt]: () =>
100
+ cellDate({
101
+ name: "createdAt",
102
+ title: t(`generic.date.create`),
103
+ }),
104
+ };
105
+
106
+ // Memoize columns to prevent infinite re-renders
107
+ const columns = useMemo(() => {
108
+ return params.fields.map((field) => fieldColumnMap[field]?.()).filter((col) => col !== undefined) as ColumnDef<
109
+ TableContent<UserInterface>
110
+ >[];
111
+ }, [params.fields, fieldColumnMap, t, generateUrl]);
112
+
113
+ // Memoize the return object to prevent infinite re-renders
114
+ return useMemo(() => ({ data: tableData, columns: columns }), [tableData, columns]);
115
+ };
@@ -1,4 +1,3 @@
1
1
  export * from "./author.module";
2
2
  // Note: contexts are exported from @carlonicora/nextjs-jsonapi/contexts to avoid "use client" issues
3
- export * from "./components";
4
3
  export * from "./data";
@@ -31,3 +31,7 @@ export type {
31
31
  UseTranslationsHook,
32
32
  } from "../i18n";
33
33
  export * from "./useCustomD3Graph";
34
+ export * from "./useNotificationSync";
35
+ export * from "./usePageTracker";
36
+ export * from "./usePushNotifications";
37
+ export * from "./useSocket";
@@ -1,3 +1,5 @@
1
+ "use client";
2
+
1
3
  import * as d3 from "d3";
2
4
  import { Loader2 } from "lucide-react";
3
5
  import { useCallback, useEffect, useMemo, useRef } from "react";
@@ -0,0 +1,20 @@
1
+ "use client";
2
+
3
+ import { useEffect } from "react";
4
+ import { useNotificationContext, useSocketContext } from "../contexts";
5
+
6
+ export function useNotificationSync() {
7
+ const { socketNotifications, clearSocketNotifications } = useSocketContext();
8
+ const { addSocketNotifications } = useNotificationContext();
9
+
10
+ useEffect(() => {
11
+ if (socketNotifications.length > 0) {
12
+ try {
13
+ addSocketNotifications(socketNotifications);
14
+ clearSocketNotifications();
15
+ } catch (error) {
16
+ console.error("💥 [useNotificationSync] Error processing notifications:", error);
17
+ }
18
+ }
19
+ }, [socketNotifications, addSocketNotifications, clearSocketNotifications]);
20
+ }